dmart 1.4.17__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 (289) hide show
  1. alembic.ini +117 -0
  2. api/__init__.py +0 -0
  3. api/info/__init__.py +0 -0
  4. api/info/router.py +109 -0
  5. api/managed/__init__.py +0 -0
  6. api/managed/router.py +1541 -0
  7. api/managed/utils.py +1879 -0
  8. api/public/__init__.py +0 -0
  9. api/public/router.py +758 -0
  10. api/qr/__init__.py +0 -0
  11. api/qr/router.py +108 -0
  12. api/user/__init__.py +0 -0
  13. api/user/model/__init__.py +0 -0
  14. api/user/model/errors.py +14 -0
  15. api/user/model/requests.py +165 -0
  16. api/user/model/responses.py +11 -0
  17. api/user/router.py +1413 -0
  18. api/user/service.py +270 -0
  19. bundler.py +55 -0
  20. config/__init__.py +0 -0
  21. config/channels.json +11 -0
  22. config/notification.json +17 -0
  23. cxb/__init__.py +0 -0
  24. cxb/client/__init__.py +0 -0
  25. cxb/client/assets/@codemirror-Rn7_6DkE.js +10 -0
  26. cxb/client/assets/@edraj-CS4NwVbD.js +1 -0
  27. cxb/client/assets/@floating-ui-BwwcF-xh.js +1 -0
  28. cxb/client/assets/@formatjs-yKEsAtjs.js +1 -0
  29. cxb/client/assets/@fortawesome-DRW1UCdr.js +9 -0
  30. cxb/client/assets/@jsonquerylang-laKNoFFq.js +12 -0
  31. cxb/client/assets/@lezer-za4Q-8Ew.js +1 -0
  32. cxb/client/assets/@marijn-DXwl3gUT.js +1 -0
  33. cxb/client/assets/@popperjs-l0sNRNKZ.js +1 -0
  34. cxb/client/assets/@replit--ERk53eB.js +1 -0
  35. cxb/client/assets/@roxi-CGMFK4i8.js +6 -0
  36. cxb/client/assets/@typewriter-cCzskkIv.js +17 -0
  37. cxb/client/assets/@zerodevx-BlBZjKxu.js +1 -0
  38. cxb/client/assets/@zerodevx-CVEpe6WZ.css +1 -0
  39. cxb/client/assets/BreadCrumbLite-DAhOx38v.js +1 -0
  40. cxb/client/assets/EntryRenderer-25YDhRen.js +32 -0
  41. cxb/client/assets/EntryRenderer-DXytdFp9.css +1 -0
  42. cxb/client/assets/ListView-BpAycA2h.js +16 -0
  43. cxb/client/assets/ListView-U8of-_c-.css +1 -0
  44. cxb/client/assets/Prism--hMplq-p.js +3 -0
  45. cxb/client/assets/Prism-Uh6uStUw.css +1 -0
  46. cxb/client/assets/Table2Cols-BsbwicQm.js +1 -0
  47. cxb/client/assets/_..-BvT6vdHa.css +1 -0
  48. cxb/client/assets/_...404_-fuLH_rX9.js +2 -0
  49. cxb/client/assets/_...fallback_-Ba_NLmAE.js +1 -0
  50. cxb/client/assets/_module-Bfk8MiCs.js +3 -0
  51. cxb/client/assets/_module-CEW0D5oI.js +4 -0
  52. cxb/client/assets/_module-Dgq0ZVtz.js +1 -0
  53. cxb/client/assets/ajv-Cpj98o6Y.js +1 -0
  54. cxb/client/assets/axios-CG2WSiiR.js +6 -0
  55. cxb/client/assets/clsx-B-dksMZM.js +1 -0
  56. cxb/client/assets/codemirror-wrapped-line-indent-DPhKvljI.js +1 -0
  57. cxb/client/assets/compare-C3AjiGFR.js +1 -0
  58. cxb/client/assets/compute-scroll-into-view-Bl8rNFhg.js +1 -0
  59. cxb/client/assets/consolite-DlCuI0F9.js +1 -0
  60. cxb/client/assets/crelt-C8TCjufn.js +1 -0
  61. cxb/client/assets/date-fns-l0sNRNKZ.js +1 -0
  62. cxb/client/assets/deepmerge-rn4rBaHU.js +1 -0
  63. cxb/client/assets/dmart_services-AL6-IdDE.js +1 -0
  64. cxb/client/assets/downloadFile-D08i0YDh.js +1 -0
  65. cxb/client/assets/easy-signal-BiPFIK3O.js +1 -0
  66. cxb/client/assets/esm-env-rsSWfq8L.js +1 -0
  67. cxb/client/assets/export-OF_rTiXu.js +1 -0
  68. cxb/client/assets/fast-deep-equal-l0sNRNKZ.js +1 -0
  69. cxb/client/assets/fast-diff-C-IidNf4.js +1 -0
  70. cxb/client/assets/fast-uri-l0sNRNKZ.js +1 -0
  71. cxb/client/assets/flowbite-svelte-BLvjb-sa.js +1 -0
  72. cxb/client/assets/flowbite-svelte-CD54FDqW.css +1 -0
  73. cxb/client/assets/flowbite-svelte-icons-BI8GVhw_.js +1 -0
  74. cxb/client/assets/github-slugger-CQ4oX9Ud.js +1 -0
  75. cxb/client/assets/global-igKv-1g9.js +1 -0
  76. cxb/client/assets/hookar-BMRD9G9H.js +1 -0
  77. cxb/client/assets/immutable-json-patch-DtRO2E_S.js +1 -0
  78. cxb/client/assets/import-1vE3gBat.js +1 -0
  79. cxb/client/assets/index-B-eTh-ZX.js +1 -0
  80. cxb/client/assets/index-BVyxzKtH.js +1 -0
  81. cxb/client/assets/index-BdeNM69f.js +1 -0
  82. cxb/client/assets/index-C6cPO4op.js +1 -0
  83. cxb/client/assets/index-CC-A1ipE.js +1 -0
  84. cxb/client/assets/index-CTxJ-lDp.js +1 -0
  85. cxb/client/assets/index-Cd-F5j_k.js +1 -0
  86. cxb/client/assets/index-D742rwaM.js +1 -0
  87. cxb/client/assets/index-DTfhnhwd.js +1 -0
  88. cxb/client/assets/index-DdXRK7n9.js +2 -0
  89. cxb/client/assets/index-DtiCmB4o.js +1 -0
  90. cxb/client/assets/index-NBrXBlLA.css +2 -0
  91. cxb/client/assets/index-ac-Buu_H.js +4 -0
  92. cxb/client/assets/index-iYkH7C67.js +1 -0
  93. cxb/client/assets/info-B986lRiM.js +1 -0
  94. cxb/client/assets/intl-messageformat-Dc5UU-HB.js +3 -0
  95. cxb/client/assets/jmespath-l0sNRNKZ.js +1 -0
  96. cxb/client/assets/json-schema-traverse-l0sNRNKZ.js +1 -0
  97. cxb/client/assets/json-source-map-DRgZidqy.js +5 -0
  98. cxb/client/assets/jsonpath-plus-l0sNRNKZ.js +1 -0
  99. cxb/client/assets/jsonrepair-B30Dx381.js +8 -0
  100. cxb/client/assets/lodash-es-DZVAA2ox.js +1 -0
  101. cxb/client/assets/marked-DKjyhwJX.js +56 -0
  102. cxb/client/assets/marked-gfm-heading-id-U5zO829x.js +2 -0
  103. cxb/client/assets/marked-mangle-CDMeiHC6.js +1 -0
  104. cxb/client/assets/memoize-one-BdPwpGay.js +1 -0
  105. cxb/client/assets/natural-compare-lite-Bg2Xcf-o.js +7 -0
  106. cxb/client/assets/pagination-svelte-D5CyoiE_.js +13 -0
  107. cxb/client/assets/pagination-svelte-v10nAbbM.css +1 -0
  108. cxb/client/assets/plantuml-encoder-C47mzt9T.js +1 -0
  109. cxb/client/assets/prismjs-DTUiLGJu.js +9 -0
  110. cxb/client/assets/profile-BUf-tKMe.js +1 -0
  111. cxb/client/assets/query-CNmXTsgf.js +1 -0
  112. cxb/client/assets/queryHelpers-C9iBWwqe.js +1 -0
  113. cxb/client/assets/scroll-into-view-if-needed-KR58zyjF.js +1 -0
  114. cxb/client/assets/spaces-0oyGvpii.js +1 -0
  115. cxb/client/assets/style-mod-Bs6eFhZE.js +3 -0
  116. cxb/client/assets/svelte-B2XmcTi_.js +4 -0
  117. cxb/client/assets/svelte-awesome-COLlx0DN.css +1 -0
  118. cxb/client/assets/svelte-awesome-DhnMA6Q_.js +1 -0
  119. cxb/client/assets/svelte-datatables-net-CY7LBj6I.js +1 -0
  120. cxb/client/assets/svelte-floating-ui-BlS3sOAQ.js +1 -0
  121. cxb/client/assets/svelte-i18n-CT2KkQaN.js +3 -0
  122. cxb/client/assets/svelte-jsoneditor-BzfX6Usi.css +1 -0
  123. cxb/client/assets/svelte-jsoneditor-CUGSvWId.js +25 -0
  124. cxb/client/assets/svelte-select-CegQKzqH.css +1 -0
  125. cxb/client/assets/svelte-select-CjHAt_85.js +6 -0
  126. cxb/client/assets/tailwind-merge-CJvxXMcu.js +1 -0
  127. cxb/client/assets/tailwind-variants-Cj20BoQ3.js +1 -0
  128. cxb/client/assets/toast-B9WDyfyI.js +1 -0
  129. cxb/client/assets/tslib-pJfR_DrR.js +1 -0
  130. cxb/client/assets/typewriter-editor-DkTVIJdm.js +25 -0
  131. cxb/client/assets/user-DeK_NB5v.js +1 -0
  132. cxb/client/assets/vanilla-picker-l5rcX3cq.js +8 -0
  133. cxb/client/assets/w3c-keyname-Vcq4gwWv.js +1 -0
  134. cxb/client/config.json +11 -0
  135. cxb/client/config.sample.json +11 -0
  136. cxb/client/favicon.ico +0 -0
  137. cxb/client/favicon.png +0 -0
  138. cxb/client/index.html +28 -0
  139. data_adapters/__init__.py +0 -0
  140. data_adapters/adapter.py +16 -0
  141. data_adapters/base_data_adapter.py +467 -0
  142. data_adapters/file/__init__.py +0 -0
  143. data_adapters/file/adapter.py +2043 -0
  144. data_adapters/file/adapter_helpers.py +1013 -0
  145. data_adapters/file/archive.py +150 -0
  146. data_adapters/file/create_index.py +331 -0
  147. data_adapters/file/create_users_folders.py +52 -0
  148. data_adapters/file/custom_validations.py +68 -0
  149. data_adapters/file/drop_index.py +40 -0
  150. data_adapters/file/health_check.py +560 -0
  151. data_adapters/file/redis_services.py +1110 -0
  152. data_adapters/helpers.py +27 -0
  153. data_adapters/sql/__init__.py +0 -0
  154. data_adapters/sql/adapter.py +3218 -0
  155. data_adapters/sql/adapter_helpers.py +491 -0
  156. data_adapters/sql/create_tables.py +451 -0
  157. data_adapters/sql/create_users_folders.py +53 -0
  158. data_adapters/sql/db_to_json_migration.py +485 -0
  159. data_adapters/sql/health_check_sql.py +232 -0
  160. data_adapters/sql/json_to_db_migration.py +454 -0
  161. data_adapters/sql/update_query_policies.py +101 -0
  162. data_generator.py +81 -0
  163. dmart-1.4.17.dist-info/METADATA +65 -0
  164. dmart-1.4.17.dist-info/RECORD +289 -0
  165. dmart-1.4.17.dist-info/WHEEL +5 -0
  166. dmart-1.4.17.dist-info/entry_points.txt +2 -0
  167. dmart-1.4.17.dist-info/top_level.txt +24 -0
  168. dmart.py +623 -0
  169. dmart_migrations/README +1 -0
  170. dmart_migrations/__init__.py +0 -0
  171. dmart_migrations/__pycache__/__init__.cpython-314.pyc +0 -0
  172. dmart_migrations/__pycache__/env.cpython-314.pyc +0 -0
  173. dmart_migrations/env.py +100 -0
  174. dmart_migrations/notes.txt +11 -0
  175. dmart_migrations/script.py.mako +28 -0
  176. dmart_migrations/scripts/__init__.py +0 -0
  177. dmart_migrations/scripts/calculate_checksums.py +77 -0
  178. dmart_migrations/scripts/migration_f7a4949eed19.py +28 -0
  179. dmart_migrations/versions/0f3d2b1a7c21_add_authz_materialized_views.py +87 -0
  180. dmart_migrations/versions/10d2041b94d4_last_checksum_history.py +62 -0
  181. dmart_migrations/versions/1cf4e1ee3cb8_ext_permission_with_filter_fields_values.py +33 -0
  182. dmart_migrations/versions/26bfe19b49d4_rm_failedloginattempts.py +42 -0
  183. dmart_migrations/versions/3c8bca2219cc_add_otp_table.py +38 -0
  184. dmart_migrations/versions/6675fd9dfe42_remove_unique_from_sessions_table.py +36 -0
  185. dmart_migrations/versions/71bc1df82e6a_adding_user_last_login_at.py +43 -0
  186. dmart_migrations/versions/74288ccbd3b5_initial.py +264 -0
  187. dmart_migrations/versions/7520a89a8467_rm_activesession_table.py +39 -0
  188. dmart_migrations/versions/848b623755a4_make_created_nd_updated_at_required.py +138 -0
  189. dmart_migrations/versions/8640dcbebf85_add_notes_to_users.py +32 -0
  190. dmart_migrations/versions/91c94250232a_adding_fk_on_owner_shortname.py +104 -0
  191. dmart_migrations/versions/98ecd6f56f9a_ext_meta_with_owner_group_shortname.py +66 -0
  192. dmart_migrations/versions/9aae9138c4ef_indexing_created_at_updated_at.py +80 -0
  193. dmart_migrations/versions/__init__.py +0 -0
  194. dmart_migrations/versions/__pycache__/0f3d2b1a7c21_add_authz_materialized_views.cpython-314.pyc +0 -0
  195. dmart_migrations/versions/__pycache__/10d2041b94d4_last_checksum_history.cpython-314.pyc +0 -0
  196. dmart_migrations/versions/__pycache__/1cf4e1ee3cb8_ext_permission_with_filter_fields_values.cpython-314.pyc +0 -0
  197. dmart_migrations/versions/__pycache__/26bfe19b49d4_rm_failedloginattempts.cpython-314.pyc +0 -0
  198. dmart_migrations/versions/__pycache__/3c8bca2219cc_add_otp_table.cpython-314.pyc +0 -0
  199. dmart_migrations/versions/__pycache__/6675fd9dfe42_remove_unique_from_sessions_table.cpython-314.pyc +0 -0
  200. dmart_migrations/versions/__pycache__/71bc1df82e6a_adding_user_last_login_at.cpython-314.pyc +0 -0
  201. dmart_migrations/versions/__pycache__/74288ccbd3b5_initial.cpython-314.pyc +0 -0
  202. dmart_migrations/versions/__pycache__/7520a89a8467_rm_activesession_table.cpython-314.pyc +0 -0
  203. dmart_migrations/versions/__pycache__/848b623755a4_make_created_nd_updated_at_required.cpython-314.pyc +0 -0
  204. dmart_migrations/versions/__pycache__/8640dcbebf85_add_notes_to_users.cpython-314.pyc +0 -0
  205. dmart_migrations/versions/__pycache__/91c94250232a_adding_fk_on_owner_shortname.cpython-314.pyc +0 -0
  206. dmart_migrations/versions/__pycache__/98ecd6f56f9a_ext_meta_with_owner_group_shortname.cpython-314.pyc +0 -0
  207. dmart_migrations/versions/__pycache__/9aae9138c4ef_indexing_created_at_updated_at.cpython-314.pyc +0 -0
  208. dmart_migrations/versions/__pycache__/b53f916b3f6d_json_to_jsonb.cpython-314.pyc +0 -0
  209. dmart_migrations/versions/__pycache__/eb5f1ec65156_adding_user_locked_to_device.cpython-314.pyc +0 -0
  210. dmart_migrations/versions/__pycache__/f7a4949eed19_adding_query_policies_to_meta.cpython-314.pyc +0 -0
  211. dmart_migrations/versions/b53f916b3f6d_json_to_jsonb.py +492 -0
  212. dmart_migrations/versions/eb5f1ec65156_adding_user_locked_to_device.py +36 -0
  213. dmart_migrations/versions/f7a4949eed19_adding_query_policies_to_meta.py +60 -0
  214. get_settings.py +7 -0
  215. info.json +1 -0
  216. languages/__init__.py +0 -0
  217. languages/arabic.json +15 -0
  218. languages/english.json +16 -0
  219. languages/kurdish.json +14 -0
  220. languages/loader.py +12 -0
  221. main.py +560 -0
  222. migrate.py +24 -0
  223. models/__init__.py +0 -0
  224. models/api.py +203 -0
  225. models/core.py +597 -0
  226. models/enums.py +255 -0
  227. password_gen.py +8 -0
  228. plugins/__init__.py +0 -0
  229. plugins/action_log/__init__.py +0 -0
  230. plugins/action_log/plugin.py +121 -0
  231. plugins/admin_notification_sender/__init__.py +0 -0
  232. plugins/admin_notification_sender/plugin.py +124 -0
  233. plugins/ldap_manager/__init__.py +0 -0
  234. plugins/ldap_manager/plugin.py +100 -0
  235. plugins/local_notification/__init__.py +0 -0
  236. plugins/local_notification/plugin.py +123 -0
  237. plugins/realtime_updates_notifier/__init__.py +0 -0
  238. plugins/realtime_updates_notifier/plugin.py +58 -0
  239. plugins/redis_db_update/__init__.py +0 -0
  240. plugins/redis_db_update/plugin.py +188 -0
  241. plugins/resource_folders_creation/__init__.py +0 -0
  242. plugins/resource_folders_creation/plugin.py +81 -0
  243. plugins/system_notification_sender/__init__.py +0 -0
  244. plugins/system_notification_sender/plugin.py +188 -0
  245. plugins/update_access_controls/__init__.py +0 -0
  246. plugins/update_access_controls/plugin.py +9 -0
  247. pytests/__init__.py +0 -0
  248. pytests/api_user_models_erros_test.py +16 -0
  249. pytests/api_user_models_requests_test.py +98 -0
  250. pytests/archive_test.py +72 -0
  251. pytests/base_test.py +300 -0
  252. pytests/get_settings_test.py +14 -0
  253. pytests/json_to_db_migration_test.py +237 -0
  254. pytests/service_test.py +26 -0
  255. pytests/test_info.py +55 -0
  256. pytests/test_status.py +15 -0
  257. run_notification_campaign.py +85 -0
  258. scheduled_notification_handler.py +121 -0
  259. schema_migration.py +208 -0
  260. schema_modulate.py +192 -0
  261. set_admin_passwd.py +55 -0
  262. sync.py +202 -0
  263. utils/__init__.py +0 -0
  264. utils/access_control.py +306 -0
  265. utils/async_request.py +8 -0
  266. utils/exporter.py +309 -0
  267. utils/firebase_notifier.py +57 -0
  268. utils/generate_email.py +37 -0
  269. utils/helpers.py +352 -0
  270. utils/hypercorn_config.py +12 -0
  271. utils/internal_error_code.py +60 -0
  272. utils/jwt.py +124 -0
  273. utils/logger.py +167 -0
  274. utils/middleware.py +99 -0
  275. utils/notification.py +75 -0
  276. utils/password_hashing.py +16 -0
  277. utils/plugin_manager.py +202 -0
  278. utils/query_policies_helper.py +128 -0
  279. utils/regex.py +44 -0
  280. utils/repository.py +529 -0
  281. utils/router_helper.py +19 -0
  282. utils/settings.py +166 -0
  283. utils/sms_notifier.py +21 -0
  284. utils/social_sso.py +67 -0
  285. utils/templates/activation.html.j2 +26 -0
  286. utils/templates/reminder.html.j2 +17 -0
  287. utils/ticket_sys_utils.py +203 -0
  288. utils/web_notifier.py +29 -0
  289. websocket.py +231 -0
