lino 25.3.2__py3-none-any.whl → 25.3.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,13 +1,58 @@
1
1
  # -*- coding: UTF-8 -*-
2
- # Copyright 2023 Rumma & Ko Ltd.
2
+ # Copyright 2009-2023 Rumma & Ko Ltd.
3
3
  # License: GNU Affero General Public License v3 (see file COPYING for details)
4
4
 
5
- from django.core.management.base import BaseCommand
5
+ from click import confirm
6
+ from django.core.management.base import BaseCommand, CommandError
7
+ from django.core.management import call_command
6
8
  from django.conf import settings
9
+ from lino import logger
7
10
 
8
11
 
9
12
  class Command(BaseCommand):
13
+ """Build the site cache files and run collectstatic for this Lino site."""
14
+
15
+ def add_arguments(self, parser):
16
+ super().add_arguments(parser)
17
+ parser.add_argument(
18
+ "-b", "--batch", "--noinput",
19
+ action="store_false",
20
+ dest="interactive",
21
+ default=True,
22
+ help="Do not prompt for input of any kind.",
23
+ ),
24
+
10
25
  def handle(self, *args, **options):
26
+ interactive = options.get("interactive")
11
27
  verbosity = options.get("verbosity")
12
- # print("20231005", verbosity)
28
+ project_dir = settings.SITE.project_dir
29
+
30
+ options = dict(interactive=False, verbosity=verbosity)
31
+
32
+ if interactive:
33
+ msg = "Build everything for ({})".format(project_dir)
34
+ msg += ".\nAre you sure?"
35
+ if not confirm(msg, default=True):
36
+ raise CommandError("User abort.")
37
+
38
+ # the following log message was useful on Travis 20150104
39
+ if verbosity > 0:
40
+ logger.info("`buildsite` started on %s.", project_dir)
41
+
42
+ # pth = project_dir / "settings.py"
43
+ # if pth.exists():
44
+ # pth.touch()
45
+
46
+ call_command("collectstatic", **options)
47
+
13
48
  settings.SITE.build_site_cache(force=True, verbosity=verbosity)
49
+
50
+ # if settings.SITE.is_installed("help"):
51
+ # call_command("makehelp", verbosity=verbosity)
52
+
53
+ # for p in settings.SITE.installed_plugins:
54
+ # p.on_buildsite(settings.SITE, verbosity=verbosity)
55
+
56
+ # settings.SITE.clear_site_config()
57
+
58
+ logger.info("`buildsite` finished on %s.", project_dir)
@@ -2,6 +2,10 @@
2
2
  # Copyright 2013-2023 by Rumma & Ko Ltd.
3
3
  # License: GNU Affero General Public License v3 (see file COPYING for details)
4
4
 
5
+ from lino.utils.mldbc.fields import BabelCharField, BabelTextField
6
+ from lino.core.choicelists import ChoiceListField
7
+ from lino.core.utils import sorted_models_list, full_model_name
8
+ from lino.utils import puts
5
9
  from io import open
6
10
  from lino import logger
7
11
 
@@ -20,12 +24,6 @@ from datetime import timezone
20
24
 
21
25
  utc = timezone.utc
22
26
 
23
- from lino.utils import puts
24
- from lino.core.utils import sorted_models_list, full_model_name
25
- from lino.core.choicelists import ChoiceListField
26
-
27
- from lino.utils.mldbc.fields import BabelCharField, BabelTextField
28
-
29
27
 
30
28
  def is_pointer_to_contenttype(f):
31
29
  if not settings.SITE.is_installed("contenttypes"):
@@ -402,7 +400,7 @@ def main(args):
402
400
  if __name__ == '__main__':
403
401
  import argparse
404
402
  parser = argparse.ArgumentParser(description='Restore the data.')
