dmart 1.4.17__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (289) hide show
  1. alembic.ini +117 -0
  2. api/__init__.py +0 -0
  3. api/info/__init__.py +0 -0
  4. api/info/router.py +109 -0
  5. api/managed/__init__.py +0 -0
  6. api/managed/router.py +1541 -0
  7. api/managed/utils.py +1879 -0
  8. api/public/__init__.py +0 -0
  9. api/public/router.py +758 -0
  10. api/qr/__init__.py +0 -0
  11. api/qr/router.py +108 -0
  12. api/user/__init__.py +0 -0
  13. api/user/model/__init__.py +0 -0
  14. api/user/model/errors.py +14 -0
  15. api/user/model/requests.py +165 -0
  16. api/user/model/responses.py +11 -0
  17. api/user/router.py +1413 -0
  18. api/user/service.py +270 -0
  19. bundler.py +55 -0
  20. config/__init__.py +0 -0
  21. config/channels.json +11 -0
  22. config/notification.json +17 -0
  23. cxb/__init__.py +0 -0
  24. cxb/client/__init__.py +0 -0
  25. cxb/client/assets/@codemirror-Rn7_6DkE.js +10 -0
  26. cxb/client/assets/@edraj-CS4NwVbD.js +1 -0
  27. cxb/client/assets/@floating-ui-BwwcF-xh.js +1 -0
  28. cxb/client/assets/@formatjs-yKEsAtjs.js +1 -0
  29. cxb/client/assets/@fortawesome-DRW1UCdr.js +9 -0
  30. cxb/client/assets/@jsonquerylang-laKNoFFq.js +12 -0
  31. cxb/client/assets/@lezer-za4Q-8Ew.js +1 -0
  32. cxb/client/assets/@marijn-DXwl3gUT.js +1 -0
  33. cxb/client/assets/@popperjs-l0sNRNKZ.js +1 -0
  34. cxb/client/assets/@replit--ERk53eB.js +1 -0
  35. cxb/client/assets/@roxi-CGMFK4i8.js +6 -0
  36. cxb/client/assets/@typewriter-cCzskkIv.js +17 -0
  37. cxb/client/assets/@zerodevx-BlBZjKxu.js +1 -0
  38. cxb/client/assets/@zerodevx-CVEpe6WZ.css +1 -0
  39. cxb/client/assets/BreadCrumbLite-DAhOx38v.js +1 -0
  40. cxb/client/assets/EntryRenderer-25YDhRen.js +32 -0
  41. cxb/client/assets/EntryRenderer-DXytdFp9.css +1 -0
  42. cxb/client/assets/ListView-BpAycA2h.js +16 -0
  43. cxb/client/assets/ListView-U8of-_c-.css +1 -0
  44. cxb/client/assets/Prism--hMplq-p.js +3 -0
  45. cxb/client/assets/Prism-Uh6uStUw.css +1 -0
  46. cxb/client/assets/Table2Cols-BsbwicQm.js +1 -0
  47. cxb/client/assets/_..-BvT6vdHa.css +1 -0
  48. cxb/client/assets/_...404_-fuLH_rX9.js +2 -0
  49. cxb/client/assets/_...fallback_-Ba_NLmAE.js +1 -0
  50. cxb/client/assets/_module-Bfk8MiCs.js +3 -0
  51. cxb/client/assets/_module-CEW0D5oI.js +4 -0
  52. cxb/client/assets/_module-Dgq0ZVtz.js +1 -0
  53. cxb/client/assets/ajv-Cpj98o6Y.js +1 -0
  54. cxb/client/assets/axios-CG2WSiiR.js +6 -0
  55. cxb/client/assets/clsx-B-dksMZM.js +1 -0
  56. cxb/client/assets/codemirror-wrapped-line-indent-DPhKvljI.js +1 -0
  57. cxb/client/assets/compare-C3AjiGFR.js +1 -0
  58. cxb/client/assets/compute-scroll-into-view-Bl8rNFhg.js +1 -0
  59. cxb/client/assets/consolite-DlCuI0F9.js +1 -0
  60. cxb/client/assets/crelt-C8TCjufn.js +1 -0
  61. cxb/client/assets/date-fns-l0sNRNKZ.js +1 -0
  62. cxb/client/assets/deepmerge-rn4rBaHU.js +1 -0
  63. cxb/client/assets/dmart_services-AL6-IdDE.js +1 -0
  64. cxb/client/assets/downloadFile-D08i0YDh.js +1 -0
  65. cxb/client/assets/easy-signal-BiPFIK3O.js +1 -0
  66. cxb/client/assets/esm-env-rsSWfq8L.js +1 -0
  67. cxb/client/assets/export-OF_rTiXu.js +1 -0
  68. cxb/client/assets/fast-deep-equal-l0sNRNKZ.js +1 -0
  69. cxb/client/assets/fast-diff-C-IidNf4.js +1 -0
  70. cxb/client/assets/fast-uri-l0sNRNKZ.js +1 -0
  71. cxb/client/assets/flowbite-svelte-BLvjb-sa.js +1 -0
  72. cxb/client/assets/flowbite-svelte-CD54FDqW.css +1 -0
  73. cxb/client/assets/flowbite-svelte-icons-BI8GVhw_.js +1 -0
  74. cxb/client/assets/github-slugger-CQ4oX9Ud.js +1 -0
  75. cxb/client/assets/global-igKv-1g9.js +1 -0
  76. cxb/client/assets/hookar-BMRD9G9H.js +1 -0
  77. cxb/client/assets/immutable-json-patch-DtRO2E_S.js +1 -0
  78. cxb/client/assets/import-1vE3gBat.js +1 -0
  79. cxb/client/assets/index-B-eTh-ZX.js +1 -0
  80. cxb/client/assets/index-BVyxzKtH.js +1 -0
  81. cxb/client/assets/index-BdeNM69f.js +1 -0
  82. cxb/client/assets/index-C6cPO4op.js +1 -0
  83. cxb/client/assets/index-CC-A1ipE.js +1 -0
  84. cxb/client/assets/index-CTxJ-lDp.js +1 -0
  85. cxb/client/assets/index-Cd-F5j_k.js +1 -0
  86. cxb/client/assets/index-D742rwaM.js +1 -0
  87. cxb/client/assets/index-DTfhnhwd.js +1 -0
  88. cxb/client/assets/index-DdXRK7n9.js +2 -0
  89. cxb/client/assets/index-DtiCmB4o.js +1 -0
  90. cxb/client/assets/index-NBrXBlLA.css +2 -0
  91. cxb/client/assets/index-ac-Buu_H.js +4 -0
  92. cxb/client/assets/index-iYkH7C67.js +1 -0
  93. cxb/client/assets/info-B986lRiM.js +1 -0
  94. cxb/client/assets/intl-messageformat-Dc5UU-HB.js +3 -0
  95. cxb/client/assets/jmespath-l0sNRNKZ.js +1 -0
  96. cxb/client/assets/json-schema-traverse-l0sNRNKZ.js +1 -0
  97. cxb/client/assets/json-source-map-DRgZidqy.js +5 -0
  98. cxb/client/assets/jsonpath-plus-l0sNRNKZ.js +1 -0
  99. cxb/client/assets/jsonrepair-B30Dx381.js +8 -0
  100. cxb/client/assets/lodash-es-DZVAA2ox.js +1 -0
  101. cxb/client/assets/marked-DKjyhwJX.js +56 -0
  102. cxb/client/assets/marked-gfm-heading-id-U5zO829x.js +2 -0
  103. cxb/client/assets/marked-mangle-CDMeiHC6.js +1 -0
  104. cxb/client/assets/memoize-one-BdPwpGay.js +1 -0
  105. cxb/client/assets/natural-compare-lite-Bg2Xcf-o.js +7 -0
  106. cxb/client/assets/pagination-svelte-D5CyoiE_.js +13 -0
  107. cxb/client/assets/pagination-svelte-v10nAbbM.css +1 -0
  108. cxb/client/assets/plantuml-encoder-C47mzt9T.js +1 -0
  109. cxb/client/assets/prismjs-DTUiLGJu.js +9 -0
  110. cxb/client/assets/profile-BUf-tKMe.js +1 -0
  111. cxb/client/assets/query-CNmXTsgf.js +1 -0
  112. cxb/client/assets/queryHelpers-C9iBWwqe.js +1 -0
  113. cxb/client/assets/scroll-into-view-if-needed-KR58zyjF.js +1 -0
  114. cxb/client/assets/spaces-0oyGvpii.js +1 -0
  115. cxb/client/assets/style-mod-Bs6eFhZE.js +3 -0
  116. cxb/client/assets/svelte-B2XmcTi_.js +4 -0
  117. cxb/client/assets/svelte-awesome-COLlx0DN.css +1 -0
  118. cxb/client/assets/svelte-awesome-DhnMA6Q_.js +1 -0
  119. cxb/client/assets/svelte-datatables-net-CY7LBj6I.js +1 -0
  120. cxb/client/assets/svelte-floating-ui-BlS3sOAQ.js +1 -0
  121. cxb/client/assets/svelte-i18n-CT2KkQaN.js +3 -0
  122. cxb/client/assets/svelte-jsoneditor-BzfX6Usi.css +1 -0
  123. cxb/client/assets/svelte-jsoneditor-CUGSvWId.js +25 -0
  124. cxb/client/assets/svelte-select-CegQKzqH.css +1 -0
  125. cxb/client/assets/svelte-select-CjHAt_85.js +6 -0
  126. cxb/client/assets/tailwind-merge-CJvxXMcu.js +1 -0
  127. cxb/client/assets/tailwind-variants-Cj20BoQ3.js +1 -0
  128. cxb/client/assets/toast-B9WDyfyI.js +1 -0
  129. cxb/client/assets/tslib-pJfR_DrR.js +1 -0
  130. cxb/client/assets/typewriter-editor-DkTVIJdm.js +25 -0
  131. cxb/client/assets/user-DeK_NB5v.js +1 -0
  132. cxb/client/assets/vanilla-picker-l5rcX3cq.js +8 -0
  133. cxb/client/assets/w3c-keyname-Vcq4gwWv.js +1 -0
  134. cxb/client/config.json +11 -0
  135. cxb/client/config.sample.json +11 -0
  136. cxb/client/favicon.ico +0 -0
  137. cxb/client/favicon.png +0 -0
  138. cxb/client/index.html +28 -0
  139. data_adapters/__init__.py +0 -0
  140. data_adapters/adapter.py +16 -0
  141. data_adapters/base_data_adapter.py +467 -0
  142. data_adapters/file/__init__.py +0 -0
  143. data_adapters/file/adapter.py +2043 -0
  144. data_adapters/file/adapter_helpers.py +1013 -0
  145. data_adapters/file/archive.py +150 -0
  146. data_adapters/file/create_index.py +331 -0
  147. data_adapters/file/create_users_folders.py +52 -0
  148. data_adapters/file/custom_validations.py +68 -0
  149. data_adapters/file/drop_index.py +40 -0
  150. data_adapters/file/health_check.py +560 -0
  151. data_adapters/file/redis_services.py +1110 -0
  152. data_adapters/helpers.py +27 -0
  153. data_adapters/sql/__init__.py +0 -0
  154. data_adapters/sql/adapter.py +3218 -0
  155. data_adapters/sql/adapter_helpers.py +491 -0
  156. data_adapters/sql/create_tables.py +451 -0
  157. data_adapters/sql/create_users_folders.py +53 -0
  158. data_adapters/sql/db_to_json_migration.py +485 -0
  159. data_adapters/sql/health_check_sql.py +232 -0
  160. data_adapters/sql/json_to_db_migration.py +454 -0
  161. data_adapters/sql/update_query_policies.py +101 -0
  162. data_generator.py +81 -0
  163. dmart-1.4.17.dist-info/METADATA +65 -0
  164. dmart-1.4.17.dist-info/RECORD +289 -0
  165. dmart-1.4.17.dist-info/WHEEL +5 -0
  166. dmart-1.4.17.dist-info/entry_points.txt +2 -0
  167. dmart-1.4.17.dist-info/top_level.txt +24 -0
  168. dmart.py +623 -0
  169. dmart_migrations/README +1 -0
  170. dmart_migrations/__init__.py +0 -0
  171. dmart_migrations/__pycache__/__init__.cpython-314.pyc +0 -0
  172. dmart_migrations/__pycache__/env.cpython-314.pyc +0 -0
  173. dmart_migrations/env.py +100 -0
  174. dmart_migrations/notes.txt +11 -0
  175. dmart_migrations/script.py.mako +28 -0
  176. dmart_migrations/scripts/__init__.py +0 -0
  177. dmart_migrations/scripts/calculate_checksums.py +77 -0
  178. dmart_migrations/scripts/migration_f7a4949eed19.py +28 -0
  179. dmart_migrations/versions/0f3d2b1a7c21_add_authz_materialized_views.py +87 -0
  180. dmart_migrations/versions/10d2041b94d4_last_checksum_history.py +62 -0
  181. dmart_migrations/versions/1cf4e1ee3cb8_ext_permission_with_filter_fields_values.py +33 -0
  182. dmart_migrations/versions/26bfe19b49d4_rm_failedloginattempts.py +42 -0
  183. dmart_migrations/versions/3c8bca2219cc_add_otp_table.py +38 -0
  184. dmart_migrations/versions/6675fd9dfe42_remove_unique_from_sessions_table.py +36 -0
  185. dmart_migrations/versions/71bc1df82e6a_adding_user_last_login_at.py +43 -0
  186. dmart_migrations/versions/74288ccbd3b5_initial.py +264 -0
  187. dmart_migrations/versions/7520a89a8467_rm_activesession_table.py +39 -0
  188. dmart_migrations/versions/848b623755a4_make_created_nd_updated_at_required.py +138 -0
  189. dmart_migrations/versions/8640dcbebf85_add_notes_to_users.py +32 -0
  190. dmart_migrations/versions/91c94250232a_adding_fk_on_owner_shortname.py +104 -0
  191. dmart_migrations/versions/98ecd6f56f9a_ext_meta_with_owner_group_shortname.py +66 -0
  192. dmart_migrations/versions/9aae9138c4ef_indexing_created_at_updated_at.py +80 -0
  193. dmart_migrations/versions/__init__.py +0 -0
  194. dmart_migrations/versions/__pycache__/0f3d2b1a7c21_add_authz_materialized_views.cpython-314.pyc +0 -0
  195. dmart_migrations/versions/__pycache__/10d2041b94d4_last_checksum_history.cpython-314.pyc +0 -0
  196. dmart_migrations/versions/__pycache__/1cf4e1ee3cb8_ext_permission_with_filter_fields_values.cpython-314.pyc +0 -0
  197. dmart_migrations/versions/__pycache__/26bfe19b49d4_rm_failedloginattempts.cpython-314.pyc +0 -0
  198. dmart_migrations/versions/__pycache__/3c8bca2219cc_add_otp_table.cpython-314.pyc +0 -0
  199. dmart_migrations/versions/__pycache__/6675fd9dfe42_remove_unique_from_sessions_table.cpython-314.pyc +0 -0
  200. dmart_migrations/versions/__pycache__/71bc1df82e6a_adding_user_last_login_at.cpython-314.pyc +0 -0
  201. dmart_migrations/versions/__pycache__/74288ccbd3b5_initial.cpython-314.pyc +0 -0
  202. dmart_migrations/versions/__pycache__/7520a89a8467_rm_activesession_table.cpython-314.pyc +0 -0
  203. dmart_migrations/versions/__pycache__/848b623755a4_make_created_nd_updated_at_required.cpython-314.pyc +0 -0
  204. dmart_migrations/versions/__pycache__/8640dcbebf85_add_notes_to_users.cpython-314.pyc +0 -0
  205. dmart_migrations/versions/__pycache__/91c94250232a_adding_fk_on_owner_shortname.cpython-314.pyc +0 -0
  206. dmart_migrations/versions/__pycache__/98ecd6f56f9a_ext_meta_with_owner_group_shortname.cpython-314.pyc +0 -0
  207. dmart_migrations/versions/__pycache__/9aae9138c4ef_indexing_created_at_updated_at.cpython-314.pyc +0 -0
  208. dmart_migrations/versions/__pycache__/b53f916b3f6d_json_to_jsonb.cpython-314.pyc +0 -0
  209. dmart_migrations/versions/__pycache__/eb5f1ec65156_adding_user_locked_to_device.cpython-314.pyc +0 -0
  210. dmart_migrations/versions/__pycache__/f7a4949eed19_adding_query_policies_to_meta.cpython-314.pyc +0 -0
  211. dmart_migrations/versions/b53f916b3f6d_json_to_jsonb.py +492 -0
  212. dmart_migrations/versions/eb5f1ec65156_adding_user_locked_to_device.py +36 -0
  213. dmart_migrations/versions/f7a4949eed19_adding_query_policies_to_meta.py +60 -0
  214. get_settings.py +7 -0
  215. info.json +1 -0
  216. languages/__init__.py +0 -0
  217. languages/arabic.json +15 -0
  218. languages/english.json +16 -0
  219. languages/kurdish.json +14 -0
  220. languages/loader.py +12 -0
  221. main.py +560 -0
  222. migrate.py +24 -0
  223. models/__init__.py +0 -0
  224. models/api.py +203 -0
  225. models/core.py +597 -0
  226. models/enums.py +255 -0
  227. password_gen.py +8 -0
  228. plugins/__init__.py +0 -0
  229. plugins/action_log/__init__.py +0 -0
  230. plugins/action_log/plugin.py +121 -0
  231. plugins/admin_notification_sender/__init__.py +0 -0
  232. plugins/admin_notification_sender/plugin.py +124 -0
  233. plugins/ldap_manager/__init__.py +0 -0
  234. plugins/ldap_manager/plugin.py +100 -0
  235. plugins/local_notification/__init__.py +0 -0
  236. plugins/local_notification/plugin.py +123 -0
  237. plugins/realtime_updates_notifier/__init__.py +0 -0
  238. plugins/realtime_updates_notifier/plugin.py +58 -0
  239. plugins/redis_db_update/__init__.py +0 -0
  240. plugins/redis_db_update/plugin.py +188 -0
  241. plugins/resource_folders_creation/__init__.py +0 -0
  242. plugins/resource_folders_creation/plugin.py +81 -0
  243. plugins/system_notification_sender/__init__.py +0 -0
  244. plugins/system_notification_sender/plugin.py +188 -0
  245. plugins/update_access_controls/__init__.py +0 -0
  246. plugins/update_access_controls/plugin.py +9 -0
  247. pytests/__init__.py +0 -0
  248. pytests/api_user_models_erros_test.py +16 -0
  249. pytests/api_user_models_requests_test.py +98 -0
  250. pytests/archive_test.py +72 -0
  251. pytests/base_test.py +300 -0
  252. pytests/get_settings_test.py +14 -0
  253. pytests/json_to_db_migration_test.py +237 -0
  254. pytests/service_test.py +26 -0
  255. pytests/test_info.py +55 -0
  256. pytests/test_status.py +15 -0
  257. run_notification_campaign.py +85 -0
  258. scheduled_notification_handler.py +121 -0
  259. schema_migration.py +208 -0
  260. schema_modulate.py +192 -0
  261. set_admin_passwd.py +55 -0
  262. sync.py +202 -0
  263. utils/__init__.py +0 -0
  264. utils/access_control.py +306 -0
  265. utils/async_request.py +8 -0
  266. utils/exporter.py +309 -0
  267. utils/firebase_notifier.py +57 -0
  268. utils/generate_email.py +37 -0
  269. utils/helpers.py +352 -0
  270. utils/hypercorn_config.py +12 -0
  271. utils/internal_error_code.py +60 -0
  272. utils/jwt.py +124 -0
  273. utils/logger.py +167 -0
  274. utils/middleware.py +99 -0
  275. utils/notification.py +75 -0
  276. utils/password_hashing.py +16 -0
  277. utils/plugin_manager.py +202 -0
  278. utils/query_policies_helper.py +128 -0
  279. utils/regex.py +44 -0
  280. utils/repository.py +529 -0
  281. utils/router_helper.py +19 -0
  282. utils/settings.py +166 -0
  283. utils/sms_notifier.py +21 -0
  284. utils/social_sso.py +67 -0
  285. utils/templates/activation.html.j2 +26 -0
  286. utils/templates/reminder.html.j2 +17 -0
  287. utils/ticket_sys_utils.py +203 -0
  288. utils/web_notifier.py +29 -0
  289. websocket.py +231 -0
