django-nativemojo 0.1.15__py3-none-any.whl → 0.1.16__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 (221) hide show
  1. {django_nativemojo-0.1.15.dist-info → django_nativemojo-0.1.16.dist-info}/METADATA +3 -1
  2. django_nativemojo-0.1.16.dist-info/RECORD +302 -0
  3. mojo/__init__.py +1 -1
  4. mojo/apps/account/management/commands/serializer_admin.py +121 -1
  5. mojo/apps/account/migrations/0006_add_device_tracking_models.py +72 -0
  6. mojo/apps/account/migrations/0007_delete_userdevicelocation.py +16 -0
  7. mojo/apps/account/migrations/0008_userdevicelocation.py +33 -0
  8. mojo/apps/account/migrations/0009_geolocatedip_subnet.py +18 -0
  9. mojo/apps/account/migrations/0010_group_avatar.py +20 -0
  10. mojo/apps/account/migrations/0011_user_org_registereddevice_pushconfig_and_more.py +118 -0
  11. mojo/apps/account/migrations/0012_remove_pushconfig_apns_key_file_and_more.py +21 -0
  12. mojo/apps/account/migrations/0013_pushconfig_test_mode_alter_pushconfig_apns_enabled_and_more.py +28 -0
  13. mojo/apps/account/migrations/0014_notificationdelivery_data_payload_and_more.py +48 -0
  14. mojo/apps/account/models/__init__.py +2 -0
  15. mojo/apps/account/models/device.py +281 -0
  16. mojo/apps/account/models/group.py +294 -8
  17. mojo/apps/account/models/member.py +14 -1
  18. mojo/apps/account/models/push/__init__.py +4 -0
  19. mojo/apps/account/models/push/config.py +112 -0
  20. mojo/apps/account/models/push/delivery.py +93 -0
  21. mojo/apps/account/models/push/device.py +66 -0
  22. mojo/apps/account/models/push/template.py +99 -0
  23. mojo/apps/account/models/user.py +190 -17
  24. mojo/apps/account/rest/__init__.py +2 -0
  25. mojo/apps/account/rest/device.py +39 -0
  26. mojo/apps/account/rest/group.py +8 -0
  27. mojo/apps/account/rest/push.py +187 -0
  28. mojo/apps/account/rest/user.py +95 -5
  29. mojo/apps/account/services/__init__.py +1 -0
  30. mojo/apps/account/services/push.py +363 -0
  31. mojo/apps/aws/migrations/0001_initial.py +206 -0
  32. mojo/apps/aws/migrations/0002_emaildomain_can_recv_emaildomain_can_send_and_more.py +28 -0
  33. mojo/apps/aws/migrations/0003_mailbox_is_domain_default_mailbox_is_system_default_and_more.py +31 -0
  34. mojo/apps/aws/migrations/0004_s3bucket.py +39 -0
  35. mojo/apps/aws/migrations/0005_alter_emaildomain_region_delete_s3bucket.py +21 -0
  36. mojo/apps/aws/models/__init__.py +19 -0
  37. mojo/apps/aws/models/email_attachment.py +99 -0
  38. mojo/apps/aws/models/email_domain.py +218 -0
  39. mojo/apps/aws/models/email_template.py +132 -0
  40. mojo/apps/aws/models/incoming_email.py +197 -0
  41. mojo/apps/aws/models/mailbox.py +288 -0
  42. mojo/apps/aws/models/sent_message.py +175 -0
  43. mojo/apps/aws/rest/__init__.py +6 -0
  44. mojo/apps/aws/rest/email.py +33 -0
  45. mojo/apps/aws/rest/email_ops.py +183 -0
  46. mojo/apps/aws/rest/messages.py +32 -0
  47. mojo/apps/aws/rest/send.py +101 -0
  48. mojo/apps/aws/rest/sns.py +403 -0
  49. mojo/apps/aws/rest/templates.py +19 -0
  50. mojo/apps/aws/services/__init__.py +32 -0
  51. mojo/apps/aws/services/email.py +390 -0
  52. mojo/apps/aws/services/email_ops.py +548 -0
  53. mojo/apps/docit/__init__.py +6 -0
  54. mojo/apps/docit/markdown_plugins/syntax_highlight.py +25 -0
  55. mojo/apps/docit/markdown_plugins/toc.py +12 -0
  56. mojo/apps/docit/migrations/0001_initial.py +113 -0
  57. mojo/apps/docit/migrations/0002_alter_book_modified_by_alter_page_modified_by.py +26 -0
  58. mojo/apps/docit/migrations/0003_alter_book_group.py +20 -0
  59. mojo/apps/docit/models/__init__.py +17 -0
  60. mojo/apps/docit/models/asset.py +231 -0
  61. mojo/apps/docit/models/book.py +227 -0
  62. mojo/apps/docit/models/page.py +319 -0
  63. mojo/apps/docit/models/page_revision.py +203 -0
  64. mojo/apps/docit/rest/__init__.py +10 -0
  65. mojo/apps/docit/rest/asset.py +17 -0
  66. mojo/apps/docit/rest/book.py +22 -0
  67. mojo/apps/docit/rest/page.py +22 -0
  68. mojo/apps/docit/rest/page_revision.py +17 -0
  69. mojo/apps/docit/services/__init__.py +11 -0
  70. mojo/apps/docit/services/docit.py +315 -0
  71. mojo/apps/docit/services/markdown.py +44 -0
  72. mojo/apps/fileman/backends/s3.py +209 -0
  73. mojo/apps/fileman/models/file.py +45 -9
  74. mojo/apps/fileman/models/manager.py +269 -3
  75. mojo/apps/incident/migrations/0007_event_uid.py +18 -0
  76. mojo/apps/incident/migrations/0008_ticket_ticketnote.py +55 -0
  77. mojo/apps/incident/migrations/0009_incident_status.py +18 -0
  78. mojo/apps/incident/migrations/0010_event_country_code.py +18 -0
  79. mojo/apps/incident/migrations/0011_incident_country_code.py +18 -0
  80. mojo/apps/incident/migrations/0012_alter_incident_status.py +18 -0
  81. mojo/apps/incident/models/__init__.py +1 -0
  82. mojo/apps/incident/models/event.py +35 -0
  83. mojo/apps/incident/models/incident.py +2 -0
  84. mojo/apps/incident/models/ticket.py +62 -0
  85. mojo/apps/incident/reporter.py +21 -3
  86. mojo/apps/incident/rest/__init__.py +1 -0
  87. mojo/apps/incident/rest/ticket.py +43 -0
  88. mojo/apps/jobs/__init__.py +489 -0
  89. mojo/apps/jobs/adapters.py +24 -0
  90. mojo/apps/jobs/cli.py +616 -0
  91. mojo/apps/jobs/daemon.py +370 -0
  92. mojo/apps/jobs/examples/sample_jobs.py +376 -0
  93. mojo/apps/jobs/examples/webhook_examples.py +203 -0
  94. mojo/apps/jobs/handlers/__init__.py +5 -0
  95. mojo/apps/jobs/handlers/webhook.py +317 -0
  96. mojo/apps/jobs/job_engine.py +734 -0
  97. mojo/apps/jobs/keys.py +203 -0
  98. mojo/apps/jobs/local_queue.py +363 -0
  99. mojo/apps/jobs/management/__init__.py +3 -0
  100. mojo/apps/jobs/management/commands/__init__.py +3 -0
  101. mojo/apps/jobs/manager.py +1327 -0
  102. mojo/apps/jobs/migrations/0001_initial.py +97 -0
  103. mojo/apps/jobs/migrations/0002_alter_job_max_retries_joblog.py +39 -0
  104. mojo/apps/jobs/models/__init__.py +6 -0
  105. mojo/apps/jobs/models/job.py +441 -0
  106. mojo/apps/jobs/rest/__init__.py +2 -0
  107. mojo/apps/jobs/rest/control.py +466 -0
  108. mojo/apps/jobs/rest/jobs.py +421 -0
  109. mojo/apps/jobs/scheduler.py +571 -0
  110. mojo/apps/jobs/services/__init__.py +6 -0
  111. mojo/apps/jobs/services/job_actions.py +465 -0
  112. mojo/apps/jobs/settings.py +209 -0
  113. mojo/apps/logit/models/log.py +3 -0
  114. mojo/apps/metrics/__init__.py +8 -1
  115. mojo/apps/metrics/redis_metrics.py +198 -0
  116. mojo/apps/metrics/rest/__init__.py +3 -0
  117. mojo/apps/metrics/rest/categories.py +266 -0
  118. mojo/apps/metrics/rest/helpers.py +48 -0
  119. mojo/apps/metrics/rest/permissions.py +99 -0
  120. mojo/apps/metrics/rest/values.py +277 -0
  121. mojo/apps/metrics/utils.py +17 -0
  122. mojo/decorators/http.py +40 -1
  123. mojo/helpers/aws/__init__.py +11 -7
  124. mojo/helpers/aws/inbound_email.py +309 -0
  125. mojo/helpers/aws/kms.py +413 -0
  126. mojo/helpers/aws/ses_domain.py +959 -0
  127. mojo/helpers/crypto/__init__.py +1 -1
  128. mojo/helpers/crypto/utils.py +15 -0
  129. mojo/helpers/location/__init__.py +2 -0
  130. mojo/helpers/location/countries.py +262 -0
  131. mojo/helpers/location/geolocation.py +196 -0
  132. mojo/helpers/logit.py +37 -0
  133. mojo/helpers/redis/__init__.py +2 -0
  134. mojo/helpers/redis/adapter.py +606 -0
  135. mojo/helpers/redis/client.py +48 -0
  136. mojo/helpers/redis/pool.py +225 -0
  137. mojo/helpers/request.py +8 -0
  138. mojo/helpers/response.py +8 -0
  139. mojo/middleware/auth.py +1 -1
  140. mojo/middleware/cors.py +40 -0
  141. mojo/middleware/logging.py +131 -12
  142. mojo/middleware/mojo.py +5 -0
  143. mojo/models/rest.py +271 -57
  144. mojo/models/secrets.py +86 -0
  145. mojo/serializers/__init__.py +16 -10
  146. mojo/serializers/core/__init__.py +90 -0
  147. mojo/serializers/core/cache/__init__.py +121 -0
  148. mojo/serializers/core/cache/backends.py +518 -0
  149. mojo/serializers/core/cache/base.py +102 -0
  150. mojo/serializers/core/cache/disabled.py +181 -0
  151. mojo/serializers/core/cache/memory.py +287 -0
  152. mojo/serializers/core/cache/redis.py +533 -0
  153. mojo/serializers/core/cache/utils.py +454 -0
  154. mojo/serializers/{manager.py → core/manager.py} +53 -4
  155. mojo/serializers/core/serializer.py +475 -0
  156. mojo/serializers/{advanced/formats → formats}/csv.py +116 -139
  157. mojo/serializers/suggested_improvements.md +388 -0
  158. testit/client.py +1 -1
  159. testit/helpers.py +14 -0
  160. testit/runner.py +23 -6
  161. django_nativemojo-0.1.15.dist-info/RECORD +0 -234
  162. mojo/apps/notify/README.md +0 -91
  163. mojo/apps/notify/README_NOTIFICATIONS.md +0 -566
  164. mojo/apps/notify/admin.py +0 -52
  165. mojo/apps/notify/handlers/example_handlers.py +0 -516
  166. mojo/apps/notify/handlers/ses/__init__.py +0 -25
  167. mojo/apps/notify/handlers/ses/complaint.py +0 -25
  168. mojo/apps/notify/handlers/ses/message.py +0 -86
  169. mojo/apps/notify/management/commands/__init__.py +0 -1
  170. mojo/apps/notify/management/commands/process_notifications.py +0 -370
  171. mojo/apps/notify/mod +0 -0
  172. mojo/apps/notify/models/__init__.py +0 -12
  173. mojo/apps/notify/models/account.py +0 -128
  174. mojo/apps/notify/models/attachment.py +0 -24
  175. mojo/apps/notify/models/bounce.py +0 -68
  176. mojo/apps/notify/models/complaint.py +0 -40
  177. mojo/apps/notify/models/inbox.py +0 -113
  178. mojo/apps/notify/models/inbox_message.py +0 -173
  179. mojo/apps/notify/models/outbox.py +0 -129
  180. mojo/apps/notify/models/outbox_message.py +0 -288
  181. mojo/apps/notify/models/template.py +0 -30
  182. mojo/apps/notify/providers/aws.py +0 -73
  183. mojo/apps/notify/rest/ses.py +0 -0
  184. mojo/apps/notify/utils/__init__.py +0 -2
  185. mojo/apps/notify/utils/notifications.py +0 -404
  186. mojo/apps/notify/utils/parsing.py +0 -202
  187. mojo/apps/notify/utils/render.py +0 -144
  188. mojo/apps/tasks/README.md +0 -118
  189. mojo/apps/tasks/__init__.py +0 -44
  190. mojo/apps/tasks/manager.py +0 -644
  191. mojo/apps/tasks/rest/__init__.py +0 -2
  192. mojo/apps/tasks/rest/hooks.py +0 -0
  193. mojo/apps/tasks/rest/tasks.py +0 -76
  194. mojo/apps/tasks/runner.py +0 -439
  195. mojo/apps/tasks/task.py +0 -99
  196. mojo/apps/tasks/tq_handlers.py +0 -132
  197. mojo/helpers/crypto/__pycache__/hash.cpython-310.pyc +0 -0
  198. mojo/helpers/crypto/__pycache__/sign.cpython-310.pyc +0 -0
  199. mojo/helpers/crypto/__pycache__/utils.cpython-310.pyc +0 -0
  200. mojo/helpers/redis.py +0 -10
  201. mojo/models/meta.py +0 -262
  202. mojo/serializers/advanced/README.md +0 -363
  203. mojo/serializers/advanced/__init__.py +0 -247
  204. mojo/serializers/advanced/formats/__init__.py +0 -28
  205. mojo/serializers/advanced/formats/excel.py +0 -516
  206. mojo/serializers/advanced/formats/json.py +0 -239
  207. mojo/serializers/advanced/formats/response.py +0 -485
  208. mojo/serializers/advanced/serializer.py +0 -568
  209. mojo/serializers/optimized.py +0 -618
  210. {django_nativemojo-0.1.15.dist-info → django_nativemojo-0.1.16.dist-info}/LICENSE +0 -0
  211. {django_nativemojo-0.1.15.dist-info → django_nativemojo-0.1.16.dist-info}/NOTICE +0 -0
  212. {django_nativemojo-0.1.15.dist-info → django_nativemojo-0.1.16.dist-info}/WHEEL +0 -0
  213. /mojo/apps/{notify → aws/migrations}/__init__.py +0 -0
  214. /mojo/apps/{notify/handlers → docit/markdown_plugins}/__init__.py +0 -0
  215. /mojo/apps/{notify/management → docit/migrations}/__init__.py +0 -0
  216. /mojo/apps/{notify/providers → jobs/examples}/__init__.py +0 -0
  217. /mojo/apps/{notify/rest → jobs/migrations}/__init__.py +0 -0
  218. /mojo/{serializers → rest}/openapi.py +0 -0
  219. /mojo/serializers/{settings_example.py → examples/settings.py} +0 -0
  220. /mojo/{apps/notify/handlers/ses/bounce.py → serializers/formats/__init__.py} +0 -0
  221. /mojo/serializers/{advanced/formats → formats}/localizers.py +0 -0