405
- parser.add_argument('--noinput', dest='interactive',
403
+ parser.add_argument('-b', '--noinput', '--batch', dest='interactive',
406
404
  action='store_false', default=True,
407
405
  help="Don't ask for confirmation before flushing the database.")
408
406
  parser.add_argument('--quick', dest='quick',
@@ -4,7 +4,7 @@
4
4
 
5
5
  import sys
6
6
  import subprocess
7
- from rstgen.utils import confirm
7
+ from click import confirm
8
8
  from django.core.management.base import BaseCommand
9
9
  from django.conf import settings
10
10
 
@@ -54,5 +54,5 @@ class Command(BaseCommand):
54
54
  return
55
55
  # cmd = "pip install --upgrade --trusted-host svn.forge.pallavi.be {}".format(' '.join(reqs))
56
56
  cmd = sys.executable + " -m pip install --upgrade pip {}".format(" ".join(reqs))
57
- if not options["interactive"] or confirm("{} (y/n) ?".format(cmd)):
57
+ if not options["interactive"] or confirm("{} ?".format(cmd), default=True):
58
58
  runcmd(cmd)
@@ -1,15 +1,11 @@
1
1
  # -*- coding: UTF-8 -*-
2
- # Copyright 2013-2023 Rumma & Ko Ltd
2
+ # Copyright 2013-2025 Rumma & Ko Ltd
3
3
  # License: GNU Affero General Public License v3 (see file COPYING for details)
4
4
 
5
- from pathlib import Path
6
5
  from django.core.management import call_command
7
6
  from django.core.management.base import BaseCommand, CommandError
8
7
  from django.conf import settings
9
- from django.db import DEFAULT_DB_ALIAS
10
- from rstgen.utils import confirm
11
- # from lino.management.commands.initdb import Command as BaseCommand
12
- # from lino.management.commands.initdb import CommandError
8
+ from lino.utils import dbhash
13
9
 
14
10
 
15
11
  class Command(BaseCommand):
@@ -19,7 +15,7 @@ class Command(BaseCommand):
19
15
  super().add_arguments(parser)
20
16
  (
21
17
  parser.add_argument(
22
- "--noinput",
18
+ "-b", "--batch", "--noinput",
23
19
  action="store_false",
24
20
  dest="interactive",
25
21
  default=True,
@@ -66,4 +62,4 @@ class Command(BaseCommand):
66
62
  kwargs["removemedia"] = True
67
63
  call_command("initdb", *args, **kwargs)
68
64
 
69
- settings.SITE.mark_virgin()
65
+ dbhash.mark_virgin()
lino/mixins/duplicable.py CHANGED
@@ -29,7 +29,7 @@ class Duplicate(actions.Action):
29
29
  sort_index = 11
30
30
  show_in_workflow = False
31
31
  # readonly = False # like ShowInsert. See docs/blog/2012/0726
32
- callable_from = "t"
32
+ callable_from = "td"
33
33
 
34
34
  # required_roles = set([Expert])
35
35
 
@@ -1,5 +1,5 @@
1
1
  # -*- coding: UTF-8 -*-
2
- # Copyright 2015-2023 Rumma & Ko Ltd
2
+ # Copyright 2015-2025 Rumma & Ko Ltd
3
3
  # License: GNU Affero General Public License v3 (see file COPYING for details)
4
4
 
5
5
  from collections import OrderedDict
@@ -9,18 +9,17 @@ from django.conf import settings
9
9
  from django.utils import translation
10
10
  from django.template.defaultfilters import pluralize
11
11
 
12
+ from lino import logger
13
+ from lino.core import constants
12
14
  from lino.core.gfks import gfk2lookup
13
15
  from lino.modlib.gfks.mixins import Controllable
14
16
  from lino.modlib.users.mixins import UserAuthored
15
17
  from lino.modlib.linod.choicelists import background_task
16
18
  from lino.core.roles import SiteStaff
17
-
18
19
  from lino.api import dd, rt, _
19
20
 
20
21
  from .choicelists import Checker, Checkers
21
-
22
22
  from .roles import CheckdataUser
23
- from lino.core import constants
24
23
 
25
24
 
26
25
  class CheckerAction(dd.Action):
@@ -32,6 +31,8 @@ class CheckerAction(dd.Action):
32
31
  Message = rt.models.checkdata.Message
33
32
  gfk = Message.owner
34
33
  for obj in objects:
34
+ if checkers is None:
35
+ checkers = get_checkers_for(obj.__class__)
35
36
  qs = Message.objects.filter(**gfk2lookup(gfk, obj))
36
37
  qs.delete()
37
38
  for chk in checkers:
@@ -84,12 +85,12 @@ class UpdateMessagesByController(CheckerAction):
84
85
  combo_group = "checkdata"
85
86
  required_roles = dd.login_required()
86
87
 
87
- def __init__(self, model):
88
- self.model = model
89
- super().__init__()
88
+ # def __init__(self, model=None, **kwargs):
89
+ # self.model = model
90
+ # super().__init__(**kwargs)
90
91
 
91
92
  def run_from_ui(self, ar, fix=None):
92
- self.run_it(ar, fix, get_checkers_for(self.model), ar.selected_rows)
93
+ self.run_it(ar, fix, None, ar.selected_rows)
93
94
  # if fix is None:
94
95
  # fix = self.fix_them
95
96
  # Message = rt.models.checkdata.Message
@@ -109,6 +110,16 @@ class FixMessagesByController(UpdateMessagesByController):
109
110
  fix_them = True
110
111
 
111
112
 
113
+ class QuickFixMessagesByController(UpdateMessagesByController):
114
+ # label = _("Fix data problems")
115
+ fix_them = True
116
+ combo_group = None
117
+ # icon_name = "lightning"
118
+ icon_name = None
119
+ button_text = ' ⚡ ' # 26A1
120
+ # button_text = "✓" # u"\u2713"
121
+
122
+
112
123
  class FixAllProblems(CheckerAction):
113
124
  select_rows = False
114
125
  show_in_plain = True
@@ -119,7 +130,7 @@ class FixAllProblems(CheckerAction):
119
130
 
120
131
  def run_from_ui(self, ar, fix=None):
121
132
  mi = ar.master_instance
122
- print(f"20250307 {mi}")
133
+ # print(f"20250307 {mi}")
123
134
  self.run_it(ar, fix, get_checkers_for(mi.__class__), [mi])
124
135
  ar.set_response(refresh=True)
125
136
 
@@ -264,11 +275,14 @@ def set_checkdata_actions(sender, **kw):
264
275
  if m is None:
265
276
  continue
266
277
  assert m is not Message
267
- m.define_action(check_data=UpdateMessagesByController(m))
268
- m.define_action(fix_problems=FixMessagesByController(m))
269
- if True:
270
- # don't add it automatically because appdev might prefer
271
- # to show it in a detail_layout:
278
+ # if hasattr(m, 'check_data'):
279
+ if (label := getattr(m, 'quickfix_checkdata_label', None)):
280
+ # print(f"20250324 Customized quickfix_checkdata_label {label} for {m}")
281
+ m.define_action(quick_fix=QuickFixMessagesByController(label=label))
282
+ else:
283
+ # print(f"20250324 Default checkdata buttons for {m}")
284
+ m.define_action(check_data=UpdateMessagesByController())
285
+ m.define_action(fix_problems=FixMessagesByController())
272
286
  m.define_action(
273
287
  show_problems=dd.ShowSlaveTable(
274
288
  MessagesByOwner,
@@ -280,7 +294,11 @@ def set_checkdata_actions(sender, **kw):
280
294
 
281
295
 
282
296
  def get_checkers_for(model):
283
- return get_checkable_models()[model]
297
+ checkers = []
298
+ for m, lst in get_checkable_models().items():
299
+ if m is not None and issubclass(model, m):
300
+ checkers += lst
301
+ return checkers
284
302
 
285
303
 
286
304
  def check_instance(obj, **kwargs):
@@ -4,6 +4,7 @@
4
4
  Adds some demo comments.
5
5
 
6
6
  """
7
+
7
8
  import datetime
8
9
  from lino.utils import i2t
9
10
  from lino.utils import Cycler
@@ -32,12 +33,6 @@ plain1 = "Some plain text."
32
33
  plain2 = "Two paragraphs of plain text.\n\nThe second paragraph."
33
34
  # plain2 += " With an 👁 (U+1F441)." #5855 (Jane fails to store certain unicode characters)
34
35
 
35
- imageDataURL = """"""
36
- body_with_img = f"""\
37
- <p>Here is an image:</p>
38
- <p><img src="{imageDataURL}" class="bar"></p>\
39
- """
40
-
41
36
  BODIES = Cycler(
42
37
  [styled, table, lorem, short_lorem, breaking, cond_comment, plain1, plain2]
43
38
  )
@@ -45,9 +40,6 @@ BODIES = Cycler(
45
40
  BODIES.items.insert(0, "")
46
41
  BODIES.items.insert(0, "")
47
42
 
48
- if dd.is_installed('blogs'):
49
- BODIES.items.append(body_with_img)
50
-
51
43
 
52
44
  def objects():
53
45
  Comment = rt.models.comments.Comment
@@ -354,7 +354,7 @@ class ApiElement(View):
354
354
  ar = ba.create_request(
355
355
  request=request, renderer=settings.SITE.kernel.default_renderer)
356
356
  if pk and pk != "-99999" and pk != "-99998":
357
- if issubclass(rpt.model, models.Model):
357
+ if rpt.model is not None and issubclass(rpt.model, models.Model):
358
358
  try:
359
359
  ar.set_selected_pks(pk)
360
360
  except rpt.model.DoesNotExist:
lino/modlib/users/ui.py CHANGED
@@ -46,8 +46,8 @@ class UserDetail(dd.DetailLayout):
46
46
  """
47
47
 
48
48
  main = """
49
- box1 box2 #MembershipsByUser:20
50
- remarks:40 AuthoritiesGiven:20 SocialAuthsByUser:30
49
+ box1 box2 #MembershipsByUser:20 remarks
50
+ AuthoritiesGiven:20 AuthoritiesTaken:20 SocialAuthsByUser:30
51
51
  """
52
52
 
53
53
  # main_m = """
@@ -203,6 +203,7 @@ class AuthoritiesGiven(Authorities):
203
203
  label = _("Authorities given")
204
204
  column_names = "authorized"
205
205
  auto_fit_column_widths = True
206
+ details_of_master_template = _("%(details)s by %(master)s")
206
207
 
207
208
 
208
209
  class AuthoritiesTaken(Authorities):
@@ -211,6 +212,7 @@ class AuthoritiesTaken(Authorities):
211
212
  label = _("Authorities taken")
212
213
  column_names = "user"
213
214
  auto_fit_column_widths = True
215
+ details_of_master_template = _("%(details)s by %(master)s")
214
216
 
215
217
 
216
218
  if has_socialauth and dd.get_plugin_setting("users", "third_party_authentication"):
@@ -1,4 +1,4 @@
1
- # Copyright 2016-2021 Rumma & Ko Ltd
1
+ # Copyright 2016-2025 Rumma & Ko Ltd
2
2
  # License: GNU Affero General Public License v3 (see file COPYING for details)
3
3
  """This plugins installs two build methods for generating
4
4
  :term:`printable documents <printable document>` using `weasyprint
@@ -25,31 +25,17 @@ from lino.api import ad, _
25
25
 
26
26
 
27
27
  class Plugin(ad.Plugin):
28
- "See :doc:`/dev/plugins`."
29
28
 
30
29
  verbose_name = _("WeasyPrint")
31
-
32
30
  needs_plugins = ["lino.modlib.jinja"]
33
31
 
34
32
  header_height = 20
35
- """Height of header in mm. Set to `None` if you want no header."""
36
-
37
33
  footer_height = 20
38
- """Height of footer in mm. Set to `None` if you want no header."""
39
-
40
34
  top_right_width = None
41
- """Width of top-right.jpg in mm. If not given, Lino computes it based on height.
42
- """
43
-
35
+ page_background_image = None
44
36
  top_right_image = None
45
- """The first image file found in config named either top-right.jpg or top-right.png."""
46
-
47
37
  header_image = None
48
- """The first image file found in config named either header.jpg or header.png."""
49
-
50
38
  margin = 10
51
- """Top and bottom page margin in mm."""
52
-
53
39
  margin_left = 17
54
40
  margin_right = 10
55
41
 
@@ -57,8 +43,8 @@ class Plugin(ad.Plugin):
57
43
  yield "imagesize"
58
44
 
59
45
  def pre_site_startup(self, site):
60
- if self.header_height:
61
- for ext in ("jpg", "png"):
46
+ for ext in ("jpg", "png"):
47
+ if self.header_height:
62
48
  fn = site.confdirs.find_config_file("top-right." + ext, "weasyprint")
63
49
  if fn:
64
50
  self.top_right_image = fn
@@ -70,6 +56,12 @@ class Plugin(ad.Plugin):
70
56
  self.top_right_width = self.header_height * w / h
71
57
  fn = site.confdirs.find_config_file("header." + ext, "weasyprint")
72
58
  if fn:
59
+ # site.logger.info("Found header_image %s", fn)
73
60
  self.header_image = fn
74
-
75
- super(Plugin, self).pre_site_startup(site)
61
+ if self.page_background_image is None:
62
+ fn = site.confdirs.find_config_file(
63
+ "page-background." + ext, "weasyprint")
64
+ if fn:
65
+ # site.logger.info("Found page_background_image %s", fn)
66
+ self.page_background_image = fn
67
+ super().pre_site_startup(site)
@@ -53,7 +53,7 @@ p {
53
53
  }
54
54
 
55
55
  div.recipient {
56
- position:relative; left:80mm;
56
+ position:relative; left:{{100-dd.plugins.weasyprint.margin_left}}mm;
57
57
  height:30mm;
58
58
  width:80mm;
59
59
  border: 1px solid grey;
@@ -80,6 +80,15 @@ div.recipient {
80
80
  {%- endif -%}
81
81
  margin-left: {{dd.plugins.weasyprint.margin_left}}mm;
82
82
  margin-right: {{dd.plugins.weasyprint.margin_right}}mm;
83
+ {%- if dd.plugins.weasyprint.page_background_image -%}
84
+ {#
85
+ background: url(file://{{dd.plugins.weasyprint.page_background_image}}) no-repeat center center fixed;
86
+ #}
87
+ background-image: url(file://{{dd.plugins.weasyprint.page_background_image}});
88
+ background-repeat: no-repeat;
89
+ background-attachment: fixed;
90
+ background-size: contain;
91
+ {%- endif -%}
83
92
  font-family: "Liberation sans", "arial";
84
93
  font-size: 10pt;
85
94
  {%- block bottomleft %}
lino/utils/__init__.py CHANGED
@@ -18,6 +18,7 @@ function for general use. It has also many subpackages and submodules.
18
18
  dataserializer
19
19
  dbfreader
20
20
  dblogger
21
+ dbhash
21
22
  diag
22
23
  djangotest
23
24
  dpy
lino/utils/dbhash.py ADDED
@@ -0,0 +1,112 @@
1
+ # -*- coding: UTF-8 -*-
2
+ # Copyright 2009-2025 Rumma & Ko Ltd
3
+ # License: GNU Affero General Public License v3 (see file COPYING for details)
4
+
5
+ """
6
+ Utilities around a "database hash".
7
+ See :doc:`/utils/dbhash`.
8
+ """
9
+
10
+ import json
11
+ from importlib import import_module
12
+ from pathlib import Path
13
+ from django.conf import settings
14
+ from django.apps import apps
15
+ from django.db.models.deletion import ProtectedError
16
+
17
+ mod = import_module(settings.SETTINGS_MODULE)
18
+ HASH_FILE = Path(mod.__file__).parent / "dbhash.json"
19
+
20
+
21
+ def fmn(m):
22
+ return f"{m._meta.app_label}.{m._meta.object_name}"
23
+
24
+
25
+ def compute_dbhash():
26
+ """
27
+ Return a dictionary with a hash value of the current database content.
28
+ """
29
+ rv = dict()
30
+ for m in apps.get_models(include_auto_created=True):
31
+ k = fmn(m)
32
+ if k != "sessions.Session":
33
+ # rv[k] = m.objects.count()
34
+ rv[k] = list(m.objects.values_list('pk', flat=True))
35
+ return rv
36
+
37
+
38
+ def mark_virgin():
39
+ """
40
+ Mark the database as virgin. This is called by :manage:`prep`.
41
+ """
42
+ dbhash = compute_dbhash()
43
+ with HASH_FILE.open("w") as fp:
44
+ json.dump(dbhash, fp)
45
+
46
+
47
+ def load_dbhash():
48
+ """
49
+ Load the dbhash that was saved in :xfile:`dbhash.json`
50
+ """
51
+ if not HASH_FILE.exists():
52
+ raise Exception(
53
+ f"No file {HASH_FILE} (did you run `django-admin prep`?)")
54
+ with HASH_FILE.open("r") as fp:
55
+ return json.load(fp)
56
+
57
+
58
+ def check_virgin(restore=True, verbose=True):
59
+ """
60
+ Verify whether the database is virgin. Print the differences if there
61
+ are any.
62
+ """
63
+ new = compute_dbhash()
64
+ old = load_dbhash()
65
+
66
+ first_diff = True
67
+ can_restore = True
68
+ must_delete = {}
69
+ for k, v in new.items():
70
+ v = set(v)
71
+ oldv = set(old.get(k, None))
72
+ if oldv != v:
73
+ if first_diff:
74
+ if verbose:
75
+ print(f"Database {HASH_FILE.parent} isn't virgin:")
76
+ first_diff = False
77
+ diffs = []
78
+ added = v - oldv
79
+ if len(added):
80
+ diffs.append(f"{len(added)} rows added")
81
+ must_delete[apps.get_model(k)] = added
82
+ # for pk in added:
83
+ # must_delete.append(m.objects.get(pk=pk))
84
+ if (removed := len(oldv-v)):
85
+ diffs.append(f"{removed} rows deleted")
86
+ can_restore = False
87
+ if verbose:
88
+ print(f"- {k}: {', '.join(diffs)}")
89
+ if len(must_delete) == 0 or not restore:
90
+ return
91
+ if not can_restore:
92
+ raise Exception(
93
+ "Cannot restore database because some rows have been deleted")
94
+ # print(f"Tidy up {len(must_delete)} rows from database")
95
+ # It can happen that some rows refer to each other with a protected fk
96
+ # We call bulk delete() to avoid Lino deleting the items of an invoice
97
+ must_delete = list(must_delete.items())
98
+ while len(must_delete):
99
+ todo = []
100
+ hope = False
101
+ for m, added in must_delete:
102
+ try:
103
+ m.objects.filter(pk__in=added).delete()
104
+ # obj.delete()
105
+ hope = True
106
+ except Exception:
107
+ todo.append((m, added))
108
+ if not hope:
109
+ raise Exception(f"Failed to delete {todo}")
110
+ must_delete = todo
111
+ if verbose:
112
+ print("Database has been restored.")
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: lino
3
- Version: 25.3.2
3
+ Version: 25.3.4
4
4
  Summary: A framework for writing desktop-like web applications using Django and ExtJS or React
5
5
  Project-URL: Homepage, https://www.lino-framework.org
6
6
  Project-URL: Repository, https://gitlab.com/lino-framework/lino