api/qr/__init__.py ADDED
File without changes
api/qr/router.py ADDED
@@ -0,0 +1,108 @@
1
+ from time import time
2
+ from fastapi import APIRouter, Depends, Path, Body, status
3
+ from fastapi.responses import StreamingResponse
4
+ from api.managed.router import retrieve_entry_meta
5
+ from utils.internal_error_code import InternalErrorCode
6
+ from utils.jwt import JWTBearer
7
+ from io import BytesIO
8
+ import models.api as api
9
+ import utils.regex as regex
10
+ import hmac
11
+ import hashlib
12
+ from utils.settings import settings
13
+ from models.enums import ResourceType
14
+ from fastapi.responses import ORJSONResponse
15
+
16
+ router = APIRouter(default_response_class=ORJSONResponse)
17
+
18
+
19
+ @router.get("/generate/{resource_type}/{space_name}/{subpath:path}/{shortname}")
20
+ async def generate_qr_user_profile(
21
+ resource_type: ResourceType = Path(...),
22
+ space_name: str = Path(..., pattern=regex.SPACENAME, examples=["data"]),
23
+ subpath: str = Path(..., pattern=regex.SUBPATH, examples=["/content"]),
24
+ shortname: str = Path(..., pattern=regex.SHORTNAME,
25
+ examples=["unique_shortname"]),
26
+ logged_in_user=Depends(JWTBearer()),
27
+ ) -> StreamingResponse:
28
+ data: str | dict = await retrieve_entry_meta(
29
+ resource_type,
30
+ space_name,
31
+ subpath,
32
+ shortname,
33
+ logged_in_user=logged_in_user,
34
+ )
35
+
36
+ if (
37
+ isinstance(data, dict)
38
+ and data.get("owner_shortname") != logged_in_user
39
+ and f"management/{subpath}/{data['shortname']}"
40
+ != f"{space_name}/users/{logged_in_user}"
41
+ ):
42
+ raise api.Exception(
43
+ status.HTTP_400_BAD_REQUEST,
44
+ api.Error(type="qr", code=InternalErrorCode.QR_ERROR,
45
+ message="QR cannot be generated"),
46
+ )
47
+ m = hmac.new(settings.jwt_secret.encode(), digestmod=hashlib.sha256)
48
+ data = f"{resource_type}/{space_name}/{subpath}/{shortname}"
49
+ date = int(time())
50
+ m.update(f"{date}.{data}".encode())
51
+ hexed_data = m.hexdigest()
52
+
53
+ try:
54
+ segno = __import__("segno")
55
+ qrcode = segno.make(f"{date}.{hexed_data}")
56
+ v_path = BytesIO()
57
+ qrcode.save(v_path, kind="png", dpi=600, scale=10)
58
+
59
+ return StreamingResponse(iter([v_path.getvalue()]), media_type="image/png")
60
+ except ModuleNotFoundError:
61
+ raise api.Exception(
62
+ status.HTTP_400_BAD_REQUEST,
63
+ api.Error(
64
+ type="request",
65
+ code=InternalErrorCode.NOT_ALLOWED,
66
+ message="segno is not installed!",
67
+ ),
68
+ )
69
+
70
+
71
+ @router.post("/validate")
72
+ async def validate_qr_user_profile(
73
+ resource_type: ResourceType = Body(...),
74
+ space_name: str = Body(..., pattern=regex.SPACENAME, examples=["data"]),
75
+ subpath: str = Body(..., pattern=regex.SUBPATH, examples=["/content"]),
76
+ shortname: str = Body(..., pattern=regex.SHORTNAME,
77
+ examples=["unique_shortname"]),
78
+ logged_in_user=Depends(JWTBearer()),
79
+ qr_data: str = Body(..., embed=True),
80
+ ):
81
+ await retrieve_entry_meta(
82
+ resource_type,
83
+ space_name,
84
+ subpath,
85
+ shortname,
86
+ logged_in_user=logged_in_user,
87
+ )
88
+ arr_data = qr_data.split(".")
89
+ req_date, req_data = arr_data[0], arr_data[1]
90
+ if int(req_date) + 60 < int(time()):
91
+ raise api.Exception(
92
+ status.HTTP_400_BAD_REQUEST,
93
+ api.Error(type="qr", code=InternalErrorCode.QR_EXPIRED,
94
+ message="QR did expire"),
95
+ )
96
+ data = f"{resource_type}/{space_name}/{subpath}/{shortname}"
97
+ m = hmac.new(settings.jwt_secret.encode(), digestmod=hashlib.sha256)
98
+ m.update(f"{req_date}.{data}".encode())
99
+ hexed_data = m.hexdigest()
100
+
101
+ if hexed_data == req_data:
102
+ return api.Response(status=api.Status.success)
103
+ else:
104
+ raise api.Exception(
105
+ status.HTTP_400_BAD_REQUEST,
106
+ api.Error(type="qr", code=InternalErrorCode.QR_INVALID,
107
+ message="Invalid data passed"),
108
+ )
api/user/__init__.py ADDED
File without changes
File without changes
@@ -0,0 +1,14 @@
1
+ from models.api import Error
2
+ from utils.internal_error_code import InternalErrorCode
3
+
4
+ INVALID_OTP = Error(
5
+ type="OTP",
6
+ code=InternalErrorCode.OTP_INVALID,
7
+ message="Invalid OTP",
8
+ )
9
+
10
+ EXPIRED_OTP = Error(
11
+ type="OTP",
12
+ code=InternalErrorCode.OTP_EXPIRED,
13
+ message="Expired OTP",
14
+ )
@@ -0,0 +1,165 @@
1
+ from typing import Dict
2
+ from pydantic import BaseModel, Field
3
+ from utils.internal_error_code import InternalErrorCode
4
+ import utils.regex as rgx
5
+ from models.api import Exception, Error
6
+ from models.enums import StrEnum
7
+
8
+
9
+ class OTPType(StrEnum):
10
+ SMS = "SMS"
11
+ EMAIL = "EMAIL"
12
+
13
+
14
+ class SendOTPRequest(BaseModel):
15
+ shortname: str | None = Field(None, pattern=rgx.SHORTNAME)
16
+ msisdn: str | None = Field(None, pattern=rgx.MSISDN)
17
+ email: str | None = Field(None, pattern=rgx.EMAIL)
18
+
19
+ def check_fields(self) -> Dict[str, str]:
20
+ if self.email is None and self.msisdn is None and self.shortname is None:
21
+ raise Exception(
22
+ 422,
23
+ Error(
24
+ type="OTP",
25
+ code=InternalErrorCode.EMAIL_OR_MSISDN_REQUIRED,
26
+ message="One of these [email, msisdn, shortname] should be set!",
27
+ ),
28
+ )
29
+
30
+ if [self.email, self.msisdn, self.shortname].count(None) != 2:
31
+ raise Exception(
32
+ 422,
33
+ Error(
34
+ type="OTP",
35
+ code=InternalErrorCode.INVALID_STANDALONE_DATA,
36
+ message="Too many input has been passed",
37
+ ),
38
+ )
39
+
40
+ elif self.msisdn:
41
+ return {"msisdn": self.msisdn}
42
+ elif self.email:
43
+ return {"email": self.email}
44
+ elif self.shortname:
45
+ return {"shortname": self.shortname}
46
+
47
+ raise Exception(
48
+ 500,
49
+ Error(
50
+ type="OTP",
51
+ code=InternalErrorCode.OTP_ISSUE,
52
+ message="Something went wrong",
53
+ ),
54
+ )
55
+
56
+ model_config = {
57
+ "json_schema_extra": {
58
+ "examples": [
59
+ {
60
+ "msisdn": "7777778110"
61
+ }
62
+ ]
63
+ }
64
+ }
65
+
66
+ class PasswordResetRequest(BaseModel):
67
+ msisdn: str | None = Field(None, pattern=rgx.MSISDN)
68
+ shortname: str | None = Field(None, pattern=rgx.SHORTNAME)
69
+ email: str | None = Field(None, pattern=rgx.EMAIL)
70
+
71
+ def check_fields(self) -> Dict[str, str]:
72
+ if self.email is None and self.msisdn is None and self.shortname is None:
73
+ raise Exception(
74
+ 422,
75
+ Error(
76
+ type="OTP",
77
+ code=InternalErrorCode.EMAIL_OR_MSISDN_REQUIRED,
78
+ message="One of these [shortname, email, msisdn] should be set!",
79
+ ),
80
+ )
81
+
82
+ if [self.email, self.msisdn, self.shortname].count(None) != 2:
83
+ raise Exception(
84
+ 422,
85
+ Error(
86
+ type="OTP",
87
+ code=InternalErrorCode.INVALID_STANDALONE_DATA,
88
+ message="Too many input has been passed",
89
+ ),
90
+ )
91
+
92
+ elif self.msisdn:
93
+ return {"msisdn": self.msisdn}
94
+ elif self.email:
95
+ return {"email": self.email}
96
+ elif self.shortname:
97
+ return {"shortname": self.shortname}
98
+
99
+ raise Exception(
100
+ 500,
101
+ Error(
102
+ type="password_reset",
103
+ code=InternalErrorCode.PASSWORD_RESET_ERROR,
104
+ message="Something went wrong",
105
+ ),
106
+ )
107
+
108
+ model_config = {
109
+ "json_schema_extra": {
110
+ "examples": [
111
+ {
112
+ "msisdn": "7777778110"
113
+ }
114
+ ]
115
+ }
116
+ }
117
+ class ConfirmOTPRequest(SendOTPRequest, BaseModel):
118
+ code: str = Field(..., pattern=rgx.OTP_CODE)
119
+
120
+ model_config = {
121
+ "json_schema_extra": {
122
+ "examples": [
123
+ {
124
+ "code": "84293201"
125
+ }
126
+ ]
127
+ }
128
+ }
129
+
130
+ class UserLoginRequest(BaseModel):
131
+ shortname: str | None = Field(None, pattern=rgx.SHORTNAME)
132
+ email: str | None = Field(None, pattern=rgx.EMAIL)
133
+ msisdn: str | None = Field(None, pattern=rgx.MSISDN)
134
+ password: str | None = Field(None)
135
+ invitation: str | None = Field(None, pattern=rgx.INVITATION)
136
+ firebase_token: str | None = Field(None)
137
+ otp: str | None = Field(None)
138
+
139
+ def check_fields(self) -> Dict[str, str] | None:
140
+ if self.shortname is None and self.email is None and self.msisdn is None:
141
+ return {}
142
+ # raise ValueError("One of these [shortname, email, msisdn] should be set!")
143
+
144
+ if [self.shortname, self.email, self.msisdn].count(None) != 2:
145
+ raise ValueError("Too many input has been passed")
146
+
147
+ if self.shortname:
148
+ return {"shortname": self.shortname}
149
+ elif self.msisdn:
150
+ return {"msisdn": self.msisdn}
151
+ elif self.email:
152
+ return {"email": self.email}
153
+
154
+ return None
155
+
156
+ model_config = {
157
+ "json_schema_extra": {
158
+ "examples": [
159
+ {
160
+ "shortname": "john_doo",
161
+ "password": "my_secure_password_@_93301"
162
+ }
163
+ ]
164
+ }
165
+ }
@@ -0,0 +1,11 @@
1
+ from pydantic import BaseModel, Field
2
+
3
+ from models.api import Response
4
+
5
+
6
+ class Confirmation(BaseModel):
7
+ confirmation: str = Field(...)
8
+
9
+
10
+ class ConfirmationResponse(Response):
11
+ data: Confirmation