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.
- lino/__init__.py +1 -1
- lino/api/doctest.py +1 -0
- lino/core/site.py +9 -76
- lino/locale/bn/LC_MESSAGES/django.po +1112 -907
- lino/locale/de/LC_MESSAGES/django.mo +0 -0
- lino/locale/de/LC_MESSAGES/django.po +1066 -918
- lino/locale/django.pot +1041 -900
- lino/locale/es/LC_MESSAGES/django.po +1103 -902
- lino/locale/et/LC_MESSAGES/django.po +1103 -915
- lino/locale/fr/LC_MESSAGES/django.po +1105 -916
- lino/locale/nl/LC_MESSAGES/django.po +1102 -917
- lino/locale/pt_BR/LC_MESSAGES/django.po +1099 -900
- lino/locale/zh_Hant/LC_MESSAGES/django.po +1099 -900
- lino/management/commands/buildcache.py +48 -3
- lino/management/commands/dump2py.py +5 -7
- lino/management/commands/install.py +2 -2
- lino/management/commands/prep.py +4 -8
- lino/mixins/duplicable.py +1 -1
- lino/modlib/checkdata/models.py +33 -15
- lino/modlib/comments/fixtures/demo2.py +1 -9
- lino/modlib/extjs/views.py +1 -1
- lino/modlib/users/ui.py +4 -2
- lino/modlib/weasyprint/__init__.py +12 -20
- lino/modlib/weasyprint/config/weasyprint/base.weasy.html +10 -1
- lino/utils/__init__.py +1 -0
- lino/utils/dbhash.py +112 -0
- {lino-25.3.2.dist-info → lino-25.3.4.dist-info}/METADATA +1 -1
- {lino-25.3.2.dist-info → lino-25.3.4.dist-info}/RECORD +31 -31
- lino/management/commands/buildsite.py +0 -57
- {lino-25.3.2.dist-info → lino-25.3.4.dist-info}/WHEEL +0 -0
- {lino-25.3.2.dist-info → lino-25.3.4.dist-info}/licenses/AUTHORS.rst +0 -0
- {lino-25.3.2.dist-info → lino-25.3.4.dist-info}/licenses/COPYING +0 -0
@@ -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
|
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
|
-
|
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
|
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("{}
|
57
|
+
if not options["interactive"] or confirm("{} ?".format(cmd), default=True):
|
58
58
|
runcmd(cmd)
|
lino/management/commands/prep.py
CHANGED
@@ -1,15 +1,11 @@
|
|
1
1
|
# -*- coding: UTF-8 -*-
|
2
|
-
# Copyright 2013-
|
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
|
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
|
-
|
65
|
+
dbhash.mark_virgin()
|
lino/mixins/duplicable.py
CHANGED
lino/modlib/checkdata/models.py
CHANGED
@@ -1,5 +1,5 @@
|
|
1
1
|
# -*- coding: UTF-8 -*-
|
2
|
-
# Copyright 2015-
|
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
|
-
|
89
|
-
|
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,
|
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
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
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
|
-
|
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
|
lino/modlib/extjs/views.py
CHANGED
@@ -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
|
-
|
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-
|
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
|
-
|
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
|
-
|
61
|
-
|
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
|
-
|
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:
|
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
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.
|
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
|