@@ -0,0 +1,315 @@
1
+ from mojo.helpers import logit
2
+ from ..models import Book, Page, PageRevision, Asset
3
+
4
+
5
+ class DocItService:
6
+ """
7
+ Business logic service for DocIt operations
8
+
9
+ Handles complex operations that span multiple models or contain
10
+ business logic that doesn't belong in individual model methods.
11
+ """
12
+
13
+ @staticmethod
14
+ def create_book_with_homepage(title, description, group, user, homepage_title="Home"):
15
+ """
16
+ Create a new book with an initial homepage
17
+
18
+ This is a common pattern where new books should have at least one page
19
+ """
20
+ try:
21
+ # Create the book
22
+ book = Book.objects.create(
23
+ title=title,
24
+ description=description,
25
+ group=group,
26
+ user=user,
27
+ created_by=user,
28
+ modified_by=user
29
+ )
30
+
31
+ # Create the homepage
32
+ homepage = Page.objects.create(
33
+ book=book,
34
+ title=homepage_title,
35
+ content=f"# {homepage_title}\n\nWelcome to {title}.",
36
+ order_priority=1000, # High priority to appear first
37
+ user=user,
38
+ created_by=user,
39
+ modified_by=user
40
+ )
41
+
42
+ # Create initial revision
43
+ homepage.create_revision(
44
+ user=user,
45
+ change_summary="Initial page creation"
46
+ )
47
+
48
+ logit.info(f"Created new book '{title}' with homepage for user {user.username}")
49
+
50
+ return book, homepage
51
+
52
+ except Exception as e:
53
+ logit.error(f"Failed to create book '{title}': {str(e)}")
54
+ raise
55
+
56
+ @staticmethod
57
+ def move_page(page, new_parent=None, new_position=None):
58
+ """
59
+ Move a page to a new location in the hierarchy
60
+
61
+ Handles validation and maintains data integrity
62
+ """
63
+ try:
64
+ old_parent = page.parent
65
+ old_path = page.full_path
66
+
67
+ # Validate the move
68
+ if new_parent and new_parent.book != page.book:
69
+ raise ValueError(f"Cannot move page to a different book: from '{page.book.id}' to '{new_parent.book.id}'")
70
+
71
+ if new_parent and page._would_create_cycle(new_parent):
72
+ raise ValueError("Move would create circular reference")
73
+
74
+ # Update the page
75
+ page.parent = new_parent
76
+
77
+ if new_position is not None:
78
+ page.order_priority = new_position
79
+
80
+ page.save()
81
+
82
+ new_path = page.full_path
83
+ logit.info(f"Moved page '{page.title}' from '{old_path}' to '{new_path}'")
84
+
85
+ return page
86
+
87
+ except Exception as e:
88
+ logit.error(f"Failed to move page '{page.title}': {str(e)}")
89
+ raise
90
+
91
+ @staticmethod
92
+ def duplicate_page(page, new_title, new_parent=None, include_children=False, user=None):
93
+ """
94
+ Create a duplicate of a page, optionally with its children
95
+ """
96
+ try:
97
+ # Create the duplicate
98
+ duplicate = Page.objects.create(
99
+ book=page.book,
100
+ parent=new_parent or page.parent,
101
+ title=new_title,
102
+ content=page.content,
103
+ order_priority=page.order_priority,
104
+ metadata=page.metadata.copy(),
105
+ is_published=False, # Start as draft
106
+ user=page.user,
107
+ created_by=user or page.created_by,
108
+ modified_by=user or page.modified_by
109
+ )
110
+
111
+ # Create initial revision for the duplicate
112
+ duplicate.create_revision(
113
+ user=user or page.created_by,
114
+ change_summary=f"Duplicated from '{page.title}'"
115
+ )
116
+
117
+ # Duplicate children if requested
118
+ if include_children:
119
+ for child in page.get_all_children(include_unpublished=True):
120
+ DocItService.duplicate_page(
121
+ page=child,
122
+ new_title=child.title,
123
+ new_parent=duplicate,
124
+ include_children=True, # Recursive
125
+ user=user
126
+ )
127
+
128
+ logit.info(f"Duplicated page '{page.title}' as '{new_title}' (children: {include_children})")
129
+
130
+ return duplicate
131
+
132
+ except Exception as e:
133
+ logit.error(f"Failed to duplicate page '{page.title}': {str(e)}")
134
+ raise
135
+
136
+ @staticmethod
137
+ def bulk_update_page_status(pages, is_published, user):
138
+ """
139
+ Bulk update publication status for multiple pages
140
+ """
141
+ try:
142
+ updated_count = 0
143
+
144
+ for page in pages:
145
+ if page.is_published != is_published:
146
+ page.is_published = is_published
147
+ page.modified_by = user
148
+ page.save()
149
+ updated_count += 1
150
+
151
+ status = "published" if is_published else "unpublished"
152
+ logit.info(f"Bulk updated {updated_count} pages to {status} by {user.username}")
153
+
154
+ return updated_count
155
+
156
+ except Exception as e:
157
+ logit.error(f"Failed to bulk update page status: {str(e)}")
158
+ raise
159
+
160
+ @staticmethod
161
+ def get_book_structure(book, include_unpublished=False):
162
+ """
163
+ Get the complete hierarchical structure of a book
164
+
165
+ Returns a nested dictionary representing the page tree
166
+ """
167
+ def build_tree(pages, parent_id=None):
168
+ tree = []
169
+ for page in pages:
170
+ if page.parent_id == parent_id:
171
+ page_data = {
172
+ 'id': page.id,
173
+ 'title': page.title,
174
+ 'slug': page.slug,
175
+ 'is_published': page.is_published,
176
+ 'order_priority': page.order_priority,
177
+ 'children': build_tree(pages, page.id)
178
+ }
179
+ tree.append(page_data)
180
+ return tree
181
+
182
+ try:
183
+ queryset = book.pages.all()
184
+ if not include_unpublished:
185
+ queryset = queryset.filter(is_published=True)
186
+
187
+ pages = list(queryset.order_by('-order_priority', 'title'))
188
+ structure = build_tree(pages)
189
+
190
+ logit.debug(f"Generated structure for book '{book.title}' with {len(pages)} pages")
191
+
192
+ return structure
193
+
194
+ except Exception as e:
195
+ logit.error(f"Failed to get book structure for '{book.title}': {str(e)}")
196
+ raise
197
+
198
+ @staticmethod
199
+ def organize_assets(book, asset_ids_in_order):
200
+ """
201
+ Reorder assets within a book based on provided ID list
202
+ """
203
+ try:
204
+ assets = Asset.objects.filter(book=book, id__in=asset_ids_in_order)
205
+
206
+ updated_count = 0
207
+ for index, asset_id in enumerate(asset_ids_in_order):
208
+ asset = assets.filter(id=asset_id).first()
209
+ if asset:
210
+ new_priority = len(asset_ids_in_order) - index # Higher index = higher priority
211
+ if asset.order_priority != new_priority:
212
+ asset.order_priority = new_priority
213
+ asset.save()
214
+ updated_count += 1
215
+
216
+ logit.info(f"Reorganized {updated_count} assets in book '{book.title}'")
217
+
218
+ return updated_count
219
+
220
+ except Exception as e:
221
+ logit.error(f"Failed to organize assets for book '{book.title}': {str(e)}")
222
+ raise
223
+
224
+ @staticmethod
225
+ def cleanup_orphaned_revisions(max_revisions_per_page=50):
226
+ """
227
+ Clean up old revisions across all pages to prevent database bloat
228
+ """
229
+ try:
230
+ total_cleaned = 0
231
+
232
+ # Get all pages that have more than the max revisions
233
+ for page in Page.objects.all():
234
+ revision_count = page.revisions.count()
235
+
236
+ if revision_count > max_revisions_per_page:
237
+ cleaned = PageRevision.cleanup_old_revisions(page, max_revisions_per_page)
238
+ total_cleaned += cleaned
239
+
240
+ if total_cleaned > 0:
241
+ logit.info(f"Cleaned up {total_cleaned} old page revisions")
242
+
243
+ return total_cleaned
244
+
245
+ except Exception as e:
246
+ logit.error(f"Failed to cleanup orphaned revisions: {str(e)}")
247
+ raise
248
+
249
+ @staticmethod
250
+ def get_book_statistics(book):
251
+ """
252
+ Get comprehensive statistics for a book
253
+ """
254
+ try:
255
+ stats = {
256
+ 'total_pages': book.get_page_count(),
257
+ 'published_pages': book.pages.filter(is_published=True).count(),
258
+ 'draft_pages': book.pages.filter(is_published=False).count(),
259
+ 'total_assets': book.get_asset_count(),
260
+ 'image_assets': book.assets.filter(file__category='image').count(),
261
+ 'document_assets': book.assets.filter(file__category='document').count(),
262
+ 'total_revisions': PageRevision.objects.filter(page__book=book).count(),
263
+ 'root_pages': book.get_root_pages(published_only=False).count(),
264
+ 'max_depth': 0
265
+ }
266
+
267
+ # Calculate maximum page depth
268
+ for page in book.pages.all():
269
+ depth = page.get_depth()
270
+ if depth > stats['max_depth']:
271
+ stats['max_depth'] = depth
272
+
273
+ logit.debug(f"Generated statistics for book '{book.title}'")
274
+
275
+ return stats
276
+
277
+ except Exception as e:
278
+ logit.error(f"Failed to get statistics for book '{book.title}': {str(e)}")
279
+ raise
280
+
281
+ @staticmethod
282
+ def validate_book_integrity(book):
283
+ """
284
+ Validate the integrity of a book and its pages
285
+
286
+ Returns a list of issues found
287
+ """
288
+ issues = []
289
+
290
+ try:
291
+ # Check for circular references in page hierarchy
292
+ for page in book.pages.all():
293
+ try:
294
+ _ = page.full_path # This will fail if there's a cycle
295
+ except RecursionError:
296
+ issues.append(f"Circular reference detected in page hierarchy: {page.title}")
297
+
298
+ # Check for orphaned assets (assets without files)
299
+ orphaned_assets = book.assets.filter(file__isnull=True)
300
+ if orphaned_assets.exists():
301
+ issues.append(f"Found {orphaned_assets.count()} orphaned assets without files")
302
+
303
+ # Check for pages with same slug
304
+ slugs = book.pages.values_list('slug', flat=True)
305
+ duplicate_slugs = [slug for slug in set(slugs) if slugs.count(slug) > 1]
306
+ if duplicate_slugs:
307
+ issues.append(f"Duplicate page slugs found: {duplicate_slugs}")
308
+
309
+ logit.info(f"Book integrity check for '{book.title}' found {len(issues)} issues")
310
+
311
+ return issues
312
+
313
+ except Exception as e:
314
+ logit.error(f"Failed to validate book integrity for '{book.title}': {str(e)}")
315
+ raise
@@ -0,0 +1,44 @@
1
+ import mistune
2
+ from pygments import highlight
3
+ from pygments.lexers import get_lexer_by_name
4
+ from pygments.styles import get_style_by_name
5
+ from pygments.formatters import HtmlFormatter
6
+
7
+
8
+ class HighlightRenderer(mistune.HTMLRenderer):
9
+ def block_code(self, code, info=None):
10
+ if not info:
11
+ return f'\n<pre>{mistune.escape(code)}</pre>\n'
12
+ lexer = get_lexer_by_name(info, stripall=True)
13
+ formatter = HtmlFormatter(
14
+ linenos=False,
15
+ cssclass="highlight",
16
+ style=get_style_by_name("monokai")
17
+ )
18
+ return highlight(code, lexer, formatter)
19
+
20
+ class MarkdownRenderer:
21
+ _renderer = None
22
+
23
+ def __init__(self):
24
+ if not self._renderer:
25
+ self._initialize_renderer()
26
+
27
+ def _initialize_renderer(self):
28
+ # plugins = self._discover_plugins()
29
+ self._renderer = mistune.create_markdown(
30
+ renderer=HighlightRenderer(escape=False),
31
+ escape=False,
32
+ hard_wrap=True,
33
+ plugins=[]
34
+ )
35
+
36
+ def _discover_plugins(self):
37
+ # from mojo.apps.docit.markdown_plugins import syntax_highlight
38
+ plugins = [
39
+ 'table', 'url', 'task_list',
40
+ 'footnotes', 'abbr', 'mark', 'math']
41
+ return plugins
42
+
43
+ def render(self, markdown_text):
44
+ return self._renderer(markdown_text)
@@ -425,6 +425,21 @@ class S3StorageBackend(StorageBackend):
425
425
 
