lino 25.2.3__py3-none-any.whl → 25.3.1__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/dd.py +11 -48
- lino/api/doctest.py +34 -36
- lino/core/actions.py +25 -23
- lino/core/actors.py +37 -17
- lino/core/choicelists.py +10 -8
- lino/core/dbtables.py +1 -1
- lino/core/elems.py +46 -30
- lino/core/fields.py +19 -9
- lino/core/inject.py +7 -6
- lino/core/kernel.py +26 -66
- lino/core/model.py +44 -31
- lino/core/plugin.py +4 -4
- lino/core/requests.py +76 -55
- lino/core/site.py +84 -30
- lino/core/store.py +5 -2
- lino/core/utils.py +12 -7
- lino/help_texts.py +3 -8
- lino/management/commands/prep.py +1 -1
- lino/mixins/duplicable.py +6 -4
- lino/mixins/sequenced.py +17 -6
- lino/modlib/__init__.py +0 -2
- lino/modlib/changes/models.py +21 -10
- lino/modlib/checkdata/models.py +59 -24
- lino/modlib/comments/fixtures/demo2.py +12 -3
- lino/modlib/comments/models.py +7 -7
- lino/modlib/comments/ui.py +8 -5
- lino/modlib/export_excel/models.py +7 -5
- lino/modlib/extjs/views.py +39 -20
- lino/modlib/help/management/commands/makehelp.py +5 -2
- lino/modlib/jinja/mixins.py +25 -14
- lino/modlib/linod/__init__.py +1 -0
- lino/modlib/linod/choicelists.py +21 -0
- lino/modlib/linod/consumers.py +13 -4
- lino/modlib/linod/management/commands/linod.py +6 -2
- lino/modlib/linod/mixins.py +16 -11
- lino/modlib/linod/models.py +4 -2
- lino/modlib/notify/models.py +18 -10
- lino/modlib/printing/actions.py +41 -30
- lino/modlib/printing/choicelists.py +11 -9
- lino/modlib/printing/mixins.py +25 -20
- lino/modlib/publisher/models.py +5 -5
- lino/modlib/summaries/models.py +3 -2
- lino/modlib/system/models.py +28 -29
- lino/modlib/uploads/__init__.py +5 -5
- lino/modlib/uploads/actions.py +2 -8
- lino/modlib/uploads/choicelists.py +10 -10
- lino/modlib/uploads/fixtures/std.py +17 -0
- lino/modlib/uploads/mixins.py +20 -8
- lino/modlib/uploads/models.py +60 -35
- lino/modlib/uploads/ui.py +10 -7
- lino/utils/media.py +45 -23
- lino/utils/report.py +5 -4
- lino/utils/soup.py +22 -4
- {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/METADATA +1 -1
- {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/RECORD +59 -80
- lino/mixins/uploadable.py +0 -3
- lino/sandbox/bcss/PerformInvestigation.py +0 -2260
- lino/sandbox/bcss/SSDNReply.py +0 -3924
- lino/sandbox/bcss/SSDNRequest.py +0 -3723
- lino/sandbox/bcss/__init__.py +0 -0
- lino/sandbox/bcss/readme.txt +0 -1
- lino/sandbox/bcss/test.py +0 -92
- lino/sandbox/bcss/test2.py +0 -128
- lino/sandbox/bcss/test3.py +0 -161
- lino/sandbox/bcss/test4.py +0 -167
- lino/sandbox/contacts/__init__.py +0 -0
- lino/sandbox/contacts/fixtures/__init__.py +0 -0
- lino/sandbox/contacts/fixtures/demo.py +0 -365
- lino/sandbox/contacts/manage.py +0 -10
- lino/sandbox/contacts/models.py +0 -395
- lino/sandbox/contacts/settings.py +0 -67
- lino/sandbox/tx25/XSD/RetrieveTIGroupsV3.wsdl +0 -65
- lino/sandbox/tx25/XSD/RetrieveTIGroupsV3.xsd +0 -286
- lino/sandbox/tx25/XSD/rn25_Release201104.xsd +0 -2855
- lino/sandbox/tx25/xsd2py1.py +0 -68
- lino/sandbox/tx25/xsd2py2.py +0 -62
- lino/sandbox/tx25/xsd2py3.py +0 -56
- {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/WHEEL +0 -0
- {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/licenses/AUTHORS.rst +0 -0
- {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/licenses/COPYING +0 -0
lino/modlib/checkdata/models.py
CHANGED
@@ -12,6 +12,7 @@ from django.template.defaultfilters import pluralize
|
|
12
12
|
from lino.core.gfks import gfk2lookup
|
13
13
|
from lino.modlib.gfks.mixins import Controllable
|
14
14
|
from lino.modlib.users.mixins import UserAuthored
|
15
|
+
from lino.modlib.linod.choicelists import background_task
|
15
16
|
from lino.core.roles import SiteStaff
|
16
17
|
|
17
18
|
from lino.api import dd, rt, _
|
@@ -22,12 +23,27 @@ from .roles import CheckdataUser
|
|
22
23
|
from lino.core import constants
|
23
24
|
|
24
25
|
|
25
|
-
class
|
26
|
+
class CheckerAction(dd.Action):
|
27
|
+
fix_them = False
|
28
|
+
|
29
|
+
def run_it(self, ar, fix, checkers, objects):
|
30
|
+
if fix is None:
|
31
|
+
fix = self.fix_them
|
32
|
+
Message = rt.models.checkdata.Message
|
33
|
+
gfk = Message.owner
|
34
|
+
for obj in objects:
|
35
|
+
qs = Message.objects.filter(**gfk2lookup(gfk, obj))
|
36
|
+
qs.delete()
|
37
|
+
for chk in checkers:
|
38
|
+
chk.update_problems(ar, obj, False, fix)
|
39
|
+
ar.set_response(refresh=True)
|
40
|
+
|
41
|
+
|
42
|
+
class UpdateMessage(CheckerAction):
|
26
43
|
icon_name = "bell"
|
27
44
|
ui5_icon_name = "sap-icon://bell"
|
28
45
|
label = _("Check data")
|
29
46
|
combo_group = "checkdata"
|
30
|
-
fix_them = False
|
31
47
|
sort_index = 90
|
32
48
|
# custom_handler = True
|
33
49
|
# select_rows = False
|
@@ -47,11 +63,11 @@ class UpdateMessage(dd.Action):
|
|
47
63
|
# has been deleted.
|
48
64
|
obj.delete()
|
49
65
|
else:
|
50
|
-
|
51
|
-
|
52
|
-
)
|
53
|
-
qs.delete()
|
54
|
-
chk.update_problems(ar, owner, False, fix)
|
66
|
+
self.run_it(ar, fix, [chk], [owner])
|
67
|
+
# qs = Message.objects.filter(
|
68
|
+
# **gfk2lookup(Message.owner, owner, checker=chk))
|
69
|
+
# qs.delete()
|
70
|
+
# chk.update_problems(ar, owner, False, fix)
|
55
71
|
ar.set_response(refresh_all=True)
|
56
72
|
|
57
73
|
|
@@ -61,12 +77,11 @@ class FixProblem(UpdateMessage):
|
|
61
77
|
sort_index = 91
|
62
78
|
|
63
79
|
|
64
|
-
class UpdateMessagesByController(
|
80
|
+
class UpdateMessagesByController(CheckerAction):
|
65
81
|
icon_name = "bell"
|
66
82
|
ui5_icon_name = "sap-icon://bell"
|
67
83
|
label = _("Check data")
|
68
84
|
combo_group = "checkdata"
|
69
|
-
fix_them = False
|
70
85
|
required_roles = dd.login_required()
|
71
86
|
|
72
87
|
def __init__(self, model):
|
@@ -74,18 +89,19 @@ class UpdateMessagesByController(dd.Action):
|
|
74
89
|
super().__init__()
|
75
90
|
|
76
91
|
def run_from_ui(self, ar, fix=None):
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
92
|
+
self.run_it(ar, fix, get_checkers_for(self.model), ar.selected_rows)
|
93
|
+
# if fix is None:
|
94
|
+
# fix = self.fix_them
|
95
|
+
# Message = rt.models.checkdata.Message
|
96
|
+
# gfk = Message.owner
|
97
|
+
# checkers = get_checkers_for(self.model)
|
98
|
+
# for obj in ar.selected_rows:
|
99
|
+
# assert isinstance(obj, self.model)
|
100
|
+
# qs = Message.objects.filter(**gfk2lookup(gfk, obj))
|
101
|
+
# qs.delete()
|
102
|
+
# for chk in checkers:
|
103
|
+
# chk.update_problems(ar, obj, False, fix)
|
104
|
+
# ar.set_response(refresh=True)
|
89
105
|
|
90
106
|
|
91
107
|
class FixMessagesByController(UpdateMessagesByController):
|
@@ -93,6 +109,21 @@ class FixMessagesByController(UpdateMessagesByController):
|
|
93
109
|
fix_them = True
|
94
110
|
|
95
111
|
|
112
|
+
class FixAllProblems(CheckerAction):
|
113
|
+
select_rows = False
|
114
|
+
show_in_plain = True
|
115
|
+
# http_method = "POST"
|
116
|
+
label = _("Fix all data problems")
|
117
|
+
button_text = "✓" # u"\u2713"
|
118
|
+
fix_them = True
|
119
|
+
|
120
|
+
def run_from_ui(self, ar, fix=None):
|
121
|
+
mi = ar.master_instance
|
122
|
+
print(f"20250307 {mi}")
|
123
|
+
self.run_it(ar, fix, get_checkers_for(mi.__class__), [mi])
|
124
|
+
ar.set_response(refresh=True)
|
125
|
+
|
126
|
+
|
96
127
|
class Message(Controllable, UserAuthored):
|
97
128
|
class Meta(object):
|
98
129
|
app_label = "checkdata"
|
@@ -182,6 +213,8 @@ class MessagesByOwner(Messages):
|
|
182
213
|
column_names = "message checker user #fixable *"
|
183
214
|
default_display_modes = {None: constants.DISPLAY_MODE_SUMMARY}
|
184
215
|
|
216
|
+
fix_all_problems = FixAllProblems()
|
217
|
+
|
185
218
|
|
186
219
|
# This was the first use case of a slave table with something else than a model
|
187
220
|
# instance as its master
|
@@ -305,8 +338,10 @@ def check_data(ar, args=[], fix=True, prune=False):
|
|
305
338
|
assert not m._meta.abstract
|
306
339
|
if settings.SITE.is_hidden_plugin(m._meta.app_label):
|
307
340
|
continue
|
308
|
-
ct = rt.models.contenttypes.ContentType.objects.get_for_model(
|
309
|
-
|
341
|
+
ct = rt.models.contenttypes.ContentType.objects.get_for_model(
|
342
|
+
m)
|
343
|
+
qs = Message.objects.filter(
|
344
|
+
owner_type=ct, checker__in=checkers)
|
310
345
|
qs.delete()
|
311
346
|
name = str(m._meta.verbose_name_plural)
|
312
347
|
qs = m.objects.all()
|
@@ -346,7 +381,7 @@ def check_data(ar, args=[], fix=True, prune=False):
|
|
346
381
|
ar.logger.info(msg, done, what, found, fixed)
|
347
382
|
|
348
383
|
|
349
|
-
@
|
384
|
+
@background_task(every_unit="daily", every=1)
|
350
385
|
def checkdata(ar):
|
351
386
|
"""Run all data checkers."""
|
352
387
|
check_data(ar, fix=False)
|
@@ -32,9 +32,21 @@ plain1 = "Some plain text."
|
|
32
32
|
plain2 = "Two paragraphs of plain text.\n\nThe second paragraph."
|
33
33
|
# plain2 += " With an 👁 (U+1F441)." #5855 (Jane fails to store certain unicode characters)
|
34
34
|
|
35
|
+
imageDataURL = """"""
|
36
|
+
body_with_img = f"""\
|
37
|
+
<p>Here is an image:</p>
|
38
|
+
<p><img src="{imageDataURL}" class="bar"></p>\
|
39
|
+
"""
|
40
|
+
|
35
41
|
BODIES = Cycler(
|
36
42
|
[styled, table, lorem, short_lorem, breaking, cond_comment, plain1, plain2]
|
37
43
|
)
|
44
|
+
# Add two empty bodies that will be filled later:
|
45
|
+
BODIES.items.insert(0, "")
|
46
|
+
BODIES.items.insert(0, "")
|
47
|
+
|
48
|
+
if dd.is_installed('blogs'):
|
49
|
+
BODIES.items.append(body_with_img)
|
38
50
|
|
39
51
|
|
40
52
|
def objects():
|
@@ -44,9 +56,6 @@ def objects():
|
|
44
56
|
# use_linod = settings.SITE.use_linod
|
45
57
|
# settings.SITE.use_linod = False
|
46
58
|
|
47
|
-
# Add two empty bodies that will be filled later:
|
48
|
-
BODIES.items.insert(0, "")
|
49
|
-
BODIES.items.insert(0, "")
|
50
59
|
MENTIONED = Cycler()
|
51
60
|
for model in rt.models_by_base(Commentable):
|
52
61
|
if model.memo_command is not None:
|
lino/modlib/comments/models.py
CHANGED
@@ -3,6 +3,8 @@
|
|
3
3
|
# License: GNU Affero General Public License v3 (see file COPYING for details)
|
4
4
|
|
5
5
|
# from html import escape
|
6
|
+
from .ui import *
|
7
|
+
from lino.modlib.checkdata.choicelists import Checker
|
6
8
|
from django.contrib.humanize.templatetags.humanize import naturaltime
|
7
9
|
from django.db import models
|
8
10
|
from django.db.models import Q
|
@@ -10,7 +12,7 @@ from django.core import validators
|
|
10
12
|
from django.utils.html import mark_safe, format_html, SafeString
|
11
13
|
from django.contrib.contenttypes.models import ContentType
|
12
14
|
from django.conf import settings
|
13
|
-
from lino.utils.html import E, tostring, fromstring
|
15
|
+
# from lino.utils.html import E, tostring, fromstring
|
14
16
|
|
15
17
|
from lino.api import dd, rt, _
|
16
18
|
|
@@ -144,7 +146,8 @@ class Comment(
|
|
144
146
|
u = ar.get_user()
|
145
147
|
if u.is_anonymous:
|
146
148
|
return
|
147
|
-
mr = rt.models.comments.Reaction.objects.filter(
|
149
|
+
mr = rt.models.comments.Reaction.objects.filter(
|
150
|
+
user=u, comment=self).first()
|
148
151
|
if mr:
|
149
152
|
return mr.emotion
|
150
153
|
|
@@ -399,7 +402,8 @@ class Reaction(CreatedModified, UserAuthored, DateRangeObservable):
|
|
399
402
|
|
400
403
|
allow_cascaded_delete = "user comment"
|
401
404
|
|
402
|
-
comment = dd.ForeignKey(
|
405
|
+
comment = dd.ForeignKey(
|
406
|
+
"comments.Comment", related_name="reactions_to_this")
|
403
407
|
emotion = Emotions.field(default="ok")
|
404
408
|
|
405
409
|
def as_summary_item(self, ar, text=None, **kwargs):
|
@@ -418,8 +422,6 @@ class Reaction(CreatedModified, UserAuthored, DateRangeObservable):
|
|
418
422
|
#
|
419
423
|
# mp.register_django_model('comment', Comment)
|
420
424
|
|
421
|
-
from lino.modlib.checkdata.choicelists import Checker
|
422
|
-
|
423
425
|
|
424
426
|
class CommentChecker(Checker):
|
425
427
|
# temporary checker to fix #4084 (Comment.owner is empty when replying to a comment)
|
@@ -436,5 +438,3 @@ class CommentChecker(Checker):
|
|
436
438
|
|
437
439
|
|
438
440
|
CommentChecker.activate()
|
439
|
-
|
440
|
-
from .ui import *
|
lino/modlib/comments/ui.py
CHANGED
@@ -2,6 +2,7 @@
|
|
2
2
|
# Copyright 2013-2023 Rumma & Ko Ltd
|
3
3
|
# License: GNU Affero General Public License v3 (see file COPYING for details)
|
4
4
|
|
5
|
+
from lino.modlib.publisher.choicelists import PageFillers
|
5
6
|
from django.utils.translation import ngettext
|
6
7
|
from django.contrib.humanize.templatetags.humanize import naturaltime
|
7
8
|
from django.contrib.contenttypes.models import ContentType
|
@@ -125,7 +126,8 @@ class Comments(dd.Table):
|
|
125
126
|
|
126
127
|
@classmethod
|
127
128
|
def comments_created(cls, user, sd, ed):
|
128
|
-
pv = dict(user=user, start_date=sd, end_date=ed,
|
129
|
+
pv = dict(user=user, start_date=sd, end_date=ed,
|
130
|
+
observed_event=CommentEvents.created)
|
129
131
|
# pv = dict(start_date=sd, end_date=ed, observed_event=CommentEvents.created)
|
130
132
|
return cls.request(user=user, param_values=pv)
|
131
133
|
|
@@ -206,6 +208,7 @@ class CommentsByType(CommentsByX):
|
|
206
208
|
master_key = "comment_type"
|
207
209
|
column_names = "body created user *"
|
208
210
|
|
211
|
+
|
209
212
|
# TODO: rename CommentsByRFC to CommentsByOwner
|
210
213
|
class CommentsByRFC(CommentsByX):
|
211
214
|
master_key = "owner"
|
@@ -232,7 +235,8 @@ class CommentsByRFC(CommentsByX):
|
|
232
235
|
@classmethod
|
233
236
|
def param_defaults(cls, ar, **kw):
|
234
237
|
kw = super().param_defaults(ar, **kw)
|
235
|
-
|
238
|
+
if ar.display_mode == constants.DISPLAY_MODE_STORY:
|
239
|
+
kw["reply_to"] = constants.CHOICES_BLANK_FILTER_VALUE
|
236
240
|
return kw
|
237
241
|
|
238
242
|
# @classmethod
|
@@ -275,7 +279,8 @@ class CommentsByMentioned(CommentsByX):
|
|
275
279
|
assert not cls.model._meta.abstract
|
276
280
|
ct = ContentType.objects.get_for_model(cls.model)
|
277
281
|
mkw = gfk2lookup(Mention.target, mi, owner_type=ct)
|
278
|
-
mentions = Mention.objects.filter(
|
282
|
+
mentions = Mention.objects.filter(
|
283
|
+
**mkw).values_list("owner_id", flat=True)
|
279
284
|
# mentions = [o.comment_id for o in Mention.objects.filter(**mkw)]
|
280
285
|
# print(mkw, mentions)
|
281
286
|
# return super(CommentsByMentioned, cls).get_filter_kw(ar, **kw)
|
@@ -312,6 +317,4 @@ class ReactionsByComment(Reactions):
|
|
312
317
|
default_display_modes = {None: constants.DISPLAY_MODE_SUMMARY}
|
313
318
|
|
314
319
|
|
315
|
-
from lino.modlib.publisher.choicelists import PageFillers
|
316
|
-
|
317
320
|
PageFillers.add_item(RecentComments)
|
@@ -86,7 +86,8 @@ def ar2workbook(ar, column_names=None):
|
|
86
86
|
time = time.strip("-")
|
87
87
|
negative = True
|
88
88
|
time = time.split(":")
|
89
|
-
value = datetime.timedelta(
|
89
|
+
value = datetime.timedelta(
|
90
|
+
hours=int(time[0]), minutes=int(time[1]))
|
90
91
|
if negative: # Make negative.
|
91
92
|
value = value - value - value
|
92
93
|
elif iselement(value) or isinstance(value, SafeString):
|
@@ -111,7 +112,8 @@ def ar2workbook(ar, column_names=None):
|
|
111
112
|
cell.style = style
|
112
113
|
cell.value = value
|
113
114
|
except ValueError as e:
|
114
|
-
raise Exception("20190222 {} {}".format(
|
115
|
+
raise Exception("20190222 {} {}".format(
|
116
|
+
value.__class__, value))
|
115
117
|
|
116
118
|
return workbook
|
117
119
|
|
@@ -131,14 +133,14 @@ class ExportExcelAction(actions.Action):
|
|
131
133
|
def run_from_ui(self, ar, **kw):
|
132
134
|
# Prepare tmp file
|
133
135
|
mf = TmpMediaFile(ar, "xlsx")
|
134
|
-
settings.SITE.makedirs_if_missing(
|
136
|
+
settings.SITE.makedirs_if_missing(mf.path.parent)
|
135
137
|
|
136
138
|
# Render
|
137
|
-
self.render(ar, mf.
|
139
|
+
self.render(ar, mf.path)
|
138
140
|
|
139
141
|
# Tell client that the action was successful and that it
|
140
142
|
# should open a new browser window on the generated file.
|
141
|
-
ar.success(open_url=mf.
|
143
|
+
ar.success(open_url=mf.url)
|
142
144
|
|
143
145
|
def render(self, ar, file):
|
144
146
|
workbook = ar2workbook(ar)
|
lino/modlib/extjs/views.py
CHANGED
@@ -183,7 +183,8 @@ class SWView(TemplateView):
|
|
183
183
|
class RunJasmine(View):
|
184
184
|
def get(self, request, *args, **kw):
|
185
185
|
return http.HttpResponse(
|
186
|
-
settings.SITE.kernel.extjs_renderer.html_page(
|
186
|
+
settings.SITE.kernel.extjs_renderer.html_page(
|
187
|
+
request, run_jasmine=True)
|
187
188
|
)
|
188
189
|
|
189
190
|
|
@@ -205,7 +206,8 @@ class ActionParamChoices(View):
|
|
205
206
|
if ba is None:
|
206
207
|
raise Exception("Unknown action %r for %s" % (an, actor))
|
207
208
|
field = ba.action.get_param_elem(field)
|
208
|
-
qs, row2dict = choices_for_field(
|
209
|
+
qs, row2dict = choices_for_field(
|
210
|
+
ba.request(request=request), ba.action, field)
|
209
211
|
if field.blank:
|
210
212
|
emptyValue = "<br/>"
|
211
213
|
else:
|
@@ -315,7 +317,8 @@ class Restful(View):
|
|
315
317
|
ar.renderer = settings.SITE.kernel.extjs_renderer
|
316
318
|
ar.form2obj_and_save(data, elem, False)
|
317
319
|
# Ext.ensible needs grid_fields, not detail_fields
|
318
|
-
ar.set_response(rows=[rh.store.row2dict(
|
320
|
+
ar.set_response(rows=[rh.store.row2dict(
|
321
|
+
ar, elem, rh.store.grid_fields)])
|
319
322
|
return json_response(ar.response)
|
320
323
|
|
321
324
|
|
@@ -346,7 +349,8 @@ class ApiElement(View):
|
|
346
349
|
if ba is None:
|
347
350
|
raise http.Http404("%s has no detail_action" % rpt)
|
348
351
|
|
349
|
-
fmt = request.GET.get(constants.URL_PARAM_FORMAT,
|
352
|
+
fmt = request.GET.get(constants.URL_PARAM_FORMAT,
|
353
|
+
ba.action.default_format)
|
350
354
|
|
351
355
|
try:
|
352
356
|
if pk and pk != "-99999" and pk != "-99998":
|
@@ -355,31 +359,39 @@ class ApiElement(View):
|
|
355
359
|
try:
|
356
360
|
ar = ba.request(request=request, selected_pks=sr)
|
357
361
|
# except ObjectDoesNotExist as e: # 20250212
|
358
|
-
except rpt.model.DoesNotExist
|
362
|
+
except rpt.model.DoesNotExist:
|
359
363
|
if fmt == constants.URL_FORMAT_JSON:
|
360
364
|
# rescue_ar: without sr and even request, to render a table request (grid view action) on breadcrumb
|
361
|
-
rescue_ar = rpt.request(
|
365
|
+
rescue_ar = rpt.request(
|
366
|
+
renderer=settings.SITE.kernel.default_renderer)
|
362
367
|
default_table = rpt.model.get_default_table()
|
363
368
|
|
364
|
-
title = tostring(rescue_ar.href_to_request(
|
369
|
+
title = tostring(rescue_ar.href_to_request(
|
370
|
+
rescue_ar, icon_name=None))
|
371
|
+
|
365
372
|
def get_response():
|
366
|
-
msg = mark_safe(
|
367
|
-
|
373
|
+
msg = mark_safe(
|
374
|
+
f'Record (pk={pk}) is no longer available on current table.')
|
375
|
+
datarec = dict(
|
376
|
+
success=False, message=msg, title=title)
|
368
377
|
datarec.update(**vm)
|
369
378
|
return datarec
|
370
379
|
|
371
380
|
try:
|
372
381
|
# take default table and try to show the row
|
373
|
-
ar = default_table.detail_action.request(
|
374
|
-
|
382
|
+
ar = default_table.detail_action.request(
|
383
|
+
request=request, selected_pks=sr)
|
384
|
+
except default_table.model.DoesNotExist:
|
375
385
|
return json_response(get_response())
|
376
386
|
|
377
387
|
url = ar.obj2url(ar.selected_rows[0])
|
378
388
|
datarec = get_response()
|
379
|
-
datarec['message'] += mark_safe(
|
389
|
+
datarec['message'] += mark_safe(
|
390
|
+
f' Reload in <a href="{url}">{default_table}</a>.')
|
380
391
|
return json_response(datarec)
|
381
392
|
# print("20240911", e)
|
382
|
-
raise http.Http404(
|
393
|
+
raise http.Http404(
|
394
|
+
f"Object {sr} does not exist on {rpt}")
|
383
395
|
else:
|
384
396
|
ar = ba.request(request=request, selected_pks=sr)
|
385
397
|
# ar = ba.request(request=request, selected_pks=sr)
|
@@ -434,7 +446,8 @@ class ApiElement(View):
|
|
434
446
|
elem = ar.create_instance()
|
435
447
|
datarec = elem2rec_empty(ar, ar.ah, elem)
|
436
448
|
elif elem is None:
|
437
|
-
datarec = dict(
|
449
|
+
datarec = dict(
|
450
|
+
success=False, message=NOT_FOUND % (rpt, pk))
|
438
451
|
else:
|
439
452
|
datarec = ar.elem2rec_detailed(elem)
|
440
453
|
datarec.update(**vm)
|
@@ -525,7 +538,8 @@ class ApiList(View):
|
|
525
538
|
# Have same-origin policy work for iframe of file upload. see ticket #2885
|
526
539
|
# https://stackoverflow.com/questions/22627392/extjs-fileuplaod-cross-origin-frame
|
527
540
|
response.content = """<html><head><script type="text/javascript">document.domain="{}";</script></head><body>{}</body></html>""".format(
|
528
|
-
request.POST["_document_domain"], response.content.decode(
|
541
|
+
request.POST["_document_domain"], response.content.decode(
|
542
|
+
"utf-8")
|
529
543
|
)
|
530
544
|
return response
|
531
545
|
|
@@ -544,7 +558,8 @@ class ApiList(View):
|
|
544
558
|
# print(20170921, fmt)
|
545
559
|
|
546
560
|
if fmt == constants.URL_FORMAT_JSON:
|
547
|
-
rows = [rh.store.row2list(ar, row)
|
561
|
+
rows = [rh.store.row2list(ar, row)
|
562
|
+
for row in ar.sliced_data_iterator]
|
548
563
|
total_count = ar.get_total_count()
|
549
564
|
# raise Exception("20171208 {}".format(ar.data_iterator.query))
|
550
565
|
for row in ar.create_phantom_rows():
|
@@ -587,7 +602,8 @@ class ApiList(View):
|
|
587
602
|
sp = request.GET.get(constants.URL_PARAM_SHOW_PARAMS_PANEL, None)
|
588
603
|
if sp is not None:
|
589
604
|
# ~ after_show.update(show_params_panel=sp)
|
590
|
-
after_show.update(
|
605
|
+
after_show.update(
|
606
|
+
show_params_panel=constants.parse_boolean(sp))
|
591
607
|
|
592
608
|
# if isinstance(ar.bound_action.action, actions.ShowInsert):
|
593
609
|
# elem = ar.create_instance()
|
@@ -606,7 +622,8 @@ class ApiList(View):
|
|
606
622
|
if fmt == "csv":
|
607
623
|
# ~ response = HttpResponse(mimetype='text/csv')
|
608
624
|
charset = settings.SITE.csv_params.get("encoding", "utf-8")
|
609
|
-
response = http.HttpResponse(
|
625
|
+
response = http.HttpResponse(
|
626
|
+
content_type='text/csv;charset="%s"' % charset)
|
610
627
|
if False:
|
611
628
|
response["Content-Disposition"] = (
|
612
629
|
'attachment; filename="%s.csv"' % ar.actor
|
@@ -629,8 +646,10 @@ class ApiList(View):
|
|
629
646
|
|
630
647
|
if fmt == constants.URL_FORMAT_PRINTER:
|
631
648
|
if ar.get_total_count() > MAX_ROW_COUNT:
|
632
|
-
raise Exception(
|
633
|
-
|
649
|
+
raise Exception(
|
650
|
+
_("List contains more than %d rows") % MAX_ROW_COUNT)
|
651
|
+
response = http.HttpResponse(
|
652
|
+
content_type='text/html;charset="utf-8"')
|
634
653
|
doc = Document(force_str(ar.get_title()))
|
635
654
|
doc.body.append(E.h1(doc.title))
|
636
655
|
t = doc.add_table()
|
@@ -42,7 +42,10 @@ from lino.core.utils import model_class_path
|
|
42
42
|
from lino.modlib.help.utils import HelpTextsLoader, simplify_name
|
43
43
|
from lino.modlib.gfks.fields import GenericForeignKey
|
44
44
|
from lino.api.dd import full_model_name
|
45
|
-
|
45
|
+
|
46
|
+
# removed import doctest because it caused "pytest not installed" during
|
47
|
+
# makehelp on LF:
|
48
|
+
# from lino.api import doctest
|
46
49
|
|
47
50
|
use_dirhtml = False
|
48
51
|
|
@@ -255,7 +258,7 @@ class Command(GeneratingCommand):
|
|
255
258
|
settings=settings,
|
256
259
|
actors=actors,
|
257
260
|
# actors_list=[a for a in actors.actors_list if not a.abstract],
|
258
|
-
doctest=doctest,
|
261
|
+
# doctest=doctest,
|
259
262
|
translation=translation,
|
260
263
|
use_dirhtml=use_dirhtml,
|
261
264
|
include_useless=include_useless,
|
lino/modlib/jinja/mixins.py
CHANGED
@@ -1,8 +1,9 @@
|
|
1
1
|
# -*- coding: UTF-8 -*-
|
2
|
-
# Copyright 2022 Rumma & Ko Ltd
|
2
|
+
# Copyright 2022-2025 Rumma & Ko Ltd
|
3
3
|
# License: GNU Affero General Public License v3 (see file COPYING for details)
|
4
4
|
|
5
5
|
import os
|
6
|
+
import base64
|
6
7
|
from pathlib import Path
|
7
8
|
from lxml import etree
|
8
9
|
|
@@ -12,6 +13,8 @@ from django.utils.html import mark_safe, escape
|
|
12
13
|
|
13
14
|
from lino.api import dd
|
14
15
|
from lino.utils.xml import validate_xml
|
16
|
+
from lino.utils.media import MediaFile
|
17
|
+
|
15
18
|
|
16
19
|
def xml_element(name, value):
|
17
20
|
if value:
|
@@ -26,37 +29,45 @@ class XMLMaker(dd.Model):
|
|
26
29
|
|
27
30
|
xml_validator_file = None
|
28
31
|
xml_file_template = None
|
29
|
-
xml_file_name = None
|
32
|
+
# xml_file_name = None
|
33
|
+
|
34
|
+
def get_xml_file_parts(self):
|
35
|
+
yield 'xml'
|
36
|
+
yield self.get_printable_target_stem() + ".xml"
|
30
37
|
|
31
38
|
def make_xml_file(self, ar):
|
32
39
|
renderer = settings.SITE.plugins.jinja.renderer
|
33
40
|
tpl = renderer.jinja_env.get_template(self.xml_file_template)
|
34
41
|
context = self.get_printable_context(ar)
|
35
42
|
context.update(xml_element=xml_element)
|
43
|
+
context.update(base64=base64)
|
36
44
|
xml = tpl.render(**context)
|
37
|
-
parts = [
|
38
|
-
|
39
|
-
|
40
|
-
xmlfile =
|
41
|
-
|
42
|
-
|
43
|
-
xmlfile.
|
45
|
+
# parts = [
|
46
|
+
# dd.plugins.accounting.xml_media_dir,
|
47
|
+
# self.xml_file_name.format(self=self)]
|
48
|
+
xmlfile = MediaFile(False, *self.get_xml_file_parts())
|
49
|
+
# xmlfile = Path(settings.MEDIA_ROOT, *parts)
|
50
|
+
ar.logger.info("Make %s from %s ...", xmlfile.path, self)
|
51
|
+
xmlfile.path.parent.mkdir(exist_ok=True, parents=True)
|
52
|
+
xmlfile.path.write_text(xml)
|
44
53
|
# xmlfile.write_text(etree.tostring(xml))
|
45
54
|
|
46
55
|
if self.xml_validator_file:
|
47
56
|
# print("20250218 {xml[:100]}")
|
48
57
|
# doc = etree.fromstring(xml.encode("utf-8"))
|
49
|
-
ar.logger.info("Validate %s against %s ...",
|
58
|
+
# ar.logger.info("Validate %s against %s ...",
|
59
|
+
# xmlfile.path.name, self.xml_validator_file)
|
50
60
|
if True:
|
51
|
-
validate_xml(xmlfile, self.xml_validator_file)
|
61
|
+
validate_xml(xmlfile.path, self.xml_validator_file)
|
52
62
|
else:
|
53
63
|
try:
|
54
|
-
validate_xml(xmlfile, self.xml_validator_file)
|
64
|
+
validate_xml(xmlfile.path, self.xml_validator_file)
|
55
65
|
except Exception as e:
|
56
66
|
msg = _("XML validation failed: {}").format(e)
|
57
67
|
# print(msg)
|
58
68
|
raise Warning(msg)
|
59
69
|
|
60
|
-
url = settings.SITE.build_media_url(*parts)
|
70
|
+
# url = settings.SITE.build_media_url(*parts)
|
61
71
|
# return mark_safe(f"""<a href="{url}">{url}</a>""")
|
62
|
-
return (xmlfile, url)
|
72
|
+
# return (xmlfile, url)
|
73
|
+
return xmlfile
|