dmart 1.4.40.post8__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 (489) hide show
  1. dmart/__init__.py +7 -0
  2. dmart/alembic/README +1 -0
  3. dmart/alembic/__init__.py +0 -0
  4. dmart/alembic/env.py +91 -0
  5. dmart/alembic/notes.txt +11 -0
  6. dmart/alembic/script.py.mako +28 -0
  7. dmart/alembic/scripts/__init__.py +0 -0
  8. dmart/alembic/scripts/calculate_checksums.py +77 -0
  9. dmart/alembic/scripts/migration_f7a4949eed19.py +28 -0
  10. dmart/alembic/versions/0f3d2b1a7c21_add_authz_materialized_views.py +87 -0
  11. dmart/alembic/versions/10d2041b94d4_last_checksum_history.py +62 -0
  12. dmart/alembic/versions/1cf4e1ee3cb8_ext_permission_with_filter_fields_values.py +33 -0
  13. dmart/alembic/versions/26bfe19b49d4_rm_failedloginattempts.py +42 -0
  14. dmart/alembic/versions/3c8bca2219cc_add_otp_table.py +38 -0
  15. dmart/alembic/versions/6675fd9dfe42_remove_unique_from_sessions_table.py +36 -0
  16. dmart/alembic/versions/71bc1df82e6a_adding_user_last_login_at.py +43 -0
  17. dmart/alembic/versions/74288ccbd3b5_initial.py +264 -0
  18. dmart/alembic/versions/7520a89a8467_rm_activesession_table.py +39 -0
  19. dmart/alembic/versions/848b623755a4_make_created_nd_updated_at_required.py +138 -0
  20. dmart/alembic/versions/8640dcbebf85_add_notes_to_users.py +32 -0
  21. dmart/alembic/versions/91c94250232a_adding_fk_on_owner_shortname.py +104 -0
  22. dmart/alembic/versions/98ecd6f56f9a_ext_meta_with_owner_group_shortname.py +66 -0
  23. dmart/alembic/versions/9aae9138c4ef_indexing_created_at_updated_at.py +80 -0
  24. dmart/alembic/versions/__init__.py +0 -0
  25. dmart/alembic/versions/b53f916b3f6d_json_to_jsonb.py +492 -0
  26. dmart/alembic/versions/eb5f1ec65156_adding_user_locked_to_device.py +36 -0
  27. dmart/alembic/versions/f7a4949eed19_adding_query_policies_to_meta.py +60 -0
  28. dmart/alembic.ini +117 -0
  29. dmart/api/__init__.py +0 -0
  30. dmart/api/info/__init__.py +0 -0
  31. dmart/api/info/router.py +109 -0
  32. dmart/api/managed/__init__.py +0 -0
  33. dmart/api/managed/router.py +1541 -0
  34. dmart/api/managed/utils.py +1879 -0
  35. dmart/api/public/__init__.py +0 -0
  36. dmart/api/public/router.py +758 -0
  37. dmart/api/qr/__init__.py +0 -0
  38. dmart/api/qr/router.py +108 -0
  39. dmart/api/user/__init__.py +0 -0
  40. dmart/api/user/model/__init__.py +0 -0
  41. dmart/api/user/model/errors.py +14 -0
  42. dmart/api/user/model/requests.py +165 -0
  43. dmart/api/user/model/responses.py +11 -0
  44. dmart/api/user/router.py +1413 -0
  45. dmart/api/user/service.py +270 -0
  46. dmart/bundler.py +52 -0
  47. dmart/cli.py +1133 -0
  48. dmart/config/__init__.py +0 -0
  49. dmart/config/channels.json +11 -0
  50. dmart/config/notification.json +17 -0
  51. dmart/config.env.sample +27 -0
  52. dmart/config.ini.sample +7 -0
  53. dmart/conftest.py +13 -0
  54. dmart/curl.sh +196 -0
  55. dmart/cxb/__init__.py +0 -0
  56. dmart/cxb/assets/@codemirror-Rn7_6DkE.js +10 -0
  57. dmart/cxb/assets/@edraj-CS4NwVbD.js +1 -0
  58. dmart/cxb/assets/@floating-ui-BwwcF-xh.js +1 -0
  59. dmart/cxb/assets/@formatjs-yKEsAtjs.js +1 -0
  60. dmart/cxb/assets/@fortawesome-DRW1UCdr.js +9 -0
  61. dmart/cxb/assets/@jsonquerylang-laKNoFFq.js +12 -0
  62. dmart/cxb/assets/@lezer-za4Q-8Ew.js +1 -0
  63. dmart/cxb/assets/@marijn-DXwl3gUT.js +1 -0
  64. dmart/cxb/assets/@popperjs-l0sNRNKZ.js +1 -0
  65. dmart/cxb/assets/@replit--ERk53eB.js +1 -0
  66. dmart/cxb/assets/@roxi-CGMFK4i8.js +6 -0
  67. dmart/cxb/assets/@typewriter-cCzskkIv.js +17 -0
  68. dmart/cxb/assets/@zerodevx-BlBZjKxu.js +1 -0
  69. dmart/cxb/assets/@zerodevx-CVEpe6WZ.css +1 -0
  70. dmart/cxb/assets/BreadCrumbLite-DAhOx38v.js +1 -0
  71. dmart/cxb/assets/EntryRenderer-CCqV8Rkg.js +32 -0
  72. dmart/cxb/assets/EntryRenderer-DXytdFp9.css +1 -0
  73. dmart/cxb/assets/ListView-BQelo7vZ.js +16 -0
  74. dmart/cxb/assets/ListView-U8of-_c-.css +1 -0
  75. dmart/cxb/assets/Prism--hMplq-p.js +3 -0
  76. dmart/cxb/assets/Prism-Uh6uStUw.css +1 -0
  77. dmart/cxb/assets/Table2Cols-BsbwicQm.js +1 -0
  78. dmart/cxb/assets/_..-BvT6vdHa.css +1 -0
  79. dmart/cxb/assets/_...404_-fuLH_rX9.js +2 -0
  80. dmart/cxb/assets/_...fallback_-Ba_NLmAE.js +1 -0
  81. dmart/cxb/assets/_module-3HrtKAWo.js +3 -0
  82. dmart/cxb/assets/_module-DFKFq0AM.js +4 -0
  83. dmart/cxb/assets/_module-Dgq0ZVtz.js +1 -0
  84. dmart/cxb/assets/ajv-Cpj98o6Y.js +1 -0
  85. dmart/cxb/assets/axios-CG2WSiiR.js +6 -0
  86. dmart/cxb/assets/clsx-B-dksMZM.js +1 -0
  87. dmart/cxb/assets/codemirror-wrapped-line-indent-DPhKvljI.js +1 -0
  88. dmart/cxb/assets/compare-C3AjiGFR.js +1 -0
  89. dmart/cxb/assets/compute-scroll-into-view-Bl8rNFhg.js +1 -0
  90. dmart/cxb/assets/consolite-DlCuI0F9.js +1 -0
  91. dmart/cxb/assets/crelt-C8TCjufn.js +1 -0
  92. dmart/cxb/assets/date-fns-l0sNRNKZ.js +1 -0
  93. dmart/cxb/assets/deepmerge-rn4rBaHU.js +1 -0
  94. dmart/cxb/assets/dmart_services-AL6-IdDE.js +1 -0
  95. dmart/cxb/assets/downloadFile-D08i0YDh.js +1 -0
  96. dmart/cxb/assets/easy-signal-BiPFIK3O.js +1 -0
  97. dmart/cxb/assets/esm-env-rsSWfq8L.js +1 -0
  98. dmart/cxb/assets/export-OF_rTiXu.js +1 -0
  99. dmart/cxb/assets/fast-deep-equal-l0sNRNKZ.js +1 -0
  100. dmart/cxb/assets/fast-diff-C-IidNf4.js +1 -0
  101. dmart/cxb/assets/fast-uri-l0sNRNKZ.js +1 -0
  102. dmart/cxb/assets/flowbite-svelte-BLvjb-sa.js +1 -0
  103. dmart/cxb/assets/flowbite-svelte-CD54FDqW.css +1 -0
  104. dmart/cxb/assets/flowbite-svelte-icons-BI8GVhw_.js +1 -0
  105. dmart/cxb/assets/github-slugger-CQ4oX9Ud.js +1 -0
  106. dmart/cxb/assets/global-igKv-1g9.js +1 -0
  107. dmart/cxb/assets/hookar-BMRD9G9H.js +1 -0
  108. dmart/cxb/assets/immutable-json-patch-DtRO2E_S.js +1 -0
  109. dmart/cxb/assets/import-1vE3gBat.js +1 -0
  110. dmart/cxb/assets/index-B-eTh-ZX.js +1 -0
  111. dmart/cxb/assets/index-BSsK-X71.js +1 -0
  112. dmart/cxb/assets/index-BVyxzKtH.js +1 -0
  113. dmart/cxb/assets/index-BdeNM69f.js +1 -0
  114. dmart/cxb/assets/index-CC-A1ipE.js +1 -0
  115. dmart/cxb/assets/index-CQohGiYB.js +1 -0
  116. dmart/cxb/assets/index-ChjnkpdZ.js +4 -0
  117. dmart/cxb/assets/index-DLP7csA4.js +1 -0
  118. dmart/cxb/assets/index-DTfhnhwd.js +1 -0
  119. dmart/cxb/assets/index-DdXRK7n9.js +2 -0
  120. dmart/cxb/assets/index-DtiCmB4o.js +1 -0
  121. dmart/cxb/assets/index-NBrXBlLA.css +2 -0
  122. dmart/cxb/assets/index-X1uNehO7.js +1 -0
  123. dmart/cxb/assets/index-nrQW6Nrr.js +1 -0
  124. dmart/cxb/assets/info-B986lRiM.js +1 -0
  125. dmart/cxb/assets/intl-messageformat-Dc5UU-HB.js +3 -0
  126. dmart/cxb/assets/jmespath-l0sNRNKZ.js +1 -0
  127. dmart/cxb/assets/json-schema-traverse-l0sNRNKZ.js +1 -0
  128. dmart/cxb/assets/json-source-map-DRgZidqy.js +5 -0
  129. dmart/cxb/assets/jsonpath-plus-l0sNRNKZ.js +1 -0
  130. dmart/cxb/assets/jsonrepair-B30Dx381.js +8 -0
  131. dmart/cxb/assets/lodash-es-DZVAA2ox.js +1 -0
  132. dmart/cxb/assets/marked-DKjyhwJX.js +56 -0
  133. dmart/cxb/assets/marked-gfm-heading-id-U5zO829x.js +2 -0
  134. dmart/cxb/assets/marked-mangle-CDMeiHC6.js +1 -0
  135. dmart/cxb/assets/memoize-one-BdPwpGay.js +1 -0
  136. dmart/cxb/assets/natural-compare-lite-Bg2Xcf-o.js +7 -0
  137. dmart/cxb/assets/pagination-svelte-D5CyoiE_.js +13 -0
  138. dmart/cxb/assets/pagination-svelte-v10nAbbM.css +1 -0
  139. dmart/cxb/assets/plantuml-encoder-C47mzt9T.js +1 -0
  140. dmart/cxb/assets/prismjs-DTUiLGJu.js +9 -0
  141. dmart/cxb/assets/profile-BUf-tKMe.js +1 -0
  142. dmart/cxb/assets/query-CNmXTsgf.js +1 -0
  143. dmart/cxb/assets/queryHelpers-C9iBWwqe.js +1 -0
  144. dmart/cxb/assets/scroll-into-view-if-needed-KR58zyjF.js +1 -0
  145. dmart/cxb/assets/spaces-0oyGvpii.js +1 -0
  146. dmart/cxb/assets/style-mod-Bs6eFhZE.js +3 -0
  147. dmart/cxb/assets/svelte-B2XmcTi_.js +4 -0
  148. dmart/cxb/assets/svelte-awesome-COLlx0DN.css +1 -0
  149. dmart/cxb/assets/svelte-awesome-DhnMA6Q_.js +1 -0
  150. dmart/cxb/assets/svelte-datatables-net-CY7LBj6I.js +1 -0
  151. dmart/cxb/assets/svelte-floating-ui-BlS3sOAQ.js +1 -0
  152. dmart/cxb/assets/svelte-i18n-CT2KkQaN.js +3 -0
  153. dmart/cxb/assets/svelte-jsoneditor-BzfX6Usi.css +1 -0
  154. dmart/cxb/assets/svelte-jsoneditor-CUGSvWId.js +25 -0
  155. dmart/cxb/assets/svelte-select-CegQKzqH.css +1 -0
  156. dmart/cxb/assets/svelte-select-CjHAt_85.js +6 -0
  157. dmart/cxb/assets/tailwind-merge-CJvxXMcu.js +1 -0
  158. dmart/cxb/assets/tailwind-variants-Cj20BoQ3.js +1 -0
  159. dmart/cxb/assets/toast-B9WDyfyI.js +1 -0
  160. dmart/cxb/assets/tslib-pJfR_DrR.js +1 -0
  161. dmart/cxb/assets/typewriter-editor-DkTVIJdm.js +25 -0
  162. dmart/cxb/assets/user-DeK_NB5v.js +1 -0
  163. dmart/cxb/assets/vanilla-picker-l5rcX3cq.js +8 -0
  164. dmart/cxb/assets/w3c-keyname-Vcq4gwWv.js +1 -0
  165. dmart/cxb/config.json +11 -0
  166. dmart/cxb/config.sample.json +11 -0
  167. dmart/cxb/favicon.ico +0 -0
  168. dmart/cxb/favicon.png +0 -0
  169. dmart/cxb/index.html +28 -0
  170. dmart/data_adapters/__init__.py +0 -0
  171. dmart/data_adapters/adapter.py +16 -0
  172. dmart/data_adapters/base_data_adapter.py +467 -0
  173. dmart/data_adapters/file/__init__.py +0 -0
  174. dmart/data_adapters/file/adapter.py +2043 -0
  175. dmart/data_adapters/file/adapter_helpers.py +1013 -0
  176. dmart/data_adapters/file/archive.py +150 -0
  177. dmart/data_adapters/file/create_index.py +331 -0
  178. dmart/data_adapters/file/create_users_folders.py +52 -0
  179. dmart/data_adapters/file/custom_validations.py +68 -0
  180. dmart/data_adapters/file/drop_index.py +40 -0
  181. dmart/data_adapters/file/health_check.py +560 -0
  182. dmart/data_adapters/file/redis_services.py +1110 -0
  183. dmart/data_adapters/helpers.py +27 -0
  184. dmart/data_adapters/sql/__init__.py +0 -0
  185. dmart/data_adapters/sql/adapter.py +3218 -0
  186. dmart/data_adapters/sql/adapter_helpers.py +491 -0
  187. dmart/data_adapters/sql/create_tables.py +451 -0
  188. dmart/data_adapters/sql/create_users_folders.py +53 -0
  189. dmart/data_adapters/sql/db_to_json_migration.py +485 -0
  190. dmart/data_adapters/sql/health_check_sql.py +232 -0
  191. dmart/data_adapters/sql/json_to_db_migration.py +454 -0
  192. dmart/data_adapters/sql/update_query_policies.py +101 -0
  193. dmart/data_generator.py +81 -0
  194. dmart/dmart.py +761 -0
  195. dmart/get_settings.py +7 -0
  196. dmart/hypercorn_config.toml +3 -0
  197. dmart/info.json +1 -0
  198. dmart/languages/__init__.py +0 -0
  199. dmart/languages/arabic.json +15 -0
  200. dmart/languages/english.json +16 -0
  201. dmart/languages/kurdish.json +14 -0
  202. dmart/languages/loader.py +12 -0
  203. dmart/login_creds.sh +7 -0
  204. dmart/login_creds.sh.sample +7 -0
  205. dmart/main.py +563 -0
  206. dmart/manifest.sh +12 -0
  207. dmart/migrate.py +24 -0
  208. dmart/models/__init__.py +0 -0
  209. dmart/models/api.py +203 -0
  210. dmart/models/core.py +597 -0
  211. dmart/models/enums.py +255 -0
  212. dmart/password_gen.py +8 -0
  213. dmart/plugins/__init__.py +0 -0
  214. dmart/plugins/action_log/__init__.py +0 -0
  215. dmart/plugins/action_log/config.json +13 -0
  216. dmart/plugins/action_log/plugin.py +121 -0
  217. dmart/plugins/admin_notification_sender/__init__.py +0 -0
  218. dmart/plugins/admin_notification_sender/config.json +13 -0
  219. dmart/plugins/admin_notification_sender/plugin.py +124 -0
  220. dmart/plugins/ldap_manager/__init__.py +0 -0
  221. dmart/plugins/ldap_manager/config.json +12 -0
  222. dmart/plugins/ldap_manager/dmart.schema +146 -0
  223. dmart/plugins/ldap_manager/plugin.py +100 -0
  224. dmart/plugins/ldap_manager/slapd.conf +53 -0
  225. dmart/plugins/local_notification/__init__.py +0 -0
  226. dmart/plugins/local_notification/config.json +13 -0
  227. dmart/plugins/local_notification/plugin.py +123 -0
  228. dmart/plugins/realtime_updates_notifier/__init__.py +0 -0
  229. dmart/plugins/realtime_updates_notifier/config.json +12 -0
  230. dmart/plugins/realtime_updates_notifier/plugin.py +58 -0
  231. dmart/plugins/redis_db_update/__init__.py +0 -0
  232. dmart/plugins/redis_db_update/config.json +13 -0
  233. dmart/plugins/redis_db_update/plugin.py +188 -0
  234. dmart/plugins/resource_folders_creation/__init__.py +0 -0
  235. dmart/plugins/resource_folders_creation/config.json +12 -0
  236. dmart/plugins/resource_folders_creation/plugin.py +81 -0
  237. dmart/plugins/system_notification_sender/__init__.py +0 -0
  238. dmart/plugins/system_notification_sender/config.json +13 -0
  239. dmart/plugins/system_notification_sender/plugin.py +188 -0
  240. dmart/plugins/update_access_controls/__init__.py +0 -0
  241. dmart/plugins/update_access_controls/config.json +12 -0
  242. dmart/plugins/update_access_controls/plugin.py +9 -0
  243. dmart/publish.sh +57 -0
  244. dmart/pylint.sh +16 -0
  245. dmart/pyrightconfig.json +7 -0
  246. dmart/redis_connections.sh +13 -0
  247. dmart/reload.sh +56 -0
  248. dmart/run.sh +3 -0
  249. dmart/run_notification_campaign.py +85 -0
  250. dmart/sample/spaces/applications/.dm/meta.space.json +30 -0
  251. dmart/sample/spaces/applications/api/.dm/meta.folder.json +1 -0
  252. dmart/sample/spaces/applications/api/.dm/query_all_applications/meta.content.json +1 -0
  253. dmart/sample/spaces/applications/api/.dm/test_by_saad/attachments.media/meta.warframe.json +1 -0
  254. dmart/sample/spaces/applications/api/.dm/test_by_saad/attachments.media/warframe.png +0 -0
  255. dmart/sample/spaces/applications/api/.dm/test_by_saad/meta.content.json +1 -0
  256. dmart/sample/spaces/applications/api/.dm/user_profile/meta.content.json +1 -0
  257. dmart/sample/spaces/applications/api/applications/.dm/create_log/meta.content.json +1 -0
  258. dmart/sample/spaces/applications/api/applications/.dm/create_public_logs/meta.content.json +1 -0
  259. dmart/sample/spaces/applications/api/applications/.dm/meta.folder.json +1 -0
  260. dmart/sample/spaces/applications/api/applications/.dm/query_all_translated_data/meta.content.json +1 -0
  261. dmart/sample/spaces/applications/api/applications/.dm/query_logs/meta.content.json +1 -0
  262. dmart/sample/spaces/applications/api/applications/.dm/query_translated_enums/meta.content.json +1 -0
  263. dmart/sample/spaces/applications/api/applications/.dm/query_translated_others/meta.content.json +1 -0
  264. dmart/sample/spaces/applications/api/applications/.dm/query_translated_resolution/meta.content.json +1 -0
  265. dmart/sample/spaces/applications/api/applications/create_log.json +1 -0
  266. dmart/sample/spaces/applications/api/applications/create_public_logs.json +1 -0
  267. dmart/sample/spaces/applications/api/applications/query_all_translated_data.json +1 -0
  268. dmart/sample/spaces/applications/api/applications/query_logs.json +1 -0
  269. dmart/sample/spaces/applications/api/applications/query_translated_enums.json +1 -0
  270. dmart/sample/spaces/applications/api/applications/query_translated_others.json +1 -0
  271. dmart/sample/spaces/applications/api/applications/query_translated_resolution.json +1 -0
  272. dmart/sample/spaces/applications/api/applications.json +1 -0
  273. dmart/sample/spaces/applications/api/management/.dm/create_subaccount/meta.content.json +1 -0
  274. dmart/sample/spaces/applications/api/management/.dm/meta.folder.json +1 -0
  275. dmart/sample/spaces/applications/api/management/.dm/update_password/meta.content.json +1 -0
  276. dmart/sample/spaces/applications/api/management/create_subaccount.json +53 -0
  277. dmart/sample/spaces/applications/api/management/update_password.json +1 -0
  278. dmart/sample/spaces/applications/api/management.json +1 -0
  279. dmart/sample/spaces/applications/api/query_all_applications.json +15 -0
  280. dmart/sample/spaces/applications/api/test_by_saad.json +1 -0
  281. dmart/sample/spaces/applications/api/user/.dm/meta.folder.json +1 -0
  282. dmart/sample/spaces/applications/api/user/.dm/test_by_saad/meta.content.json +1 -0
  283. dmart/sample/spaces/applications/api/user/.dm/user_profile/meta.content.json +1 -0
  284. dmart/sample/spaces/applications/api/user/test_by_saad.json +1 -0
  285. dmart/sample/spaces/applications/api/user/user_profile.json +1 -0
  286. dmart/sample/spaces/applications/api/user_profile.json +1 -0
  287. dmart/sample/spaces/applications/api.json +1 -0
  288. dmart/sample/spaces/applications/collections/.dm/meta.folder.json +19 -0
  289. dmart/sample/spaces/applications/collections.json +1 -0
  290. dmart/sample/spaces/applications/configurations/.dm/meta.folder.json +1 -0
  291. dmart/sample/spaces/applications/configurations/time_out.json +1 -0
  292. dmart/sample/spaces/applications/configurations.json +19 -0
  293. dmart/sample/spaces/applications/errors.json +1 -0
  294. dmart/sample/spaces/applications/logs/.dm/meta.folder.json +1 -0
  295. dmart/sample/spaces/applications/logs.json +1 -0
  296. dmart/sample/spaces/applications/queries/.dm/meta.folder.json +1 -0
  297. dmart/sample/spaces/applications/queries/.dm/order/meta.content.json +1 -0
  298. dmart/sample/spaces/applications/queries/order.json +1 -0
  299. dmart/sample/spaces/applications/queries.json +1 -0
  300. dmart/sample/spaces/applications/schema/.dm/api/meta.schema.json +1 -0
  301. dmart/sample/spaces/applications/schema/.dm/configuration/meta.schema.json +1 -0
  302. dmart/sample/spaces/applications/schema/.dm/error/meta.schema.json +1 -0
  303. dmart/sample/spaces/applications/schema/.dm/log/meta.schema.json +1 -0
  304. dmart/sample/spaces/applications/schema/.dm/meta.folder.json +1 -0
  305. dmart/sample/spaces/applications/schema/.dm/query/meta.schema.json +16 -0
  306. dmart/sample/spaces/applications/schema/.dm/translation/meta.schema.json +1 -0
  307. dmart/sample/spaces/applications/schema/api.json +28 -0
  308. dmart/sample/spaces/applications/schema/configuration.json +1 -0
  309. dmart/sample/spaces/applications/schema/error.json +43 -0
  310. dmart/sample/spaces/applications/schema/log.json +1 -0
  311. dmart/sample/spaces/applications/schema/query.json +118 -0
  312. dmart/sample/spaces/applications/schema/translation.json +26 -0
  313. dmart/sample/spaces/applications/schema.json +1 -0
  314. dmart/sample/spaces/applications/translations/.dm/meta.folder.json +1 -0
  315. dmart/sample/spaces/applications/translations.json +1 -0
  316. dmart/sample/spaces/archive/.dm/meta.space.json +27 -0
  317. dmart/sample/spaces/custom_plugins/dummy/__pycache__/plugin.cpython-314.pyc +0 -0
  318. dmart/sample/spaces/custom_plugins/dummy/config.json +28 -0
  319. dmart/sample/spaces/custom_plugins/dummy/plugin.py +6 -0
  320. dmart/sample/spaces/custom_plugins/missed_entry/config.json +12 -0
  321. dmart/sample/spaces/custom_plugins/missed_entry/plugin.py +119 -0
  322. dmart/sample/spaces/custom_plugins/own_changed_notification/__pycache__/plugin.cpython-314.pyc +0 -0
  323. dmart/sample/spaces/custom_plugins/own_changed_notification/config.json +12 -0
  324. dmart/sample/spaces/custom_plugins/own_changed_notification/plugin.py +65 -0
  325. dmart/sample/spaces/custom_plugins/reports_stats/config.json +14 -0
  326. dmart/sample/spaces/custom_plugins/reports_stats/plugin.py +82 -0
  327. dmart/sample/spaces/custom_plugins/system_notification_sender/config.json +22 -0
  328. dmart/sample/spaces/custom_plugins/system_notification_sender/notification.py +268 -0
  329. dmart/sample/spaces/custom_plugins/system_notification_sender/plugin.py +98 -0
  330. dmart/sample/spaces/management/.dm/events.jsonl +32 -0
  331. dmart/sample/spaces/management/.dm/meta.space.json +48 -0
  332. dmart/sample/spaces/management/.dm/notifications/attachments.view.json/admin.json +36 -0
  333. dmart/sample/spaces/management/.dm/notifications/attachments.view.json/meta.admin.json +1 -0
  334. dmart/sample/spaces/management/.dm/notifications/attachments.view.json/meta.system.json +1 -0
  335. dmart/sample/spaces/management/.dm/notifications/attachments.view.json/system.json +32 -0
  336. dmart/sample/spaces/management/collections/.dm/meta.folder.json +1 -0
  337. dmart/sample/spaces/management/collections.json +1 -0
  338. dmart/sample/spaces/management/groups/.dm/meta.folder.json +1 -0
  339. dmart/sample/spaces/management/groups.json +1 -0
  340. dmart/sample/spaces/management/health_check/.dm/meta.folder.json +1 -0
  341. dmart/sample/spaces/management/health_check.json +1 -0
  342. dmart/sample/spaces/management/notifications/.dm/meta.folder.json +1 -0
  343. dmart/sample/spaces/management/notifications/admin/.dm/meta.folder.json +9 -0
  344. dmart/sample/spaces/management/notifications/system/.dm/meta.folder.json +9 -0
  345. dmart/sample/spaces/management/notifications.json +1 -0
  346. dmart/sample/spaces/management/permissions/.dm/access_applications/meta.permission.json +31 -0
  347. dmart/sample/spaces/management/permissions/.dm/access_applications_world/meta.permission.json +31 -0
  348. dmart/sample/spaces/management/permissions/.dm/access_messages/meta.permission.json +23 -0
  349. dmart/sample/spaces/management/permissions/.dm/access_personal/meta.permission.json +40 -0
  350. dmart/sample/spaces/management/permissions/.dm/access_protected/meta.permission.json +33 -0
  351. dmart/sample/spaces/management/permissions/.dm/access_public/meta.permission.json +24 -0
  352. dmart/sample/spaces/management/permissions/.dm/browse_all_folders/meta.permission.json +23 -0
  353. dmart/sample/spaces/management/permissions/.dm/create_log/meta.permission.json +24 -0
  354. dmart/sample/spaces/management/permissions/.dm/interviewer/meta.permission.json +1 -0
  355. dmart/sample/spaces/management/permissions/.dm/manage_applications/meta.permission.json +1 -0
  356. dmart/sample/spaces/management/permissions/.dm/manage_debug/meta.permission.json +25 -0
  357. dmart/sample/spaces/management/permissions/.dm/manage_spaces/meta.permission.json +24 -0
  358. dmart/sample/spaces/management/permissions/.dm/meta.folder.json +1 -0
  359. dmart/sample/spaces/management/permissions/.dm/rules_management_default/meta.permission.json +32 -0
  360. dmart/sample/spaces/management/permissions/.dm/super_manager/meta.permission.json +52 -0
  361. dmart/sample/spaces/management/permissions/.dm/view_activity_log/meta.permission.json +26 -0
  362. dmart/sample/spaces/management/permissions/.dm/view_collections/meta.permission.json +29 -0
  363. dmart/sample/spaces/management/permissions/.dm/view_logs/meta.permission.json +30 -0
  364. dmart/sample/spaces/management/permissions/.dm/view_roles/meta.permission.json +29 -0
  365. dmart/sample/spaces/management/permissions/.dm/view_users/meta.permission.json +25 -0
  366. dmart/sample/spaces/management/permissions/.dm/view_world/meta.permission.json +31 -0
  367. dmart/sample/spaces/management/permissions/.dm/world/meta.permission.json +35 -0
  368. dmart/sample/spaces/management/permissions.json +1 -0
  369. dmart/sample/spaces/management/requests.json +1 -0
  370. dmart/sample/spaces/management/roles/.dm/dummy/meta.role.json +12 -0
  371. dmart/sample/spaces/management/roles/.dm/logged_in/meta.role.json +18 -0
  372. dmart/sample/spaces/management/roles/.dm/manager/meta.role.json +13 -0
  373. dmart/sample/spaces/management/roles/.dm/meta.folder.json +1 -0
  374. dmart/sample/spaces/management/roles/.dm/moderator/meta.role.json +13 -0
  375. dmart/sample/spaces/management/roles/.dm/super_admin/meta.role.json +14 -0
  376. dmart/sample/spaces/management/roles/.dm/test_role/meta.role.json +13 -0
  377. dmart/sample/spaces/management/roles/.dm/world/meta.role.json +15 -0
  378. dmart/sample/spaces/management/roles.json +1 -0
  379. dmart/sample/spaces/management/schema/.dm/admin_notification_request/attachments.media/meta.ui_schema.json +10 -0
  380. dmart/sample/spaces/management/schema/.dm/admin_notification_request/attachments.media/ui_schema.json +32 -0
  381. dmart/sample/spaces/management/schema/.dm/admin_notification_request/meta.schema.json +1 -0
  382. dmart/sample/spaces/management/schema/.dm/api/meta.schema.json +1 -0
  383. dmart/sample/spaces/management/schema/.dm/folder_rendering/meta.schema.json +1 -0
  384. dmart/sample/spaces/management/schema/.dm/health_check/meta.schema.json +17 -0
  385. dmart/sample/spaces/management/schema/.dm/meta.folder.json +1 -0
  386. dmart/sample/spaces/management/schema/.dm/meta_schema/meta.schema.json +1 -0
  387. dmart/sample/spaces/management/schema/.dm/metafile/meta.schema.json +14 -0
  388. dmart/sample/spaces/management/schema/.dm/notification/meta.schema.json +1 -0
  389. dmart/sample/spaces/management/schema/.dm/system_notification_request/attachments.media/meta.ui_schema.json +10 -0
  390. dmart/sample/spaces/management/schema/.dm/system_notification_request/attachments.media/ui_schema.json +32 -0
  391. dmart/sample/spaces/management/schema/.dm/system_notification_request/meta.schema.json +1 -0
  392. dmart/sample/spaces/management/schema/.dm/view/meta.schema.json +1 -0
  393. dmart/sample/spaces/management/schema/.dm/workflow/meta.schema.json +1 -0
  394. dmart/sample/spaces/management/schema/admin_notification_request.json +89 -0
  395. dmart/sample/spaces/management/schema/api.json +1 -0
  396. dmart/sample/spaces/management/schema/folder_rendering.json +238 -0
  397. dmart/sample/spaces/management/schema/health_check.json +8 -0
  398. dmart/sample/spaces/management/schema/meta_schema.json +74 -0
  399. dmart/sample/spaces/management/schema/metafile.json +153 -0
  400. dmart/sample/spaces/management/schema/notification.json +28 -0
  401. dmart/sample/spaces/management/schema/system_notification_request.json +57 -0
  402. dmart/sample/spaces/management/schema/view.json +23 -0
  403. dmart/sample/spaces/management/schema/workflow.json +87 -0
  404. dmart/sample/spaces/management/schema.json +1 -0
  405. dmart/sample/spaces/management/users/.dm/alibaba/meta.user.json +23 -0
  406. dmart/sample/spaces/management/users/.dm/anonymous/meta.user.json +18 -0
  407. dmart/sample/spaces/management/users/.dm/dmart/meta.user.json +26 -0
  408. dmart/sample/spaces/management/users/.dm/meta.folder.json +14 -0
  409. dmart/sample/spaces/management/workflows/.dm/channel/meta.content.json +1 -0
  410. dmart/sample/spaces/management/workflows/.dm/meta.folder.json +1 -0
  411. dmart/sample/spaces/management/workflows/channel.json +148 -0
  412. dmart/sample/spaces/management/workflows.json +1 -0
  413. dmart/sample/spaces/maqola/.dm/meta.space.json +33 -0
  414. dmart/sample/spaces/personal/.dm/meta.space.json +24 -0
  415. dmart/sample/spaces/personal/people/.dm/meta.folder.json +1 -0
  416. dmart/sample/spaces/personal/people/dmart/.dm/meta.folder.json +1 -0
  417. dmart/sample/spaces/personal/people/dmart/messages/.dm/0b5f7e7f/meta.content.json +1 -0
  418. dmart/sample/spaces/personal/people/dmart/messages/.dm/meta.folder.json +1 -0
  419. dmart/sample/spaces/personal/people/dmart/messages/.dm/mytest/meta.content.json +1 -0
  420. dmart/sample/spaces/personal/people/dmart/messages/0b5f7e7f.json +1 -0
  421. dmart/sample/spaces/personal/people/dmart/messages/mytest.json +1 -0
  422. dmart/sample/spaces/personal/people/dmart/notifications/.dm/meta.folder.json +1 -0
  423. dmart/sample/spaces/personal/people/dmart/private/.dm/inner/meta.content.json +1 -0
  424. dmart/sample/spaces/personal/people/dmart/private/.dm/meta.folder.json +1 -0
  425. dmart/sample/spaces/personal/people/dmart/private/inner.json +1 -0
  426. dmart/sample/spaces/personal/people/dmart/protected/.dm/avatar/meta.content.json +1 -0
  427. dmart/sample/spaces/personal/people/dmart/protected/.dm/meta.folder.json +1 -0
  428. dmart/sample/spaces/personal/people/dmart/protected/avatar.png +0 -0
  429. dmart/sample/spaces/personal/people/dmart/public/.dm/meta.folder.json +1 -0
  430. dmart/sample/test/.gitignore +2 -0
  431. dmart/sample/test/createcontent.json +9 -0
  432. dmart/sample/test/createmedia.json +9 -0
  433. dmart/sample/test/createmedia_entry.json +6 -0
  434. dmart/sample/test/createschema.json +8 -0
  435. dmart/sample/test/createschemawork.json +11 -0
  436. dmart/sample/test/createticket.json +13 -0
  437. dmart/sample/test/data.json +4 -0
  438. dmart/sample/test/deletecontent.json +12 -0
  439. dmart/sample/test/logo.jpeg +0 -0
  440. dmart/sample/test/my.jpg +0 -0
  441. dmart/sample/test/myticket.json +23 -0
  442. dmart/sample/test/resources.csv +12 -0
  443. dmart/sample/test/schema.json +16 -0
  444. dmart/sample/test/temp.json +1 -0
  445. dmart/sample/test/test.dmart +45 -0
  446. dmart/sample/test/ticket_schema.json +23 -0
  447. dmart/sample/test/ticket_workflow.json +85 -0
  448. dmart/sample/test/ticketbody.json +4 -0
  449. dmart/sample/test/ticketcontent.json +14 -0
  450. dmart/sample/test/updatecontent.json +20 -0
  451. dmart/sample/test/workflow_schema.json +68 -0
  452. dmart/scheduled_notification_handler.py +121 -0
  453. dmart/schema_migration.py +208 -0
  454. dmart/schema_modulate.py +192 -0
  455. dmart/set_admin_passwd.py +75 -0
  456. dmart/sync.py +202 -0
  457. dmart/test_utils.py +34 -0
  458. dmart/utils/__init__.py +0 -0
  459. dmart/utils/access_control.py +306 -0
  460. dmart/utils/async_request.py +8 -0
  461. dmart/utils/exporter.py +309 -0
  462. dmart/utils/firebase_notifier.py +57 -0
  463. dmart/utils/generate_email.py +37 -0
  464. dmart/utils/helpers.py +352 -0
  465. dmart/utils/hypercorn_config.py +12 -0
  466. dmart/utils/internal_error_code.py +60 -0
  467. dmart/utils/jwt.py +124 -0
  468. dmart/utils/logger.py +167 -0
  469. dmart/utils/middleware.py +99 -0
  470. dmart/utils/notification.py +75 -0
  471. dmart/utils/password_hashing.py +16 -0
  472. dmart/utils/plugin_manager.py +202 -0
  473. dmart/utils/query_policies_helper.py +128 -0
  474. dmart/utils/regex.py +44 -0
  475. dmart/utils/repository.py +529 -0
  476. dmart/utils/router_helper.py +19 -0
  477. dmart/utils/settings.py +212 -0
  478. dmart/utils/sms_notifier.py +21 -0
  479. dmart/utils/social_sso.py +67 -0
  480. dmart/utils/templates/activation.html.j2 +26 -0
  481. dmart/utils/templates/reminder.html.j2 +17 -0
  482. dmart/utils/ticket_sys_utils.py +203 -0
  483. dmart/utils/web_notifier.py +29 -0
  484. dmart/websocket.py +231 -0
  485. dmart-1.4.40.post8.dist-info/METADATA +75 -0
  486. dmart-1.4.40.post8.dist-info/RECORD +489 -0
  487. dmart-1.4.40.post8.dist-info/WHEEL +5 -0
  488. dmart-1.4.40.post8.dist-info/entry_points.txt +2 -0
  489. dmart-1.4.40.post8.dist-info/top_level.txt +1 -0