@@ -0,0 +1,485 @@
1
+ #!/usr/bin/env -S BACKEND_ENV=config.env python3
2
+
3
+ import os
4
+ import json
5
+ from datetime import datetime
6
+ from sqlmodel import Session, create_engine, select, col
7
+ from data_adapters.sql.create_tables import (
8
+ Entries,
9
+ Users,
10
+ Attachments,
11
+ Roles,
12
+ Permissions,
13
+ Histories,
14
+ Spaces,
15
+ )
16
+ from models.core import Payload
17
+ from utils.settings import settings
18
+ from models import core
19
+ import base64
20
+
21
+ def get_engine():
22
+ if "sqlite" in settings.database_driver:
23
+ return create_engine(f"sqlite:///{settings.database_name}", echo=False)
24
+ postgresql_url = f"{settings.database_driver.replace('+asyncpg','+psycopg')}://{settings.database_username}:{settings.database_password}@{settings.database_host}:{settings.database_port}/{settings.database_name}"
25
+ return create_engine(postgresql_url, echo=False)
26
+
27
+ def subpath_checker(subpath: str):
28
+ if subpath.endswith("/"):
29
+ subpath = subpath[:-1]
30
+ if not subpath.startswith("/"):
31
+ subpath = "/" + subpath
32
+ return subpath
33
+
34
+ def ensure_directory_exists(path: str):
35
+ os.makedirs(path, exist_ok=True)
36
+
37
+ def clean_json(data: dict):
38
+ if not isinstance(data, dict):
39
+ return data
40
+
41
+ for key, value in list(data.items()):
42
+ if key in ["tags", "mirrors", "active_plugins", "roles", "groups"]:
43
+ continue
44
+ if not value:
45
+ del data[key]
46
+ elif isinstance(value, datetime):
47
+ data[key] = value.isoformat()
48
+ elif isinstance(value, dict):
49
+ clean_json(value)
50
+
51
+ return data
52
+
53
+ def write_json_file(path, data):
54
+ with open(path, "w") as f:
55
+ if data.get("query_policies", False):
56
+ del data["query_policies"]
57
+ clean = clean_json(data)
58
+ json.dump(clean, f, indent=2, default=str)
59
+
60
+ def write_file(path, data):
61
+ with open(path, "w") as f:
62
+ f.write(data)
63
+
64
+ def write_binary_file(path, data):
65
+ with open(path, "wb") as f:
66
+ f.write(data)
67
+
68
+ def process_attachments(session, space_folder):
69
+ attachments = session.exec(select(Attachments)).all()
70
+ for attachment in attachments:
71
+ subpath = subpath_checker(attachment.subpath)
72
+
73
+ parts = subpath.split('/')
74
+ parts.insert(-1, '.dm')
75
+ new_path = '/'.join(parts)
76
+
77
+ dir_path = f"{space_folder}/{attachment.space_name}{new_path}"
78
+ ensure_directory_exists(dir_path)
79
+
80
+ media_path = f"{dir_path}/attachments.{attachment.resource_type}"
81
+ ensure_directory_exists(media_path)
82
+ if attachment.payload.get("body", None) is not None:
83
+ if attachment.payload["content_type"] == 'json':
84
+ write_json_file(f"{media_path}/{attachment.shortname}.json", attachment.payload.get("body", {}))
85
+ attachment.payload["body"] = f"{attachment.shortname}.json"
86
+ else:
87
+ if attachment.media is None:
88
+ print(f"Warning: empty media for @{attachment.space_name}:{attachment.subpath}/{attachment.shortname}")
89
+ continue
90
+ write_binary_file(f"{media_path}/{attachment.payload['body']}", attachment.media)
91
+ _attachment = attachment.model_dump()
92
+
93
+ del _attachment["media"]
94
+ del _attachment["resource_type"]
95
+ write_json_file(f"{media_path}/meta.{attachment.shortname}.json", _attachment)
96
+
97
+ def process_entries(session, space_folder):
98
+ entries = session.exec(select(Entries)).all()
99
+ for entry in entries:
100
+ subpath = subpath_checker(entry.subpath)
101
+ dir_path = f"{space_folder}/{entry.space_name}{subpath}".replace("//", "/") # Ensure absolute path
102
+ ensure_directory_exists(dir_path)
103
+
104
+ if entry.resource_type == "folder":
105
+ dir_meta_path = f"{dir_path}/{entry.shortname}/.dm/".replace("//", "/")
106
+ ensure_directory_exists(dir_meta_path)
107
+ _entry = entry.model_dump()
108
+ body = None
109
+ if _entry.get("payload", None) is not None:
110
+ if _entry.get("payload", None).get("body", None) is not None:
111
+ body = _entry.get("payload", None)["body"]
112
+ _entry["payload"]["body"] = f"{entry.shortname}.json"
113
+ del _entry["space_name"]
114
+ del _entry["subpath"]
115
+ del _entry["resource_type"]
116
+
117
+ write_json_file(f"{dir_meta_path}/meta.folder.json", _entry)
118
+ if body is not None:
119
+ write_json_file(f"{dir_path}/{entry.shortname}.json", body)
120
+ continue
121
+
122
+ dir_meta_path = f"{dir_path}/.dm/{entry.shortname}".replace("//", "/")
123
+ ensure_directory_exists(dir_meta_path)
124
+
125
+ _entry = entry.model_dump()
126
+ del _entry["space_name"]
127
+ del _entry["subpath"]
128
+ del _entry["resource_type"]
129
+
130
+ if entry.payload:
131
+ if "content_type" not in entry.payload:
132
+ print(f"Warning : empty content type for @{entry.space_name}:{entry.subpath}/{entry.shortname}")
133
+ elif entry.payload["content_type"] == core.ContentType.json:
134
+
135
+ if _entry["payload"].get("body", None) is not None:
136
+ if isinstance(_entry["payload"].get("body", None), dict):
137
+ write_json_file(f"{dir_path}/{entry.shortname}.json", _entry["payload"].get("body", None))
138
+
139
+ _entry["payload"]["body"] = f"{entry.shortname}.json"
140
+
141
+ elif entry.payload["content_type"] == core.ContentType.html:
142
+ if _entry["payload"].get("body", None) is not None:
143
+ with open(f"{dir_path}/{entry.shortname}.html", "w", encoding="utf-8") as f:
144
+ f.write(_entry["payload"]["body"])
145
+ _entry["payload"]["body"] = f"{entry.shortname}.html"
146
+
147
+ elif entry.payload["content_type"] == core.ContentType.image:
148
+ if _entry["payload"].get("body", None) is not None:
149
+ body_data = _entry["payload"]["body"]
150
+
151
+ if isinstance(body_data, str):
152
+ if "." in body_data and any(body_data.lower().endswith(ext) for ext in ['.png', '.jpg', '.jpeg', '.gif', '.webp', '.svg', '.bmp']):
153
+ _entry["payload"]["body"] = body_data
154
+ print(f"Image reference: @{entry.space_name}:{entry.subpath}/{entry.shortname} -> {body_data}")
155
+ else:
156
+ try:
157
+ if len(body_data) % 4 == 0:
158
+ decoded_data = base64.b64decode(body_data, validate=True)
159
+
160
+ extension = "png"
161
+ if "content_sub_type" in entry.payload:
162
+ extension = entry.payload["content_sub_type"].lower()
163
+
164
+ filename = f"{entry.shortname}.{extension}"
165
+ with open(f"{dir_path}/{filename}", "wb") as f:
166
+ f.write(decoded_data)
167
+ _entry["payload"]["body"] = filename
168
+ else:
169
+ print(f"Warning: Invalid image data for @{entry.space_name}:{entry.subpath}/{entry.shortname}")
170
+ _entry["payload"]["body"] = body_data
171
+ except Exception as e:
172
+ print(f"Error processing image @{entry.space_name}:{entry.subpath}/{entry.shortname}: {e}")
173
+ _entry["payload"]["body"] = body_data
174
+ else:
175
+ extension = "png"
176
+ if "content_sub_type" in entry.payload:
177
+ extension = entry.payload["content_sub_type"].lower()
178
+
179
+ filename = f"{entry.shortname}.{extension}"
180
+ with open(f"{dir_path}/{filename}", "wb") as f:
181
+ f.write(body_data)
182
+ _entry["payload"]["body"] = filename
183
+
184
+
185
+ else:
186
+ print(f"Unprocessed content type({entry.payload['content_type']}): @{entry.space_name}:{entry.subpath}/{entry.shortname}")
187
+
188
+ if entry.resource_type != "ticket":
189
+ del _entry["state"]
190
+ del _entry["is_open"]
191
+ del _entry["reporter"]
192
+ del _entry["workflow_shortname"]
193
+ del _entry["collaborators"]
194
+ del _entry["resolution_reason"]
195
+
196
+ write_json_file(f"{dir_meta_path}/meta.{entry.resource_type}.json", _entry)
197
+
198
+ def process_users(session, space_folder):
199
+ users = session.exec(select(Users)).all()
200
+ dir_path = f"{space_folder}/management/users" # Ensure absolute path
201
+ for user in users:
202
+ dir_meta_path = f"{dir_path}/.dm/{user.shortname}"
203
+ ensure_directory_exists(dir_meta_path)
204
+
205
+ _user = user.model_dump()
206
+ del _user["space_name"]
207
+ del _user["resource_type"]
208
+ if _user.get("payload", None) and _user["payload"].get("body" , None):
209
+ write_json_file(
210
+ f"{dir_path}/{user.shortname}.json",
211
+ _user["payload"]["body"]
212
+ )
213
+ _user["payload"]["body"] = f"{user.shortname}.json"
214
+
215
+ write_json_file(f"{dir_meta_path}/meta.user.json", _user)
216
+
217
+ def process_roles(session, space_folder):
218
+ roles = session.exec(select(Roles)).all()
219
+ dir_path = f"{space_folder}/management/roles/.dm" # Ensure absolute path
220
+ for role in roles:
221
+ ensure_directory_exists(f"{dir_path}/{role.shortname}")
222
+
223
+ _role = role.model_dump()
224
+ del _role["space_name"]
225
+ del _role["subpath"]
226
+ del _role["resource_type"]
227
+
228
+ write_json_file(f"{dir_path}/{role.shortname}/meta.role.json", _role)
229
+
230
+ def process_permissions(session, space_folder):
231
+ permissions = session.exec(select(Permissions)).all()
232
+ dir_path = f"{space_folder}/management/permissions/.dm"
233
+ for permission in permissions:
234
+ ensure_directory_exists(f"{dir_path}/{permission.shortname}")
235
+
236
+ _permission = permission.model_dump()
237
+ del _permission["space_name"]
238
+ del _permission["subpath"]
239
+ del _permission["resource_type"]
240
+
241
+ write_json_file(f"{dir_path}/{permission.shortname}/meta.permission.json", _permission)
242
+
243
+ def process_histories(session, space_folder):
244
+ histories = session.exec(select(Histories)).all()
245
+ for history in histories:
246
+ dir_path = f"{space_folder}/{history.space_name}" # Ensure absolute path
247
+ ensure_directory_exists(dir_path)
248
+
249
+ file_path = f"{dir_path}{history.subpath}/.dm/{history.shortname}"
250
+ ensure_directory_exists(file_path)
251
+
252
+ _history_one: str = history.model_dump_json()
253
+ _history: dict = json.loads(_history_one)
254
+ _history["shortname"] = "history"
255
+
256
+ del _history["space_name"]
257
+ del _history["subpath"]
258
+ if _history.get("resource_type", None):
259
+ del _history["resource_type"]
260
+ with open(f"{file_path}/history.jsonl", "a+") as f:
261
+ f.write(json.dumps(_history) + "\n")
262
+
263
+ def process_spaces(session, space_folder):
264
+ spaces = session.exec(select(Spaces)).all()
265
+ for space in spaces:
266
+ dir_path = f"{space_folder}/{space.space_name}/.dm/"
267
+ ensure_directory_exists(dir_path)
268
+
269
+ _space = space.model_dump()
270
+ del _space["space_name"]
271
+ del _space["resource_type"]
272
+
273
+ write_json_file(f"{dir_path}/meta.space.json", _space)
274
+
275
+ async def export_data_with_query(query, user_shortname):
276
+ from utils.repository import serve_query
277
+
278
+ space_folder = os.path.relpath(str(settings.spaces_folder))
279
+
280
+ total, records = await serve_query(query, user_shortname)
281
+
282
+ with Session(get_engine()) as session:
283
+ space = session.exec(select(Spaces).where(col(Spaces.space_name) == query.space_name)).first()
284
+ if space:
285
+ dir_path = f"{space_folder}/{space.space_name}/.dm/"
286
+ ensure_directory_exists(dir_path)
287
+
288
+ _space = space.model_dump()
289
+ del _space["space_name"]
290
+ del _space["resource_type"]
291
+
292
+ write_json_file(f"{dir_path}/meta.space.json", _space)
293
+ if query.subpath and query.subpath != "/":
294
+ path_parts = query.subpath.strip("/").split("/")
295
+ current_path = ""
296
+
297
+ for part in path_parts:
298
+ current_path += f"/{part}"
299
+
300
+ folder = session.exec(select(Entries).where(
301
+ (Entries.space_name == query.space_name) &
302
+ (Entries.subpath == str(current_path.rsplit("/", 1)[0] or "/")) &
303
+ (Entries.shortname == part) &
304
+ (Entries.resource_type == "folder")
305
+ )).first()
306
+
307
+ if folder:
308
+ folder_subpath = subpath_checker(folder.subpath)
309
+ folder_dir_path = f"{space_folder}/{folder.space_name}{folder_subpath}".replace("//", "/")
310
+ ensure_directory_exists(folder_dir_path)
311
+
312
+ dir_meta_path = f"{folder_dir_path}/{folder.shortname}/.dm/".replace("//", "/")
313
+ ensure_directory_exists(dir_meta_path)
314
+
315
+ _folder = folder.model_dump()
316
+ _folder = {
317
+ **_folder,
318
+ **_folder.get("attributes", {})
319
+ }
320
+ if "attributes" in _folder:
321
+ del _folder["attributes"]
322
+ body = None
323
+ if _folder and _folder.get("payload", None) is not None:
324
+ if _folder and _folder.get("payload", {}).get("body", None) is not None:
325
+ body = _folder.get("payload", {}).get("body", None)
326
+ _folder["payload"]["body"] = f"{folder.shortname}.json"
327
+
328
+ del _folder["space_name"]
329
+ del _folder["subpath"]
330
+ del _folder["resource_type"]
331
+
332
+ write_json_file(f"{dir_meta_path}/meta.folder.json", _folder)
333
+ if body is not None:
334
+ write_json_file(f"{folder_dir_path}/{folder.shortname}.json", body)
335
+
336
+ for entry in records:
337
+ subpath = subpath_checker(entry.subpath)
338
+ dir_path = f"{space_folder}/{query.space_name}{subpath}".replace("//", "/") # Ensure absolute path
339
+ ensure_directory_exists(dir_path)
340
+
341
+ if entry.resource_type == "folder":
342
+ dir_meta_path = f"{dir_path}/{entry.shortname}/.dm/".replace("//", "/")
343
+ ensure_directory_exists(dir_meta_path)
344
+ _entry = entry.model_dump()
345
+ body = None
346
+ if _entry.get("payload", None) is not None:
347
+ if _entry.get("payload", {}).get("body", None) is not None:
348
+ body = _entry.get("payload",{}).get("body", None)
349
+ _entry["payload"]["body"] = f"{entry.shortname}.json"
350
+
351
+ del _entry["subpath"]
352
+ del _entry["resource_type"]
353
+
354
+ _entry = {
355
+ **_entry,
356
+ **_entry.get("attributes", {})
357
+ }
358
+ if "attributes" in _entry:
359
+ del _entry["attributes"]
360
+
361
+ write_json_file(f"{dir_meta_path}/meta.folder.json", _entry)
362
+ if body is not None:
363
+ write_json_file(f"{dir_path}/{entry.shortname}.json", body)
364
+ continue
365
+
366
+ dir_meta_path = f"{dir_path}/.dm/{entry.shortname}".replace("//", "/")
367
+ ensure_directory_exists(dir_meta_path)
368
+
369
+ _entry = entry.model_dump()
370
+ del _entry["subpath"]
371
+ del _entry["resource_type"]
372
+
373
+ if entry.attributes.get("payload"):
374
+ if entry.attributes.get("payload", {}).get("content_type") == core.ContentType.json:
375
+ if _entry.get("attributes",{}).get("payload",{}).get("body", None) is not None:
376
+ if isinstance( _entry.get("attributes",{}).get("payload").get("body", None), dict):
377
+ write_json_file(f"{dir_path}/{entry.shortname}.json", _entry.get("attributes",{}).get("payload").get("body", None))
378
+ _entry.get("attributes",{}).get("payload")["body"] = f"{entry.shortname}.json"
379
+
380
+ _entry = {
381
+ **_entry,
382
+ **_entry.get("attributes",{})
383
+ }
384
+ if "attributes" in _entry:
385
+ del _entry["attributes"]
386
+ if "attachments" in _entry:
387
+ del _entry["attachments"]
388
+
389
+ write_json_file(f"{dir_meta_path}/meta.{entry.resource_type}.json", _entry)
390
+
391
+ histories = session.exec(select(Histories).where(
392
+ (Histories.space_name == query.space_name) &
393
+ (Histories.subpath == entry.subpath) &
394
+ (Histories.shortname == entry.shortname)
395
+ )).all()
396
+
397
+ for history in histories:
398
+ file_path = f"{dir_path}/.dm/{entry.shortname}"
399
+ ensure_directory_exists(file_path)
400
+
401
+ _history_one: str = history.model_dump_json()
402
+ _history: dict = json.loads(_history_one)
403
+ _history["shortname"] = "history"
404
+
405
+ del _history["space_name"]
406
+ del _history["subpath"]
407
+ if _history.get("resource_type", None):
408
+ del _history["resource_type"]
409
+ with open(f"{file_path}/history.jsonl", "a+") as f:
410
+ f.write(json.dumps(_history) + "\n")
411
+
412
+ attachments = session.exec(select(Attachments).where(
413
+ (Attachments.space_name == query.space_name) &
414
+ (Attachments.subpath == f"/{entry.subpath}/{entry.shortname}")
415
+ )).all()
416
+
417
+ __attachment = None
418
+ for attachment in attachments:
419
+ __attachment = attachment
420
+ subpath = subpath_checker(attachment.subpath)
421
+
422
+ parts = subpath.split('/')
423
+ parts.insert(-1, '.dm')
424
+ new_path = '/'.join(parts)
425
+
426
+ dir_path = f"{space_folder}/{query.space_name}{new_path}"
427
+ ensure_directory_exists(dir_path)
428
+
429
+ media_path = f"{dir_path}/attachments.{attachment.resource_type}"
430
+ ensure_directory_exists(media_path)
431
+
432
+ attachment_body = None
433
+ if attachment.payload is not None:
434
+ if isinstance(attachment.payload, Payload):
435
+ attachment_body = attachment.payload.body
436
+ else:
437
+ attachment_body = attachment.payload["body"]
438
+
439
+ if attachment_body is not None:
440
+ if isinstance(attachment.payload, dict) and attachment.payload.get("content_type") == 'json':
441
+ write_json_file(f"{media_path}/{attachment.shortname}.json", attachment_body)
442
+ attachment.payload["body"] = f"{attachment.shortname}.json"
443
+ elif isinstance(attachment.payload, dict) and attachment.payload.get("content_type") == 'comment':
444
+ write_json_file(f"{media_path}/{attachment.shortname}.json", attachment_body)
445
+ attachment.payload["body"] = f"{attachment.shortname}.json"
446
+ else:
447
+ if attachment.media:
448
+ write_binary_file(f"{media_path}/{attachment_body}", attachment.media)
449
+
450
+ _attachment = attachment.model_dump()
451
+
452
+ del _attachment["media"]
453
+ del _attachment["resource_type"]
454
+ write_json_file(f"{media_path}/meta.{attachment.shortname}.json", _attachment)
455
+ else:
456
+ with open(f"{dir_meta_path}/meta.{entry.resource_type}.json", "r") as f:
457
+ entry_data = json.load(f)
458
+ if __attachment is not None:
459
+ if "payload" in entry_data:
460
+ entry_data["payload"] = __attachment.payload
461
+
462
+ return space_folder
463
+
464
+ def main():
465
+ space_folder = os.path.relpath(str(settings.spaces_folder))
466
+
467
+ with Session(get_engine()) as session:
468
+ print("Processing spaces...")
469
+ process_spaces(session, space_folder)
470
+ print("Processing entries...")
471
+ process_entries(session, space_folder)
472
+ print("Processing users...")
473
+ process_users(session, space_folder)
474
+ print("Processing roles...")
475
+ process_roles(session, space_folder)
476
+ print("Processing permissions...")
477
+ process_permissions(session, space_folder)
478
+ print("Processing attachments...")
479
+ process_attachments(session, space_folder)
480
+ print("Processing histories...")
481
+ process_histories(session, space_folder)
482
+
483
+
484
+ if __name__ == "__main__":
485
+ main()