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
dmart/utils/regex.py ADDED
@@ -0,0 +1,44 @@
1
+ import re
2
+
3
+ SUBPATH = "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_/]{1,128}$"
4
+ SHORTNAME = "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_]{1,64}$"
5
+ SLUG = "^[a-zA-Z0-9_-]{1,64}$"
6
+ FILENAME = "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_]{1,32}\\.(gif|png|jpeg|jpg|pdf|wsq|mp3|mp4|csv|jsonl|parquet|sqlite|sqlite3|sqlite|db|duckdb|svg|apk)$"
7
+ SPACENAME = "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_]{1,32}$"
8
+ EXT = "^(gif|png|jpeg|jpg|webp|json|md|pdf|wsq|mp3|mp4|csv|jsonl|parquet|sqlite|sqlite3|db|db3|s3db|sl3|duckdb|svg|xsls|docx|apk)$"
9
+ IMG_EXT = "^(gif|png|jpeg|jpg|wsq|svg|webp)$"
10
+ USERNAME = "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_]{3,10}$"
11
+ PASSWORD = "^(?=.*[0-9\u0660-\u0669])(?=.*[A-Z\u0621-\u064A])[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_#@%*!?$^-]{8,24}$"
12
+ EMAIL = (
13
+ "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_\\.-]+@([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_-]+\\.)+"
14
+ "[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_-]{2,4}$"
15
+ )
16
+ META_DOC_ID = (
17
+ "^[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]*:[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]"
18
+ "[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]*:meta:[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_/]+$"
19
+ )
20
+ MSISDN = "^[1-9][0-9]{9, 14}$" # Between 10 and 14 digits, not starting with zero
21
+ EXTENDED_MSISDN = (
22
+ "^[1-9][0-9]{9,14}$" # Between 10 and 14 digits, not starting with zero
23
+ )
24
+ OTP_CODE = "^[0-9\u0660-\u0669]{6}$" # Exactly 6 digits
25
+ INVITATION = (
26
+ "^([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_=]+)\\.([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_=]+)"
27
+ "\\.([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_+/=-]*)$"
28
+ )
29
+
30
+ FILE_PATTERN = re.compile(
31
+ "\\.dm/([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]*)/meta\\.([a-zA-Z\u0621-\u064A]*)\\.json$"
32
+ )
33
+ PAYLOAD_FILE_PATTERN = re.compile("([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]*)\\.json$")
34
+ # HISTORY_PATTERN = re.compile("([0-9\u0660-\u0669]*)\\.json$")
35
+ ATTACHMENT_PATTERN = re.compile(
36
+ "attachments\\.*[a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]*\\.([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]+)"
37
+ "/meta\\.([a-zA-Z\u0621-\u064A0-9\u0660-\u0669_]*)\\.json$"
38
+ )
39
+ FOLDER_PATTERN = re.compile(
40
+ "/([a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_]*)/\\.dm/meta\\.folder\\.json$"
41
+ )
42
+ SPACES_PATTERN = re.compile(
43
+ "/([a-zA-Z\u0621-\u064A0-9\u0660-\u0669\u064B-\u065F_]*)/\\.dm/meta\\.space\\.json$"
44
+ )
@@ -0,0 +1,529 @@
1
+ import os
2
+ import re
3
+ import sys
4
+ import asyncio
5
+ import json
6
+ import subprocess
7
+ from pathlib import Path
8
+ from typing import Any
9
+ from uuid import uuid4
10
+ from fastapi import status
11
+ import models.api as api
12
+ import models.core as core
13
+ import utils.regex as regex
14
+ from models.enums import ContentType, Language
15
+ from data_adapters.adapter import data_adapter as db
16
+ from utils.helpers import (
17
+ camel_case,
18
+ jq_dict_parser,
19
+ )
20
+ from utils.internal_error_code import InternalErrorCode
21
+ from utils.jwt import generate_jwt
22
+ from utils.settings import settings
23
+
24
+ async def serve_query(
25
+ query: api.Query, logged_in_user: str
26
+ ) -> tuple[int, list[core.Record]]:
27
+ records: list[core.Record] = []
28
+ total: int = 0
29
+
30
+ total, records = await db.query(query, logged_in_user)
31
+
32
+ try:
33
+ for _r in records or []:
34
+ attrs = getattr(_r, "attributes", None)
35
+ if isinstance(attrs, dict):
36
+ attrs.pop("password", None)
37
+ except Exception:
38
+ pass
39
+
40
+ if query.jq_filter:
41
+ try:
42
+ def _run_jq_subprocess() -> list:
43
+ _input_local = [record.model_dump() for record in records]
44
+ _input_local = jq_dict_parser(_input_local)
45
+ input_json = json.dumps(_input_local)
46
+
47
+ cmd = ["jq", "-c", query.jq_filter]
48
+
49
+ try:
50
+ completed = subprocess.run(
51
+ cmd, # type: ignore
52
+ input=input_json.encode("utf-8"),
53
+ stdout=subprocess.PIPE,
54
+ stderr=subprocess.PIPE,
55
+ timeout=settings.jq_timeout,
56
+ check=False,
57
+ )
58
+ except subprocess.TimeoutExpired:
59
+ raise api.Exception(
60
+ status.HTTP_400_BAD_REQUEST,
61
+ api.Error(
62
+ type="request",
63
+ code=InternalErrorCode.JQ_TIMEOUT,
64
+ message="jq filter took too long to execute",
65
+ ),
66
+ )
67
+
68
+ if completed.returncode != 0:
69
+ raise api.Exception(
70
+ status.HTTP_400_BAD_REQUEST,
71
+ api.Error(
72
+ type="request",
73
+ code=InternalErrorCode.JQ_ERROR,
74
+ message="jq filter failed to be executed",
75
+ ),
76
+ )
77
+
78
+ stdout = completed.stdout.decode("utf-8")
79
+ results: list = []
80
+ if stdout.startswith("[") and stdout.endswith("]\n"):
81
+ results = json.loads(stdout)
82
+ else:
83
+ for line in stdout.splitlines():
84
+ line = line.strip()
85
+ if not line:
86
+ continue
87
+ results.append(json.loads(line))
88
+ return results
89
+
90
+ loop = asyncio.get_running_loop()
91
+ records = await asyncio.wait_for(
92
+ loop.run_in_executor(None, _run_jq_subprocess),
93
+ timeout=settings.jq_timeout,
94
+ )
95
+
96
+ except FileNotFoundError:
97
+ raise api.Exception(
98
+ status.HTTP_400_BAD_REQUEST,
99
+ api.Error(
100
+ type="request",
101
+ code=InternalErrorCode.NOT_ALLOWED,
102
+ message="jq is not installed!",
103
+ ),
104
+ )
105
+
106
+ return total, records
107
+
108
+
109
+ async def get_last_updated_entry(
110
+ space_name: str,
111
+ schema_names: list,
112
+ retrieve_json_payload: bool,
113
+ logged_in_user: str,
114
+ ):
115
+ report_query = api.Query(
116
+ type=api.QueryType.search,
117
+ space_name=space_name,
118
+ subpath="/",
119
+ search=f"@schema_shortname:{'|'.join(schema_names)}",
120
+ filter_schema_names=["meta"],
121
+ sort_by="updated_at",
122
+ sort_type=api.SortType.descending,
123
+ limit=50, # to be in safe side if the query filtered out some invalid entries
124
+ retrieve_json_payload=retrieve_json_payload,
125
+ )
126
+ _, records = await serve_query(report_query, logged_in_user)
127
+
128
+ return records[0] if records else None
129
+
130
+
131
+ async def get_resource_obj_or_none(
132
+ *,
133
+ space_name: str,
134
+ subpath: str,
135
+ shortname: str,
136
+ resource_type: str,
137
+ user_shortname: str,
138
+ ):
139
+ resource_cls = getattr(
140
+ sys.modules["models.core"], camel_case(resource_type))
141
+ try:
142
+ return await db.load(
143
+ space_name=space_name,
144
+ subpath=subpath,
145
+ shortname=shortname,
146
+ class_type=resource_cls,
147
+ user_shortname=user_shortname,
148
+ )
149
+ except Exception:
150
+ return None
151
+
152
+
153
+ async def get_payload_obj_or_none(
154
+ *,
155
+ space_name: str,
156
+ subpath: str,
157
+ filename: str,
158
+ resource_type: str,
159
+ ):
160
+ resource_cls = getattr(
161
+ sys.modules["models.core"], camel_case(resource_type))
162
+ try:
163
+ return await db.load_resource_payload(
164
+ space_name=space_name,
165
+ subpath=subpath,
166
+ filename=filename,
167
+ class_type=resource_cls,
168
+ )
169
+ except Exception:
170
+ return None
171
+
172
+
173
+ async def folder_meta_content_check(
174
+ space_name, subpath, folder_name, spaces_path_parts,
175
+ user_shortname, folder_name_index, invalid_folders
176
+ ):
177
+ try:
178
+ folder_meta_content = await db.load(
179
+ space_name=space_name,
180
+ subpath=folder_name,
181
+ shortname="",
182
+ class_type=core.Folder,
183
+ user_shortname=user_shortname,
184
+ )
185
+ if (
186
+ folder_meta_content.payload
187
+ and folder_meta_content.payload.content_type == ContentType.json
188
+ ):
189
+ payload_path = "/"
190
+ subpath_parts = subpath.split("/")
191
+ if len(subpath_parts) > (len(spaces_path_parts) + 2):
192
+ payload_path = "/".join(
193
+ subpath_parts[folder_name_index:-1])
194
+ folder_meta_payload = await db.load_resource_payload(
195
+ space_name,
196
+ payload_path,
197
+ str(folder_meta_content.payload.body),
198
+ core.Folder,
199
+ )
200
+ if folder_meta_content.payload.schema_shortname and folder_meta_payload:
201
+ await db.validate_payload_with_schema(
202
+ payload_data=folder_meta_payload,
203
+ space_name=space_name,
204
+ schema_shortname=folder_meta_content.payload.schema_shortname,
205
+ )
206
+ except Exception:
207
+ invalid_folders.append(folder_name)
208
+
209
+
210
+ async def health_check_entry_vsd(
211
+ space_name, folder_name, entry_shortname, entry_resource_type, user_shortname,
212
+ folder, folders_report, max_invalid_size, entry_meta_obj
213
+ ):
214
+ await health_check_entry(
215
+ space_name=space_name,
216
+ subpath=folder_name,
217
+ shortname=entry_shortname,
218
+ resource_type=entry_resource_type,
219
+ user_shortname=user_shortname,
220
+ )
221
+
222
+ # VALIDATE ENTRY ATTACHMENTS
223
+ attachments_path = f"{folder.path}/{entry_shortname}"
224
+ attachment_folders = os.scandir(attachments_path)
225
+ for attachment_folder in attachment_folders:
226
+ # i.e. attachment_folder = attachments.media
227
+ if attachment_folder.is_file():
228
+ continue
229
+
230
+ attachment_folder_files = os.scandir(attachment_folder)
231
+ for attachment_folder_file in attachment_folder_files:
232
+ # i.e. attachment_folder_file = meta.*.json or *.png
233
+ if (
234
+ not attachment_folder_file.is_file()
235
+ or not os.access(attachment_folder_file.path, os.W_OK)
236
+ or not os.access(attachment_folder_file.path, os.R_OK)
237
+ ):
238
+ raise Exception(
239
+ f"can't access this attachment {attachment_folder_file.path[len(str(settings.spaces_folder)):]}"
240
+ )
241
+
242
+ attachment_match = regex.ATTACHMENT_PATTERN.search(attachment_folder_file.path)
243
+ if not attachment_match:
244
+ # if it's the media file not its meta json file
245
+ continue
246
+ attachment_shortname = attachment_match.group(2)
247
+ attachment_resource_type = attachment_match.group(1)
248
+ await health_check_entry(
249
+ space_name=space_name,
250
+ subpath=f"{folder_name}/{entry_shortname}",
251
+ shortname=attachment_shortname,
252
+ resource_type=attachment_resource_type,
253
+ user_shortname=user_shortname,
254
+ )
255
+
256
+ if "valid_entries" not in folders_report[folder_name]:
257
+ folders_report[folder_name]["valid_entries"] = 1
258
+ else:
259
+ folders_report[folder_name]["valid_entries"] += 1
260
+
261
+
262
+ async def validate_subpath_data(
263
+ space_name: str,
264
+ subpath: str,
265
+ user_shortname: str,
266
+ invalid_folders: list[str],
267
+ folders_report: dict[str, dict[str, Any]],
268
+ meta_folders_health: list[str],
269
+ max_invalid_size: int,
270
+ ):
271
+ """
272
+ Params:
273
+ @subpath: str holding the full path, ex: ../spaces/aftersales/reports
274
+
275
+ Algorithm:
276
+ - if subpath ends with .dm return
277
+ - for folder in scandir(subpath)
278
+ - if folder ends with .dm
279
+ - get folder_meta = folder/meta.folder.json
280
+ - validate folder_meta
281
+ - loop over folder.entries and validate them along with theire attachments
282
+ - else
283
+ - call myself with subpath = folder
284
+ """
285
+ spaces_path_parts = str(settings.spaces_folder).split("/")
286
+ folder_name_index = len(spaces_path_parts) + 1
287
+
288
+ if subpath.endswith(".dm"):
289
+ return
290
+
291
+ subpath_folders = os.scandir(subpath)
292
+ for folder in subpath_folders:
293
+ if not folder.is_dir():
294
+ continue
295
+
296
+ if folder.name != ".dm":
297
+ await validate_subpath_data(
298
+ space_name,
299
+ folder.path,
300
+ user_shortname,
301
+ invalid_folders,
302
+ folders_report,
303
+ meta_folders_health,
304
+ max_invalid_size,
305
+ )
306
+ continue
307
+
308
+ folder_meta = Path(f"{folder.path}/meta.folder.json")
309
+ folder_name = "/".join(subpath.split("/")[folder_name_index:])
310
+ if not folder_meta.is_file():
311
+ meta_folders_health.append(
312
+ str(folder_meta)[len(str(settings.spaces_folder)):]
313
+ )
314
+ continue
315
+
316
+ await folder_meta_content_check(
317
+ space_name, subpath, folder_name, spaces_path_parts,
318
+ user_shortname, folder_name_index, invalid_folders
319
+ )
320
+
321
+ folders_report.setdefault(folder_name, {})
322
+
323
+ # VALIDATE FOLDER ENTRIES
324
+ folder_entries = os.scandir(folder.path)
325
+ for entry in folder_entries:
326
+ if entry.is_file():
327
+ continue
328
+
329
+ entry_files = os.scandir(entry)
330
+ entry_match = None
331
+ for file in entry_files:
332
+ if file.is_dir():
333
+ continue
334
+ entry_match = regex.FILE_PATTERN.search(file.path)
335
+
336
+ if entry_match:
337
+ break
338
+
339
+ if not entry_match:
340
+ issue = {
341
+ "issues": ["meta"],
342
+ "uuid": "",
343
+ "shortname": entry.name,
344
+ "exception": f"Can't access this meta {subpath[len(str(settings.spaces_folder)):]}/{entry.name}",
345
+ }
346
+
347
+ if "invalid_entries" not in folders_report[folder_name]:
348
+ folders_report[folder_name]["invalid_entries"] = [issue]
349
+ else:
350
+ if (
351
+ len(folders_report[folder_name]["invalid_entries"])
352
+ >= max_invalid_size
353
+ ):
354
+ break
355
+ folders_report[folder_name]["invalid_entries"].append(
356
+ issue)
357
+ continue
358
+
359
+ entry_shortname = entry_match.group(1)
360
+ entry_resource_type = entry_match.group(2)
361
+
362
+ if folder_name == "schema" and entry_shortname == "meta_schema":
363
+ folders_report[folder_name].setdefault("valid_entries", 0)
364
+ folders_report[folder_name]["valid_entries"] += 1
365
+ continue
366
+
367
+ entry_meta_obj = None
368
+ try:
369
+ await health_check_entry_vsd(
370
+ space_name, folder_name, entry_shortname, entry_resource_type, user_shortname,
371
+ folder, folders_report, max_invalid_size, entry_meta_obj
372
+ )
373
+ except Exception as e:
374
+ issue_type = "payload"
375
+ uuid = ""
376
+ if not entry_meta_obj:
377
+ issue_type = "meta"
378
+ else:
379
+ uuid = str(
380
+ entry_meta_obj.uuid) if entry_meta_obj.uuid else ""
381
+
382
+ issue = {
383
+ "issues": [issue_type],
384
+ "uuid": uuid,
385
+ "shortname": entry_shortname,
386
+ "resource_type": entry_resource_type,
387
+ "exception": str(e),
388
+ }
389
+
390
+ if "invalid_entries" not in folders_report[folder_name]:
391
+ folders_report[folder_name]["invalid_entries"] = [issue]
392
+ else:
393
+ if (
394
+ len(folders_report[folder_name]["invalid_entries"])
395
+ >= max_invalid_size
396
+ ):
397
+ break
398
+ folders_report[folder_name]["invalid_entries"].append(
399
+ issue
400
+ )
401
+
402
+ if not folders_report.get(folder_name, {}):
403
+ del folders_report[folder_name]
404
+
405
+
406
+ async def health_check_entry(
407
+ space_name: str,
408
+ subpath: str,
409
+ resource_type: str,
410
+ shortname: str,
411
+ user_shortname: str,
412
+ ):
413
+ resource_class = getattr(
414
+ sys.modules["models.core"], camel_case(resource_type)
415
+ )
416
+ entry_meta_obj = resource_class.model_validate(await db.load(
417
+ space_name=space_name,
418
+ subpath=subpath,
419
+ shortname=shortname,
420
+ class_type=resource_class,
421
+ user_shortname=user_shortname,
422
+ ))
423
+ if entry_meta_obj.shortname != shortname:
424
+ raise Exception(
425
+ "the shortname which got from the folder path doesn't match the shortname in the meta file."
426
+ )
427
+ payload_file_path = None
428
+ if (
429
+ entry_meta_obj.payload
430
+ and entry_meta_obj.payload.content_type == ContentType.image
431
+ ):
432
+ payload_file_path = Path(f"{subpath}/{entry_meta_obj.payload.body}")
433
+ if (
434
+ not payload_file_path.is_file()
435
+ or not bool(
436
+ re.match(
437
+ regex.IMG_EXT,
438
+ entry_meta_obj.payload.body.split(".")[-1],
439
+ )
440
+ )
441
+ or not os.access(payload_file_path, os.R_OK)
442
+ or not os.access(payload_file_path, os.W_OK)
443
+ ):
444
+ if payload_file_path:
445
+ raise Exception(
446
+ f"can't access this payload {payload_file_path}"
447
+ )
448
+ else:
449
+ raise Exception(
450
+ f"can't access this payload {subpath}"
451
+ f"/{entry_meta_obj.shortname}"
452
+ )
453
+ elif (
454
+ entry_meta_obj.payload
455
+ and isinstance(entry_meta_obj.payload.body, str)
456
+ and entry_meta_obj.payload.content_type == ContentType.json
457
+ ):
458
+ payload_file_path = db.payload_path(space_name, subpath, resource_class)
459
+ if not entry_meta_obj.payload.body.endswith(
460
+ ".json"
461
+ ) or not os.access(payload_file_path, os.W_OK):
462
+ raise Exception(
463
+ f"can't access this payload {payload_file_path}"
464
+ )
465
+ payload_file_content = await db.load_resource_payload(
466
+ space_name,
467
+ subpath,
468
+ entry_meta_obj.payload.body,
469
+ resource_class,
470
+ )
471
+ if entry_meta_obj.payload.schema_shortname and payload_file_content:
472
+ await db.validate_payload_with_schema(
473
+ payload_data=payload_file_content,
474
+ space_name=space_name,
475
+ schema_shortname=entry_meta_obj.payload.schema_shortname,
476
+ )
477
+
478
+ if (
479
+ entry_meta_obj.payload.checksum and
480
+ entry_meta_obj.payload.client_checksum and
481
+ entry_meta_obj.payload.checksum != entry_meta_obj.payload.client_checksum
482
+ ):
483
+ raise Exception(
484
+ f"payload.checksum not equal payload.client_checksum {subpath}/{entry_meta_obj.shortname}"
485
+ )
486
+
487
+ async def url_shortner(url: str) -> str:
488
+ token_uuid = str(uuid4())[:8]
489
+ await db.set_url_shortner(token_uuid, url)
490
+ return f"{settings.public_app_url}/managed/s/{token_uuid}"
491
+
492
+
493
+ async def store_user_invitation_token(user: core.User, channel: str) -> str | None:
494
+ """Generate and Store an invitation token
495
+
496
+ Returns:
497
+ invitation link or None if the user is not eligible
498
+ """
499
+ invitation_value = None
500
+ if channel == "SMS" and user.msisdn:
501
+ invitation_value = f"{channel}:{user.msisdn}"
502
+ elif channel == "EMAIL" and user.email:
503
+ invitation_value = f"{channel}:{user.email}"
504
+
505
+ if not invitation_value:
506
+ return None
507
+
508
+ invitation_token = generate_jwt(
509
+ {"shortname": user.shortname, "channel": channel},
510
+ settings.jwt_access_expires,
511
+ )
512
+
513
+ await db.set_invitation(invitation_token, invitation_value)
514
+
515
+ return core.User.invitation_url_template() \
516
+ .replace("{url}", settings.invitation_link) \
517
+ .replace("{token}", invitation_token) \
518
+ .replace("{lang}", Language.code(user.language)) \
519
+ .replace("{user_type}", user.type)
520
+
521
+
522
+ async def delete_space(space_name, record, owner_shortname):
523
+ if settings.active_data_db == "sql":
524
+ resource_obj = core.Meta.from_record(
525
+ record=record, owner_shortname=owner_shortname
526
+ )
527
+ await db.delete(space_name, record.subpath, resource_obj, owner_shortname)
528
+
529
+ os.system(f"rm -r {settings.spaces_folder}/{space_name}")
@@ -0,0 +1,19 @@
1
+ from utils.internal_error_code import InternalErrorCode
2
+ from data_adapters.adapter import data_adapter as db
3
+ import models.api as api
4
+ from fastapi import status
5
+
6
+
7
+ async def is_space_exist(space_name, should_exist=True):
8
+
9
+ space = await db.fetch_space(space_name)
10
+
11
+ if (space is not None) ^ should_exist:
12
+ raise api.Exception(
13
+ status.HTTP_400_BAD_REQUEST,
14
+ api.Error(
15
+ type="request",
16
+ code=InternalErrorCode.INVALID_SPACE_NAME,
17
+ message=f"Space name {space_name} provided is empty or invalid [3]",
18
+ ),
19
+ )