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.
Files changed (81) hide show
  1. lino/__init__.py +1 -1
  2. lino/api/dd.py +11 -48
  3. lino/api/doctest.py +34 -36
  4. lino/core/actions.py +25 -23
  5. lino/core/actors.py +37 -17
  6. lino/core/choicelists.py +10 -8
  7. lino/core/dbtables.py +1 -1
  8. lino/core/elems.py +46 -30
  9. lino/core/fields.py +19 -9
  10. lino/core/inject.py +7 -6
  11. lino/core/kernel.py +26 -66
  12. lino/core/model.py +44 -31
  13. lino/core/plugin.py +4 -4
  14. lino/core/requests.py +76 -55
  15. lino/core/site.py +84 -30
  16. lino/core/store.py +5 -2
  17. lino/core/utils.py +12 -7
  18. lino/help_texts.py +3 -8
  19. lino/management/commands/prep.py +1 -1
  20. lino/mixins/duplicable.py +6 -4
  21. lino/mixins/sequenced.py +17 -6
  22. lino/modlib/__init__.py +0 -2
  23. lino/modlib/changes/models.py +21 -10
  24. lino/modlib/checkdata/models.py +59 -24
  25. lino/modlib/comments/fixtures/demo2.py +12 -3
  26. lino/modlib/comments/models.py +7 -7
  27. lino/modlib/comments/ui.py +8 -5
  28. lino/modlib/export_excel/models.py +7 -5
  29. lino/modlib/extjs/views.py +39 -20
  30. lino/modlib/help/management/commands/makehelp.py +5 -2
  31. lino/modlib/jinja/mixins.py +25 -14
  32. lino/modlib/linod/__init__.py +1 -0
  33. lino/modlib/linod/choicelists.py +21 -0
  34. lino/modlib/linod/consumers.py +13 -4
  35. lino/modlib/linod/management/commands/linod.py +6 -2
  36. lino/modlib/linod/mixins.py +16 -11
  37. lino/modlib/linod/models.py +4 -2
  38. lino/modlib/notify/models.py +18 -10
  39. lino/modlib/printing/actions.py +41 -30
  40. lino/modlib/printing/choicelists.py +11 -9
  41. lino/modlib/printing/mixins.py +25 -20
  42. lino/modlib/publisher/models.py +5 -5
  43. lino/modlib/summaries/models.py +3 -2
  44. lino/modlib/system/models.py +28 -29
  45. lino/modlib/uploads/__init__.py +5 -5
  46. lino/modlib/uploads/actions.py +2 -8
  47. lino/modlib/uploads/choicelists.py +10 -10
  48. lino/modlib/uploads/fixtures/std.py +17 -0
  49. lino/modlib/uploads/mixins.py +20 -8
  50. lino/modlib/uploads/models.py +60 -35
  51. lino/modlib/uploads/ui.py +10 -7
  52. lino/utils/media.py +45 -23
  53. lino/utils/report.py +5 -4
  54. lino/utils/soup.py +22 -4
  55. {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/METADATA +1 -1
  56. {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/RECORD +59 -80
  57. lino/mixins/uploadable.py +0 -3
  58. lino/sandbox/bcss/PerformInvestigation.py +0 -2260
  59. lino/sandbox/bcss/SSDNReply.py +0 -3924
  60. lino/sandbox/bcss/SSDNRequest.py +0 -3723
  61. lino/sandbox/bcss/__init__.py +0 -0
  62. lino/sandbox/bcss/readme.txt +0 -1
  63. lino/sandbox/bcss/test.py +0 -92
  64. lino/sandbox/bcss/test2.py +0 -128
  65. lino/sandbox/bcss/test3.py +0 -161
  66. lino/sandbox/bcss/test4.py +0 -167
  67. lino/sandbox/contacts/__init__.py +0 -0
  68. lino/sandbox/contacts/fixtures/__init__.py +0 -0
  69. lino/sandbox/contacts/fixtures/demo.py +0 -365
  70. lino/sandbox/contacts/manage.py +0 -10
  71. lino/sandbox/contacts/models.py +0 -395
  72. lino/sandbox/contacts/settings.py +0 -67
  73. lino/sandbox/tx25/XSD/RetrieveTIGroupsV3.wsdl +0 -65
  74. lino/sandbox/tx25/XSD/RetrieveTIGroupsV3.xsd +0 -286
  75. lino/sandbox/tx25/XSD/rn25_Release201104.xsd +0 -2855
  76. lino/sandbox/tx25/xsd2py1.py +0 -68
  77. lino/sandbox/tx25/xsd2py2.py +0 -62
  78. lino/sandbox/tx25/xsd2py3.py +0 -56
  79. {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/WHEEL +0 -0
  80. {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/licenses/AUTHORS.rst +0 -0
  81. {lino-25.2.3.dist-info → lino-25.3.1.dist-info}/licenses/COPYING +0 -0
@@ -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 UpdateMessage(dd.Action):
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
- qs = Message.objects.filter(
51
- **gfk2lookup(Message.owner, owner, checker=chk)
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(dd.Action):
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
- if fix is None:
78
- fix = self.fix_them
79
- Message = rt.models.checkdata.Message
80
- gfk = Message.owner
81
- checkers = get_checkers_for(self.model)
82
- for obj in ar.selected_rows:
83
- assert isinstance(obj, self.model)
84
- qs = Message.objects.filter(**gfk2lookup(gfk, obj))
85
- qs.delete()
86
- for chk in checkers:
87
- chk.update_problems(ar, obj, False, fix)
88
- ar.set_response(refresh=True)
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(m)
309
- qs = Message.objects.filter(owner_type=ct, checker__in=checkers)
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
- @dd.background_task(every_unit="daily", every=1)
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:
@@ -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(user=u, comment=self).first()
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("comments.Comment", related_name="reactions_to_this")
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 *
@@ -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, observed_event=CommentEvents.created)
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
- kw["reply_to"] = constants.CHOICES_BLANK_FILTER_VALUE
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(**mkw).values_list("owner_id", flat=True)
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(hours=int(time[0]), minutes=int(time[1]))
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(value.__class__, value))
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(os.path.dirname(mf.name))
136
+ settings.SITE.makedirs_if_missing(mf.path.parent)
135
137
 
136
138
  # Render
137
- self.render(ar, mf.name)
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.get_url(ar.request))
143
+ ar.success(open_url=mf.url)
142
144
 
143
145
  def render(self, ar, file):
144
146
  workbook = ar2workbook(ar)
@@ -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(request, run_jasmine=True)
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(ba.request(request=request), ba.action, 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(ar, elem, rh.store.grid_fields)])
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, ba.action.default_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 as e:
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(renderer=settings.SITE.kernel.default_renderer)
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(rescue_ar, icon_name=None))
369
+ title = tostring(rescue_ar.href_to_request(
370
+ rescue_ar, icon_name=None))
371
+
365
372
  def get_response():
366
- msg = mark_safe(f'Record (pk={pk}) is no longer available on current table.')
367
- datarec = dict(success=False, message=msg, title=title)
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(request=request, selected_pks=sr)
374
- except default_table.model.DoesNotExist as e:
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(f' Reload in <a href="{url}">{default_table}</a>.')
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(f"Object {sr} does not exist on {rpt}")
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(success=False, message=NOT_FOUND % (rpt, pk))
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("utf-8")
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) for row in ar.sliced_data_iterator]
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(show_params_panel=constants.parse_boolean(sp))
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(content_type='text/csv;charset="%s"' % charset)
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(_("List contains more than %d rows") % MAX_ROW_COUNT)
633
- response = http.HttpResponse(content_type='text/html;charset="utf-8"')
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
- from lino.api import doctest
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,
@@ -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
- dd.plugins.accounting.xml_media_dir,
39
- self.xml_file_name.format(self=self)]
40
- xmlfile = Path(settings.MEDIA_ROOT, *parts)
41
- ar.logger.info("Make %s from %s ...", xmlfile, self)
42
- xmlfile.parent.mkdir(exist_ok=True, parents=True)
43
- xmlfile.write_text(xml)
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 ...", xmlfile.name, self.xml_validator_file)
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