@@ -0,0 +1,232 @@
1
+ #!/usr/bin/env -S BACKEND_ENV=config.env python3
2
+
3
+ import argparse
4
+ import asyncio
5
+
6
+ from jsonschema import ValidationError
7
+ from jsonschema.validators import Draft7Validator
8
+ import json
9
+ import time
10
+ from sqlmodel import select, col, delete
11
+ from adapter import SQLAdapter
12
+ from data_adapters.adapter import data_adapter as db
13
+ from data_adapters.sql.create_tables import Entries, Spaces
14
+ from typing import Any
15
+
16
+ from models import core, api
17
+ from models.enums import ContentType, RequestType, ResourceType
18
+ from api.managed.router import serve_request
19
+
20
+
21
+ duplicated_entries : dict= {}
22
+
23
+ key_entries: dict = {}
24
+ MAX_INVALID_SIZE = 100
25
+
26
+ # {"space_name": {"schema_name": SCHEMA_DATA_DICT}}
27
+ spaces_schemas: dict[str, dict[str, dict]] = {}
28
+
29
+
30
+ async def main(health_type: str, space_param: str, schemas_param: list):
31
+ async with SQLAdapter().get_session() as session:
32
+ session.execute(
33
+ delete(Entries).where(col(Entries.subpath) == "/health_check")
34
+ )
35
+ await session.commit()
36
+
37
+ health_type = "hard" if health_type is None else health_type
38
+ space_param = "all" if space_param is None else space_param
39
+
40
+ if health_type not in ["soft", "hard"]:
41
+ print("Wrong mode specify [soft or hard]")
42
+ return
43
+
44
+ spaces = await db.get_spaces()
45
+ spaces_names : list = []
46
+
47
+ if space_param != "all":
48
+ if space_param not in spaces.keys():
49
+ print(f"space name {space_param} is not found")
50
+ return
51
+ spaces_names = [spaces[space_param]]
52
+ else:
53
+ spaces_names = list(spaces.keys())
54
+
55
+ if health_type == "soft":
56
+ pass
57
+ elif health_type == "hard":
58
+ for space in spaces_names:
59
+ await hard_space_check(space)
60
+
61
+
62
+ async def hard_space_check(space):
63
+ async with SQLAdapter().get_session() as session:
64
+ sql_stm = select(Entries).where(col(Entries.space_name) == space)
65
+ _result = session.exec(sql_stm).all()
66
+ _result = [r[0] for r in _result]
67
+ entries = list(_result)
68
+ folders_report: dict[str, dict[str, Any]] = {}
69
+
70
+ _sql_stm = select(Spaces).where(col(Spaces.shortname) == space)
71
+ target_space: Spaces | None = session.exec(_sql_stm).first()
72
+ if target_space:
73
+ schema_data_space: Entries | None = session.exec(
74
+ select(Entries)
75
+ .where(Entries.shortname == 'metafile')
76
+ .where(Entries.subpath == "/schema")
77
+ ).first()
78
+ if "/" not in folders_report:
79
+ folders_report["/" ] = {
80
+ "valid_entries": 0,
81
+ }
82
+
83
+ if schema_data_space and schema_data_space.payload:
84
+ try:
85
+ if isinstance(schema_data_space.payload, dict):
86
+ Draft7Validator(
87
+ schema_data_space.payload["body"]
88
+ ).validate(
89
+ json.loads(target_space.model_dump_json())
90
+ )
91
+ folders_report['/']["valid_entries"] += 1
92
+ except ValidationError as e:
93
+ issue = {
94
+ "issues": ["payload"],
95
+ "uuid": str(target_space.uuid),
96
+ "shortname": target_space.shortname,
97
+ "resource_type": 'space',
98
+ "exception": str(e),
99
+ }
100
+ if folders_report['/'].get("invalid_entries", None) is None:
101
+ folders_report['/']["invalid_entries"] = []
102
+ folders_report['/']["invalid_entries"] = [
103
+ *folders_report['/']["invalid_entries"],
104
+ issue
105
+ ]
106
+
107
+ for entry in entries:
108
+ subpath = entry.subpath[1:]
109
+ if subpath == "":
110
+ subpath = "/"
111
+
112
+ payload : core.Payload
113
+ if entry.payload and isinstance(entry.payload, dict):
114
+ try:
115
+ payload = core.Payload.model_validate(entry.payload)
116
+ except Exception as e:
117
+ issue = {
118
+ "issues": ["payload"],
119
+ "uuid": str(entry.uuid),
120
+ "shortname": entry.shortname,
121
+ "resource_type": 'space',
122
+ "exception": str(e),
123
+ }
124
+ if folders_report['/'].get("invalid_entries", None) is None:
125
+ folders_report['/']["invalid_entries"] = []
126
+ folders_report['/']["invalid_entries"] = [
127
+ *folders_report['/']["invalid_entries"],
128
+ issue
129
+ ]
130
+ continue
131
+ elif isinstance(entry.payload, core.Payload):
132
+ payload = entry.payload
133
+ else:
134
+ continue
135
+
136
+ if not payload.schema_shortname:
137
+ continue
138
+
139
+ body = payload.body
140
+ schema_data = session.exec(
141
+ select(Entries)
142
+ .where(Entries.shortname == payload.schema_shortname)
143
+ .where(Entries.subpath == "/schema")
144
+ ).first()
145
+
146
+ if not schema_data:
147
+ continue
148
+
149
+ schema_payload : core.Payload
150
+ if schema_data.payload and isinstance(schema_data.payload, dict):
151
+ schema_payload = core.Payload.model_validate(schema_data.payload)
152
+ elif schema_data.payload and isinstance(schema_data.payload, core.Payload):
153
+ schema_payload = schema_data.payload
154
+ else:
155
+ continue
156
+
157
+ if not schema_payload.body:
158
+ continue
159
+ schema_body = schema_payload.body
160
+ if isinstance(schema_body, str):
161
+ continue
162
+
163
+ if subpath not in folders_report:
164
+ folders_report[subpath] = {
165
+ "valid_entries": 0,
166
+ }
167
+
168
+ try:
169
+ Draft7Validator(
170
+ schema_body
171
+ ).validate(body)
172
+ folders_report[subpath]["valid_entries"] += 1
173
+ except ValidationError as e:
174
+ issue = {
175
+ "issues": ["payload"],
176
+ "uuid": str(entry.uuid),
177
+ "shortname": entry.shortname,
178
+ "resource_type": entry.resource_type,
179
+ "exception": str(e),
180
+ }
181
+ if folders_report[subpath].get("invalid_entries", None) is None:
182
+ folders_report[subpath]["invalid_entries"] = []
183
+ folders_report[subpath]["invalid_entries"] = [
184
+ *folders_report[subpath]["invalid_entries"],
185
+ issue
186
+ ]
187
+
188
+ await save_health_check_entry(
189
+ {"folders_report": folders_report}, space
190
+ )
191
+
192
+
193
+ async def save_health_check_entry(health_check, space_name: str):
194
+ try:
195
+ await serve_request(
196
+ request=api.Request(
197
+ space_name="management",
198
+ request_type=RequestType.create,
199
+ records=[
200
+ core.Record(
201
+ resource_type=ResourceType.content,
202
+ shortname=space_name,
203
+ subpath="/health_check",
204
+ attributes={
205
+ "is_active": True,
206
+ "payload": {
207
+ "schema_shortname": "health_check",
208
+ "content_type": ContentType.json,
209
+ "body": health_check
210
+ }
211
+ },
212
+ )
213
+ ],
214
+ ),
215
+ owner_shortname='dmart',
216
+ )
217
+ except Exception as e:
218
+ print(e)
219
+
220
+ if __name__ == "__main__":
221
+ parser = argparse.ArgumentParser(
222
+ description="This created for doing health check functionality",
223
+ formatter_class=argparse.ArgumentDefaultsHelpFormatter,
224
+ )
225
+ parser.add_argument("-t", "--type", help="type of health check (soft or hard)")
226
+ parser.add_argument("-s", "--space", help="hit the target space or pass (all) to make the full health check")
227
+ parser.add_argument("-m", "--schemas", nargs="*", help="hit the target schema inside the space")
228
+
229
+ args = parser.parse_args()
230
+ before_time = time.time()
231
+ asyncio.run(main(args.type, args.space or "all", args.schemas))
232
+ print(f'total time: {"{:.2f}".format(time.time() - before_time)} sec')
@@ -0,0 +1,454 @@
1
+ from datetime import datetime
2
+ from data_adapters.sql.adapter import SQLAdapter
3
+ from utils.settings import settings
4
+ import asyncio
5
+ import hashlib
6
+ import json
7
+ import os
8
+ import sys
9
+ from pathlib import Path
10
+ from typing import Any
11
+ from uuid import uuid4
12
+ from utils.query_policies_helper import generate_query_policies
13
+ from models.enums import ResourceType, ContentType
14
+ from data_adapters.sql.create_tables import Entries, Users, Attachments, Roles, Permissions, Spaces, generate_tables, \
15
+ Histories
16
+
17
+ async def save_health_check_entry():
18
+ health_check_entry = {
19
+ "space_name": "management",
20
+ "resource_type": "content",
21
+ "shortname": "migration_json_to_db",
22
+ "subpath": "/health_check",
23
+ "is_active": True,
24
+ "payload": {
25
+ "schema_shortname": "health_check",
26
+ "content_type": ContentType.json,
27
+ "body": {"folders_report": folders_report}
28
+ }
29
+ }
30
+
31
+ try:
32
+ async with SQLAdapter().get_session() as session:
33
+ session.add(Spaces.model_validate(health_check_entry))
34
+ await session.commit()
35
+ except Exception as e:
36
+ print(e)
37
+
38
+ def subpath_checker(subpath: str):
39
+ if subpath.endswith("/"):
40
+ subpath = subpath[:-1]
41
+ if not subpath.startswith("/"):
42
+ subpath = '/' + subpath
43
+ return subpath
44
+
45
+ folders_report: Any = {}
46
+ invalid_entries: Any = []
47
+
48
+ def save_issue(resource_type, entry, e):
49
+ entry_uuid = None
50
+ entry_shortname = None
51
+ if isinstance(entry, dict):
52
+ entry_uuid = str(entry["uuid"])
53
+ entry_shortname = entry["shortname"]
54
+ else:
55
+ entry_uuid = str(entry.uuid)
56
+ entry_shortname = entry.shortname
57
+
58
+ return {
59
+ "issues": ["entry"],
60
+ "uuid": entry_uuid,
61
+ "shortname": entry_shortname,
62
+ "resource_type": resource_type,
63
+ "exception": str(e),
64
+ }
65
+
66
+ def save_report(isubpath: str, issue):
67
+ if folders_report.get(isubpath, False):
68
+ if folders_report[isubpath].get("invalid_entries", False):
69
+ folders_report[isubpath]["invalid_entries"] = [
70
+ *folders_report[isubpath]["invalid_entries"],
71
+ issue
72
+ ]
73
+ else:
74
+ folders_report[isubpath]["invalid_entries"] = [
75
+ issue
76
+ ]
77
+ else:
78
+ folders_report[isubpath] = {
79
+ "invalid_entries": [
80
+ issue
81
+ ]
82
+ }
83
+
84
+ async def bulk_insert_in_batches(model, records, batch_size=2000):
85
+ async with SQLAdapter().get_session() as session:
86
+ try:
87
+ for i in range(0, len(records), batch_size):
88
+ batch = []
89
+ try:
90
+ batch = records[i:i + batch_size]
91
+ for record in batch:
92
+ if isinstance(record.get('created_at'), str):
93
+ record['created_at'] = datetime.fromisoformat(record['created_at'])
94
+ if isinstance(record.get('updated_at'), str):
95
+ record['updated_at'] = datetime.fromisoformat(record['updated_at'])
96
+ await session.run_sync(lambda ses: ses.bulk_insert_mappings(model, batch))
97
+ await session.commit()
98
+ except Exception as e:
99
+ print("[!bulk_insert_in_batches]", e)
100
+ await session.rollback()
101
+ for _batch in batch:
102
+ try:
103
+ session.add(model.model_validate(_batch))
104
+ await session.commit()
105
+ except Exception as e:
106
+ await session.rollback()
107
+ print(
108
+ "[!bulk_insert_in_batches_single]",
109
+ e,
110
+ f"* {_batch['subpath']}/{_batch['shortname']}"
111
+ )
112
+ save_report('/', save_issue(_batch['resource_type'], _batch, e))
113
+ except Exception as e:
114
+ print("[!fatal_bulk_insert_in_batches]", e)
115
+
116
+
117
+ async def _process_directory(root, dirs, space_name, subpath):
118
+ # asyncio.run()
119
+ await process_directory(root, dirs, space_name, subpath)
120
+
121
+ async def process_directory(root, dirs, space_name, subpath):
122
+ histories = []
123
+ attachments = []
124
+ entries = []
125
+ users = []
126
+ roles = []
127
+ permissions = []
128
+
129
+ for dir in dirs:
130
+ for file in os.listdir(os.path.join(root, dir)):
131
+ if not file.startswith('meta'):
132
+ if file == 'history.jsonl':
133
+ lines = open(os.path.join(root, dir, file), 'r').readlines()
134
+ for line in lines:
135
+ history = None
136
+ try:
137
+ history = json.loads(line.replace('\n', ''))
138
+ history['shortname'] = dir
139
+ history['space_name'] = space_name
140
+ history['subpath'] = subpath_checker(subpath)
141
+ history['timestamp'] = datetime.strptime(history['timestamp'], '%Y-%m-%dT%H:%M:%S.%f')
142
+
143
+ histories.append(history)
144
+
145
+ except Exception as e:
146
+ print(f"Error processing Histories {space_name}/{subpath}/{dir}/{history} ... ")
147
+ print(e)
148
+
149
+ p = os.path.join(root, dir, file)
150
+ if Path(p).is_file():
151
+ if 'attachments' in p:
152
+ if file.startswith("meta") and file.endswith(".json"):
153
+ _attachment = json.load(open(os.path.join(root, dir, file)))
154
+ _attachment['space_name'] = space_name
155
+ _attachment['uuid'] = _attachment.get('uuid', uuid4())
156
+ _attachment['subpath'] = subpath.replace('//', '/')
157
+ _attachment['subpath'] = subpath_checker(_attachment['subpath'])
158
+ _attachment['acl'] = _attachment.get('acl', [])
159
+ _attachment['relationships'] = _attachment.get('relationships', [])
160
+ _attachment['tags'] = _attachment.get('tags', [])
161
+ _attachment['owner_shortname'] = _attachment.get('owner_shortname', '')
162
+ _attachment['media'] = None
163
+ if file.replace("attachments.", "") == 'comment':
164
+ _attachment['payload'] = {
165
+ 'body': _attachment.get('body', ''),
166
+ 'state': _attachment.get('state', '')
167
+ }
168
+ elif file.replace("attachments.", "") == 'ticket':
169
+ _attachment['payload'] = {
170
+ 'state': _attachment.get('state', ''),
171
+ 'is_open': _attachment.get('is_open', True),
172
+ 'reporter': _attachment.get('reporter', ''),
173
+ 'workflow_shortname': _attachment.get('workflow_shortname', ''),
174
+ 'collaborators': _attachment.get('collaborators', {})
175
+ }
176
+ else:
177
+ _body: str = _attachment.get('payload', {}).get('body', None)
178
+ if _body and _body.endswith('.json'):
179
+ _attachment_body = json.load(open(os.path.join(root, dir, _body)))
180
+ _attachment['payload']['body'] = _attachment_body
181
+ elif _body:
182
+ if not _attachment.get('payload', {}).get('content_type', False):
183
+ _attachment['media'] = None
184
+ else:
185
+ try:
186
+ _attachment['media'] = open(os.path.join(root, dir, _body), 'rb').read()
187
+ except Exception as e:
188
+ print(f"Error reading media file {os.path.join(root, dir, _body)}: {e}")
189
+ _attachment['media'] = None
190
+ if _attachment.get('payload', None) is None:
191
+ _attachment['payload'] = {}
192
+ try:
193
+ _attachment['resource_type'] = dir.replace('attachments.', '')
194
+ attachments.append(_attachment)
195
+ except Exception as e:
196
+ print(f"Error processing Attachments {space_name}/{subpath}/{dir}/{file} ... ")
197
+ print("!!", e)
198
+ save_report('/', save_issue(_attachment['resource_type'], _attachment, e))
199
+ elif file.startswith('meta.') and file.endswith('.json'):
200
+ entry = json.load(open(p))
201
+ entry['space_name'] = space_name
202
+ body = None
203
+ _payload = entry.get('payload', {})
204
+ if _payload:
205
+ if payload := entry.get('payload', {}).get('body', None):
206
+ if entry.get('payload', {}).get('content_type', None) == 'json':
207
+ try:
208
+ body = json.load(open(
209
+ os.path.join(root, dir, '../..', payload)
210
+ ))
211
+ except Exception as e:
212
+ save_report('/', save_issue(ResourceType.json, entry, e))
213
+ else:
214
+ body = payload
215
+
216
+ sha1 = hashlib.sha1()
217
+ sha1.update(json.dumps(body).encode())
218
+ checksum = sha1.hexdigest()
219
+ entry['payload']['checksum'] = checksum
220
+ entry['payload']['body'] = body
221
+ else:
222
+ entry['payload'] = None
223
+ entry['subpath'] = subpath_checker(subpath)
224
+ entry['acl'] = entry.get('acl', [])
225
+ entry['relationships'] = entry.get('relationships', [])
226
+ try:
227
+ if file.startswith("meta.user"):
228
+ entry['query_policies'] = generate_query_policies(
229
+ space_name=space_name,
230
+ subpath=subpath,
231
+ resource_type=ResourceType.user,
232
+ is_active=True,
233
+ owner_shortname=entry.get('owner_shortname', 'dmart'),
234
+ owner_group_shortname=entry.get('owner_group_shortname', None),
235
+ )
236
+ entry['resource_type'] = 'user'
237
+ entry['firebase_token'] = entry.get('firebase_token', '')
238
+ entry['type'] = entry.get('type', 'web')
239
+ entry['language'] = entry.get('language', '')
240
+ entry['google_id'] = entry.get('google_id', '')
241
+ entry['facebook_id'] = entry.get('facebook_id', '')
242
+ entry['social_avatar_url'] = entry.get('social_avatar_url', '')
243
+ entry['displayname'] = entry.get('displayname', {})
244
+ entry['description'] = entry.get('description', {})
245
+ users.append(entry)
246
+ elif file.startswith("meta.role"):
247
+ entry['query_policies'] = generate_query_policies(
248
+ space_name=space_name,
249
+ subpath=subpath,
250
+ resource_type=ResourceType.role,
251
+ is_active=True,
252
+ owner_shortname=entry.get('owner_shortname', 'dmart'),
253
+ owner_group_shortname=entry.get('owner_group_shortname', None),
254
+ )
255
+ entry['resource_type'] = 'role'
256
+ entry['permissions'] = entry.get('permissions', [])
257
+ roles.append(entry)
258
+ elif file.startswith("meta.permission"):
259
+ entry['query_policies'] = generate_query_policies(
260
+ space_name=space_name,
261
+ subpath=subpath,
262
+ resource_type=ResourceType.permission,
263
+ is_active=True,
264
+ owner_shortname=entry.get('owner_shortname', 'dmart'),
265
+ owner_group_shortname=entry.get('owner_group_shortname', None),
266
+ )
267
+ entry['resource_type'] = 'permission'
268
+ entry['subpaths'] = entry.get('subpaths', {})
269
+ entry['resource_types'] = entry.get('resource_types', [])
270
+ entry['actions'] = entry.get('actions', [])
271
+ entry['conditions'] = entry.get('conditions', [])
272
+ entry['restricted_fields'] = entry.get('restricted_fields', [])
273
+ entry['allowed_fields_values'] = entry.get('allowed_fields_values', {})
274
+ permissions.append(entry)
275
+ else:
276
+ entry['resource_type'] = file.replace('.json', '').replace('meta.', '')
277
+
278
+ entry['query_policies'] = generate_query_policies(
279
+ space_name=space_name,
280
+ subpath=subpath,
281
+ resource_type=entry['resource_type'],
282
+ is_active=True,
283
+ owner_shortname=entry.get('owner_shortname', 'dmart'),
284
+ owner_group_shortname=entry.get('owner_group_shortname', None),
285
+ )
286
+
287
+ if entry['resource_type'] == 'folder':
288
+ new_subpath = entry['subpath'].split('/')
289
+ entry['subpath'] = '/'.join(new_subpath[:-1]) + '/'
290
+ elif entry['resource_type'] == 'ticket':
291
+ entry["state"] = entry.get("state", "")
292
+ entry["is_open"] = entry.get("is_open", True)
293
+ entry["reporter"] = entry.get("reporter", None)
294
+ entry["workflow_shortname"] = entry.get("workflow_shortname", "")
295
+ entry["collaborators"] = entry.get("collaborators", None)
296
+ entry["resolution_reason"] = entry.get("resolution_reason", None)
297
+ entry['displayname'] = entry.get('displayname', {})
298
+ entry['description'] = entry.get('description', {})
299
+ entry["subpath"] = subpath_checker(entry["subpath"])
300
+ entries.append(entry)
301
+ continue
302
+ entry["subpath"] = subpath_checker(entry["subpath"])
303
+
304
+ entries.append(entry)
305
+ except Exception as e:
306
+ save_report('/', save_issue(entry['resource_type'], entry, e))
307
+
308
+ await bulk_insert_in_batches(Users, users)
309
+ await bulk_insert_in_batches(Roles, roles)
310
+ await bulk_insert_in_batches(Permissions, permissions)
311
+ await bulk_insert_in_batches(Entries, entries)
312
+ await bulk_insert_in_batches(Attachments, attachments)
313
+ await bulk_insert_in_batches(Histories, histories)
314
+
315
+
316
+ async def main():
317
+ generate_tables()
318
+
319
+ target_path = settings.spaces_folder
320
+
321
+ if len(sys.argv) == 2 and sys.argv[1] != 'json_to_db':
322
+ target_path = target_path.joinpath(sys.argv[1])
323
+
324
+ if not target_path.exists():
325
+ print(f"Space '{str(target_path).replace('/', '')}' does not exist")
326
+ sys.exit(1)
327
+
328
+ all_dirs = []
329
+ user_dirs = []
330
+ for root, dirs, _ in os.walk(str(target_path)):
331
+ if root.startswith(os.path.join(str(target_path), 'management/users')):
332
+ user_dirs.append((root, sorted(dirs, key=lambda d: d != 'dmart')))
333
+ else:
334
+ all_dirs.append((root, dirs))
335
+
336
+ user_dirs.sort(key=lambda x: (
337
+ not x[0].startswith(os.path.join(str(target_path), 'management/users/.dm')),
338
+ x[0]
339
+ ))
340
+
341
+
342
+ for root, dirs in user_dirs:
343
+ tmp = root.replace(str(settings.spaces_folder), '')
344
+ if tmp == '':
345
+ continue
346
+ if tmp[0] == '/':
347
+ tmp = tmp[1:]
348
+ space_name = tmp.split('/')[0]
349
+ subpath = '/'.join(tmp.split('/')[1:])
350
+ if space_name == '..':
351
+ continue
352
+
353
+ if space_name.startswith('.git'):
354
+ continue
355
+
356
+ subpath = subpath.replace('.dm', '')
357
+ if subpath != '/' and subpath.endswith('/'):
358
+ subpath = subpath[:-1]
359
+
360
+ if subpath == '':
361
+ subpath = '/'
362
+
363
+ await process_directory(root, dirs, space_name, subpath)
364
+
365
+ # with ThreadPoolExecutor() as executor:
366
+ # futures = []
367
+ for root, dirs in all_dirs:
368
+ tmp = root.replace(str(settings.spaces_folder), '')
369
+ if tmp == '':
370
+ continue
371
+ if tmp[0] == '/':
372
+ tmp = tmp[1:]
373
+ space_name = tmp.split('/')[0]
374
+ subpath = '/'.join(tmp.split('/')[1:])
375
+ if space_name == '..':
376
+ continue
377
+
378
+ if space_name.startswith('.git'):
379
+ continue
380
+
381
+ print(".", end='')
382
+ if subpath == '' or subpath == '/':
383
+ subpath = '/'
384
+ p = os.path.join(root, '.dm', 'meta.space.json')
385
+ entry = {}
386
+ if Path(p).is_file():
387
+ try:
388
+ entry = json.load(open(p))
389
+ entry['space_name'] = space_name
390
+ entry['shortname'] = space_name
391
+ entry['query_policies'] = generate_query_policies(
392
+ space_name=space_name,
393
+ subpath=subpath,
394
+ resource_type=ResourceType.space,
395
+ is_active=True,
396
+ owner_shortname=entry.get('owner_shortname', 'dmart'),
397
+ owner_group_shortname=entry.get('owner_group_shortname', None),
398
+ )
399
+
400
+ _payload = entry.get('payload', {})
401
+ if _payload:
402
+ if payload := _payload.get('body', None):
403
+ if entry.get('payload', {}).get('content_type', None) == 'json':
404
+ body = json.load(open(
405
+ os.path.join(root, '.dm', '../..', str(payload))
406
+ ))
407
+ else:
408
+ body = payload
409
+ sha1 = hashlib.sha1()
410
+ sha1.update(json.dumps(body).encode())
411
+ checksum = sha1.hexdigest()
412
+ entry['payload']['checksum'] = checksum
413
+ entry['payload']['body'] = body
414
+ else:
415
+ entry['payload'] = None
416
+ entry['subpath'] = '/'
417
+ entry['resource_type'] = 'space'
418
+ entry['tags'] = entry.get('tags', [])
419
+ entry['acl'] = entry.get('acl', [])
420
+ entry['hide_folders'] = entry.get('hide_folders', [])
421
+ entry['relationships'] = entry.get('relationships', [])
422
+ entry['hide_space'] = entry.get('hide_space', False)
423
+
424
+ async with SQLAdapter().get_session() as session:
425
+ session.add(Spaces.model_validate(entry))
426
+ await session.commit()
427
+ except Exception as e:
428
+ save_report('/', save_issue(ResourceType.space, entry, e))
429
+ continue
430
+
431
+ subpath = subpath.replace('.dm', '')
432
+ if subpath != '/' and subpath.endswith('/'):
433
+ subpath = subpath[:-1]
434
+
435
+ if subpath == '':
436
+ subpath = '/'
437
+
438
+ await _process_directory(root, dirs, space_name, subpath)
439
+ # futures.append(executor.submit(_process_directory, root, dirs, space_name, subpath))
440
+ # as_completed(futures)
441
+
442
+ # for future in as_completed(futures):
443
+ # future.result()
444
+
445
+ if settings.active_data_db == 'file':
446
+ print("[Warning] you are using active_data_db='file', please don't forget to set it to active_data_db='sql' in your config.env")
447
+
448
+ await save_health_check_entry()
449
+
450
+ # await SQLAdapter().ensure_authz_materialized_views_fresh()
451
+
452
+
453
+ if __name__ == "__main__":
454
+ asyncio.run(main())