426
426
  return len(errors) == 0, errors
427
427
 
428
+ def test_connection(self):
429
+ try:
430
+ self.client.head_bucket(Bucket=self.bucket_name)
431
+ return True
432
+ except NoCredentialsError:
433
+ raise ValueError("Invalid AWS credentials")
434
+ except ClientError as e:
435
+ error_code = e.response['Error']['Code']
436
+ if error_code == '404':
437
+ raise ValueError(f"S3 bucket '{self.bucket_name}' does not exist")
438
+ elif error_code == '403':
439
+ raise ValueError(f"Access denied to S3 bucket '{self.bucket_name}'")
440
+ else:
441
+ raise ValueError(f"S3 connection error: {e}")
442
+
428
443
  def make_path_public(self):
429
444
  # Get the current bucket policy (if any)
430
445
  try:
@@ -488,3 +503,197 @@ class S3StorageBackend(StorageBackend):
488
503
  self.client.download_fileobj(self.bucket_name, file_path, local_file)
489
504
  except ClientError as e:
490
505
  raise Exception(f"Failed to download file from S3: {e}")
506
+
507
+ # -------------------------------
508
+ # CORS MANAGEMENT FOR DIRECT UPLOADS
509
+ # -------------------------------
510
+ def get_cors_configuration(self) -> Optional[Dict[str, Any]]:
511
+ """
512
+ Return the current CORS configuration for the bucket, or None if not set.
513
+ """
514
+ try:
515
+ resp = self.client.get_bucket_cors(Bucket=self.bucket_name)
516
+ return resp
517
+ except ClientError as e:
518
+ if e.response.get("Error", {}).get("Code") == "NoSuchCORSConfiguration":
519
+ return None
520
+ raise
521
+
522
+ def _default_direct_upload_cors_rule(
523
+ self,
524
+ allowed_origins: List[str],
525
+ allowed_methods: Optional[List[str]] = None,
526
+ allowed_headers: Optional[List[str]] = None,
527
+ expose_headers: Optional[List[str]] = None,
528
+ max_age_seconds: int = 3000,
529
+ ) -> Dict[str, Any]:
530
+ """
531
+ Build a single CORS rule suitable for direct uploads via pre-signed PUT/POST.
532
+ Note: S3 CORS applies at the bucket level, not per-prefix. Access is still
533
+ enforced by IAM policies and the fact that we use pre-signed URLs.
534
+ """
535
+ if not allowed_origins or not any(str(o).strip() for o in allowed_origins):
536
+ raise ValueError("allowed_origins must contain at least one origin")
537
+ methods = allowed_methods or ["PUT", "HEAD"]
538
+ headers = allowed_headers or ["*"] # simplest and safest for signed uploads
539
+ expose = expose_headers or ["ETag", "x-amz-request-id", "x-amz-id-2", "x-amz-version-id"]
540
+
541
+ return {
542
+ "CORSRules": [
543
+ {
544
+ "AllowedOrigins": allowed_origins,
545
+ "AllowedMethods": methods,
546
+ "AllowedHeaders": headers,
547
+ "ExposeHeaders": expose,
548
+ "MaxAgeSeconds": max_age_seconds,
549
+ }
550
+ ]
551
+ }
552
+
553
+ def check_cors_configuration_for_direct_upload(
554
+ self,
555
+ allowed_origins: List[str],
556
+ required_methods: Optional[List[str]] = None,
557
+ required_headers: Optional[List[str]] = None,
558
+ ) -> Tuple[bool, List[str], Optional[Dict[str, Any]]]:
559
+ """
560
+ Validate current CORS config can support direct uploads from the given origins.
561
+
562
+ Returns:
563
+ (ok, issues, current_config)
564
+ """
565
+ if not allowed_origins or not any(str(o).strip() for o in allowed_origins):
566
+ raise ValueError("allowed_origins must contain at least one origin")
567
+ issues: List[str] = []
568
+ config = self.get_cors_configuration()
569
+ if config is None:
570
+ return False, ["No CORS configuration set on this bucket."], None
571
+
572
+ if required_methods is None:
573
+ required_methods = ["POST", "HEAD"] if self.server_side_encryption else ["PUT", "HEAD"]
574
+ required_methods = [m.upper() for m in required_methods]
575
+ # For PUT we often need Content-Type. For POST, headers are not required (fields are in the form).
576
+ # Using "*" for AllowedHeaders is the simplest and reduces edge cases.
577
+ if required_headers is None:
578
+ required_headers = [] if self.server_side_encryption else ["content-type"]
579
+ required_headers = [h.lower() for h in required_headers]
580
+
581
+ # Flatten rules
582
+ cors_rules: List[Dict[str, Any]] = config.get("CORSRules", [])
583
+
584
+ def origin_is_covered(origin: str) -> bool:
585
+ for rule in cors_rules:
586
+ origins = rule.get("AllowedOrigins", [])
587
+ if "*" in origins or origin in origins:
588
+ # methods
589
+ methods = [m.upper() for m in rule.get("AllowedMethods", [])]
590
+ if not all(m in methods for m in required_methods):
591
+ continue
592
+ # headers
593
+ hdrs = [h.lower() for h in rule.get("AllowedHeaders", [])]
594
+ if "*" in hdrs:
595
+ return True
596
+ if not all(h in hdrs for h in required_headers):
597
+ continue
598
+ return True
599
+ return False
600
+
601
+ for origin in allowed_origins:
602
+ if not origin_is_covered(origin):
603
+ issues.append(f"Origin not covered for direct upload: {origin}")
604
+
605
+ return (len(issues) == 0), issues, config
606
+
607
+ def update_cors_configuration_for_direct_upload(
608
+ self,
609
+ allowed_origins: List[str],
610
+ allowed_methods: Optional[List[str]] = None,
611
+ allowed_headers: Optional[List[str]] = None,
612
+ expose_headers: Optional[List[str]] = None,
613
+ max_age_seconds: int = 3000,
614
+ merge: bool = True,
615
+ ) -> Dict[str, Any]:
616
+ """
617
+ Ensure CORS allows direct uploads from allowed_origins.
618
+ If merge=True, append our rule to any existing rules instead of replacing.
619
+ """
620
+ # Validate input
621
+ if not allowed_origins or not any(str(o).strip() for o in allowed_origins):
622
+ raise ValueError("allowed_origins must contain at least one origin")
623
+ # If current config already satisfies requirements, do nothing
624
+ ok, issues, current = self.check_cors_configuration_for_direct_upload(
625
+ allowed_origins=allowed_origins,
626
+ required_methods=allowed_methods,
627
+ required_headers=allowed_headers,
628
+ )
629
+ if ok:
630
+ return {
631
+ "changed": False,
632
+ "message": "Existing CORS configuration already supports direct uploads.",
633
+ "issues": [],
634
+ "applied_configuration": current,
635
+ }
636
+
637
+ new_rule_config = self._default_direct_upload_cors_rule(
638
+ allowed_origins=allowed_origins,
639
+ allowed_methods=allowed_methods or (["POST", "HEAD"] if self.server_side_encryption else ["PUT", "HEAD"]),
640
+ allowed_headers=allowed_headers or ["*"],
641
+ expose_headers=expose_headers,
642
+ max_age_seconds=max_age_seconds,
643
+ )
644
+
645
+ if merge and current:
646
+ merged = dict(CORSRules=[*current.get("CORSRules", []), *new_rule_config["CORSRules"]])
647
+ self.client.put_bucket_cors(Bucket=self.bucket_name, CORSConfiguration=merged)
648
+ applied = merged
649
+ else:
650
+ # Replace entirely with our single rule
651
+ self.client.put_bucket_cors(Bucket=self.bucket_name, CORSConfiguration=new_rule_config)
652
+ applied = new_rule_config
653
+
654
+ return {
655
+ "changed": True,
656
+ "message": "CORS configuration updated to support direct uploads.",
657
+ "issues": issues,
658
+ "applied_configuration": applied,
659
+ }
660
+
661
+ def ensure_cors_for_direct_upload(
662
+ self,
663
+ allowed_origins: List[str],
664
+ allowed_methods: Optional[List[str]] = None,
665
+ allowed_headers: Optional[List[str]] = None,
666
+ expose_headers: Optional[List[str]] = None,
667
+ max_age_seconds: int = 3000,
668
+ merge: bool = True,
669
+ ) -> Dict[str, Any]:
670
+ """
671
+ Convenience wrapper that checks and updates CORS as needed.
672
+
673
+ Example:
674
+ backend.ensure_cors_for_direct_upload(
675
+ ["http://localhost:3000", "https://your-prod-domain.com"]
676
+ )
677
+ """
678
+ if not allowed_origins or not any(str(o).strip() for o in allowed_origins):
679
+ raise ValueError("allowed_origins must contain at least one origin")
680
+ result = self.update_cors_configuration_for_direct_upload(
681
+ allowed_origins=allowed_origins,
682
+ allowed_methods=allowed_methods,
683
+ allowed_headers=allowed_headers,
684
+ expose_headers=expose_headers,
685
+ max_age_seconds=max_age_seconds,
686
+ merge=merge,
687
+ )
688
+ # Re-check to confirm
689
+ ok, issues, current = self.check_cors_configuration_for_direct_upload(
690
+ allowed_origins=allowed_origins,
691
+ required_methods=allowed_methods,
692
+ required_headers=allowed_headers,
693
+ )
694
+ result.update({
695
+ "verified": ok,
696
+ "post_update_issues": issues,
697
+ "current_configuration": current,
698
+ })
699
+ return result
@@ -24,6 +24,7 @@ class File(models.Model, MojoModel):
24
24
  DEFAULT_SORT = "-created"
25
25
  VIEW_PERMS = ["view_fileman", "manage_files"]
26
26
  SEARCH_FIELDS = ["filename", "content_type"]
27
+ POST_SAVE_ACTIONS = ["action"]
27
28
  SEARCH_TERMS = [
28
29
  "filename", "content_type",
29
30
  ("group", "group__name"),
@@ -43,7 +44,7 @@ class File(models.Model, MojoModel):
43
44
  },
44
45
  "basic": {
45
46
  "fields": ["id", "filename", "content_type", "category"],
46
- "extra": ["url", "renditions"],
47
+ "extra": ["url", "thumbnail"],
47
48
  },
48
49
  "default": {
49
50
  "extra": ["url", "renditions"],
@@ -291,6 +292,16 @@ class File(models.Model, MojoModel):
291
292
  def url(self):
292
293
  return self.generate_download_url()
293
294
 
295
+ @property
296
+ def thumbnail(self):
297
+ r = self.get_rendition_by_role('thumbnail')
298
+ if r:
299
+ return r.url
300
+ return None
301
+
302
+ def get_rendition_by_role(self, role):
303
+ return self.file_renditions.filter(role=role).first()
304
+
294
305
  def generate_download_url(self):
295
306
  if self.download_url:
296
307
  return self.download_url
@@ -299,13 +310,13 @@ class File(models.Model, MojoModel):
299
310
  return self.download_url
300
311
  return self.file_manager.backend.get_url(self.storage_file_path, self.get_setting("urls_expire_in", 3600))
301
312
 
302
- def set_action(self, action):
313
+ def on_action_action(self, action):
303
314
  if action == "mark_as_completed":
304
- self.mark_as_completed()
315
+ self.mark_as_completed(commit=True)
305
316
  elif action == "mark_as_failed":
306
- self.mark_as_failed()
317
+ self.mark_as_failed(commit=True)
307
318
  elif action == "mark_as_uploading":
308
- self.mark_as_uploading()
319
+ self.mark_as_uploading(commit=True)
309
320
 
310
321
  def set_filename(self, filename):
311
322
  self.filename = filename
@@ -421,17 +432,42 @@ class File(models.Model, MojoModel):
421
432
  def on_rest_related_save(cls, related_instance, related_field_name, field_value, current_instance=None):
422
433
  # this allows us to handle json posts with inline base64 file data
423
434
  if isinstance(field_value, str):
424
- # assume base64 encoded data
425
- file_bytes = base64.b64decode(field_value)
426
- mime_type = magic.from_buffer(file_bytes, mime=True)
427
- ext = mimetypes.guess_extension(mime_type)
435
+ mime_type = None
436
+ b64_data = field_value
437
+
438
+ # Check for and parse Data URL scheme (e.g., "...")
439
+ if field_value.startswith('data:') and ',' in field_value:
440
+ header, b64_data = field_value.split(',', 1)
441
+ mime_type = header.split(';')[0].split(':')[1]
442
+
443
+ # Fix incorrect padding, which can occur with base64 strings from web clients
444
+ missing_padding = len(b64_data) % 4
445
+ if missing_padding:
446
+ b64_data += '=' * (4 - missing_padding)
447
+
448
+ try:
449
+ file_bytes = base64.b64decode(b64_data)
450
+ except (TypeError, base64.binascii.Error):
451
+ # If decoding fails, it's not a valid base64 string.
452
+ # In a real app, you might want to raise a validation error here.
453
+ return
454
+
455
+ # If mime_type wasn't in the data URL, detect it with python-magic
456
+ if not mime_type:
457
+ mime_type = magic.from_buffer(file_bytes, mime=True)
458
+
459
+ # Safely guess the extension, defaulting to an empty string if unknown
460
+ ext = mimetypes.guess_extension(mime_type) or ''
461
+
428
462
  file_obj = io.BytesIO(file_bytes)
429
463
  file_obj.name = f"{related_field_name}{ext}"
430
464
  file_obj.content_type = mime_type
431
465
  file_obj.size = len(file_bytes)
466
+
432
467
  # now we need to upload the file
433
468
  instance = cls.create_from_file(file_obj, file_obj.name)
434
469
  setattr(related_instance, related_field_name, instance)
470
+
435
471
  elif isinstance(field_value, int):
436
472
  # assume file id
437
473
  instance = File.objects.get(id=field_value)