@vex-chat/spire 0.8.0 → 1.0.0

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 (370) hide show
  1. package/README.md +82 -26
  2. package/dist/ClientManager.d.ts +23 -25
  3. package/dist/ClientManager.js +230 -249
  4. package/dist/ClientManager.js.map +1 -0
  5. package/dist/Database.d.ts +49 -47
  6. package/dist/Database.js +698 -773
  7. package/dist/Database.js.map +1 -0
  8. package/dist/Spire.d.ts +22 -14
  9. package/dist/Spire.js +496 -236
  10. package/dist/Spire.js.map +1 -0
  11. package/dist/__tests__/Database.spec.js +116 -75
  12. package/dist/__tests__/Database.spec.js.map +1 -0
  13. package/dist/db/schema.d.ts +134 -0
  14. package/dist/db/schema.js +2 -0
  15. package/dist/db/schema.js.map +1 -0
  16. package/dist/index.d.ts +1 -1
  17. package/dist/index.js +3 -5
  18. package/dist/index.js.map +1 -0
  19. package/dist/middleware/validate.d.ts +12 -0
  20. package/dist/middleware/validate.js +35 -0
  21. package/dist/middleware/validate.js.map +1 -0
  22. package/dist/migrations/2026-04-06_initial-schema.d.ts +3 -0
  23. package/dist/migrations/2026-04-06_initial-schema.js +192 -0
  24. package/dist/migrations/2026-04-06_initial-schema.js.map +1 -0
  25. package/dist/run.js +26 -21
  26. package/dist/run.js.map +1 -0
  27. package/dist/server/avatar.d.ts +3 -4
  28. package/dist/server/avatar.js +64 -64
  29. package/dist/server/avatar.js.map +1 -0
  30. package/dist/server/errors.d.ts +59 -0
  31. package/dist/server/errors.js +94 -0
  32. package/dist/server/errors.js.map +1 -0
  33. package/dist/server/file.d.ts +3 -4
  34. package/dist/server/file.js +81 -62
  35. package/dist/server/file.js.map +1 -0
  36. package/dist/server/index.d.ts +8 -10
  37. package/dist/server/index.js +414 -405
  38. package/dist/server/index.js.map +1 -0
  39. package/dist/server/invite.d.ts +4 -5
  40. package/dist/server/invite.js +18 -52
  41. package/dist/server/invite.js.map +1 -0
  42. package/dist/server/openapi.d.ts +2 -0
  43. package/dist/server/openapi.js +40 -0
  44. package/dist/server/openapi.js.map +1 -0
  45. package/dist/server/permissions.d.ts +16 -0
  46. package/dist/server/permissions.js +22 -0
  47. package/dist/server/permissions.js.map +1 -0
  48. package/dist/server/rateLimit.d.ts +28 -0
  49. package/dist/server/rateLimit.js +58 -0
  50. package/dist/server/rateLimit.js.map +1 -0
  51. package/dist/server/user.d.ts +4 -7
  52. package/dist/server/user.js +55 -66
  53. package/dist/server/user.js.map +1 -0
  54. package/dist/server/utils.d.ts +35 -7
  55. package/dist/server/utils.js +50 -6
  56. package/dist/server/utils.js.map +1 -0
  57. package/dist/types/express.d.ts +20 -0
  58. package/dist/types/express.js +2 -0
  59. package/dist/types/express.js.map +1 -0
  60. package/dist/utils/createLogger.js +13 -19
  61. package/dist/utils/createLogger.js.map +1 -0
  62. package/dist/utils/createUint8UUID.js +6 -10
  63. package/dist/utils/createUint8UUID.js.map +1 -0
  64. package/dist/utils/jwtSecret.d.ts +7 -0
  65. package/dist/utils/jwtSecret.js +15 -0
  66. package/dist/utils/jwtSecret.js.map +1 -0
  67. package/dist/utils/loadEnv.js +7 -22
  68. package/dist/utils/loadEnv.js.map +1 -0
  69. package/dist/utils/msgpack.d.ts +2 -0
  70. package/dist/utils/msgpack.js +4 -0
  71. package/dist/utils/msgpack.js.map +1 -0
  72. package/package.json +91 -65
  73. package/src/ClientManager.ts +434 -0
  74. package/src/Database.ts +925 -0
  75. package/src/Spire.ts +878 -0
  76. package/src/__tests__/Database.spec.ts +167 -0
  77. package/src/ambient-modules.d.ts +1 -0
  78. package/src/db/schema.ts +165 -0
  79. package/src/index.ts +3 -0
  80. package/src/middleware/validate.ts +38 -0
  81. package/src/migrations/2026-04-06_initial-schema.ts +218 -0
  82. package/src/run.ts +37 -0
  83. package/src/server/avatar.ts +141 -0
  84. package/src/server/errors.ts +133 -0
  85. package/src/server/file.ts +172 -0
  86. package/src/server/index.ts +855 -0
  87. package/src/server/invite.ts +65 -0
  88. package/src/server/openapi.ts +51 -0
  89. package/src/server/permissions.ts +40 -0
  90. package/src/server/rateLimit.ts +86 -0
  91. package/src/server/user.ts +125 -0
  92. package/src/server/utils.ts +59 -0
  93. package/src/types/express.ts +23 -0
  94. package/src/utils/createLogger.ts +47 -0
  95. package/src/utils/createUint8UUID.ts +9 -0
  96. package/src/utils/jwtSecret.ts +16 -0
  97. package/src/utils/loadEnv.ts +15 -0
  98. package/src/utils/msgpack.ts +4 -0
  99. package/avatars/052242d0-4129-4a6e-8076-22709c157549 +0 -0
  100. package/avatars/0a677c9a-4986-4b2c-ae2e-12faf22f55db +0 -0
  101. package/avatars/0ba21b91-decb-4a3e-ac86-4dd54d805a9a +0 -0
  102. package/avatars/0c48d8b6-1d1b-4297-a6c6-2fe50af6fc35 +0 -0
  103. package/avatars/0d993cdf-19a6-4299-a4ee-a06579d106cf +0 -0
  104. package/avatars/17b000e4-ac38-46ec-9dec-d2568086129a +0 -0
  105. package/avatars/19dc5594-0f06-4ac6-af18-8740dd39ef6b +0 -0
  106. package/avatars/20444fa3-6d5e-429e-b55f-b81c3d2c61ee +0 -0
  107. package/avatars/21c0512a-5630-4931-9442-d66db66737be +0 -0
  108. package/avatars/22830a60-0b6f-4912-83a5-72245465f332 +0 -0
  109. package/avatars/243639ce-f59f-4404-a1f1-4ec0eb5d2af3 +0 -0
  110. package/avatars/30d2c01d-7b7f-4ea9-9859-1c90837a23f7 +0 -0
  111. package/avatars/315a04f0-9a6f-4b0f-bb9f-5fa774c4752b +0 -0
  112. package/avatars/3563d333-53fe-4885-ac2d-9a4f761db85e +0 -0
  113. package/avatars/36a10c00-3b4c-437f-8e1f-4428ecde0003 +0 -0
  114. package/avatars/40b83eeb-c6e8-4268-82ab-69799a796405 +0 -0
  115. package/avatars/45b5ddb9-ad2c-4404-8ab3-cb4699e6d61c +0 -0
  116. package/avatars/4e4f0ffb-9a75-479a-bccb-446d0bf85020 +0 -0
  117. package/avatars/4e62c3bd-08c6-4fdd-bd65-f01c7322ed64 +0 -0
  118. package/avatars/5004d2e7-51af-44af-8776-6c71f7019843 +0 -0
  119. package/avatars/5041eb29-5c4b-4dea-8c1b-31ba4473161a +0 -0
  120. package/avatars/5065cf78-31c5-46cb-8d5c-c0b6be2d994e +0 -0
  121. package/avatars/51b91d2c-8956-4d73-b4ad-ca6a8d9da9a8 +0 -0
  122. package/avatars/58264a2c-5651-4a42-8ca2-a9907b311e48 +0 -0
  123. package/avatars/58c2357c-8080-4725-a0ce-182c96b037c4 +0 -0
  124. package/avatars/59b5f6dd-8e04-4d15-b4dc-c1c652558a74 +0 -0
  125. package/avatars/5b417a78-b274-48bc-98a4-6e54b74ee62d +0 -0
  126. package/avatars/611f5a93-1ed4-45a1-bc8e-e8e413f9b171 +0 -0
  127. package/avatars/65abe919-9921-46f6-9bc9-183e9cc53c8a +0 -0
  128. package/avatars/6934202d-1546-4270-8a15-97ba8b8c6fa9 +0 -0
  129. package/avatars/6acd24be-f4b7-4399-9e7c-807821828d29 +0 -0
  130. package/avatars/6b2d6ac5-e35c-4297-994e-f0eb6fe56740 +0 -0
  131. package/avatars/6cae9ddd-f163-44e7-a632-30425716a159 +0 -0
  132. package/avatars/6d90e79e-b9c1-4b89-843b-96636de8d26b +0 -0
  133. package/avatars/6f82c7bb-a974-4372-8a64-ce287e668c8c +0 -0
  134. package/avatars/74a45091-5a76-4bb7-ae8a-cd7373adc128 +0 -0
  135. package/avatars/7d071ab2-e0b5-4dbb-8bf5-1fae50c3663f +0 -0
  136. package/avatars/7de818c8-bda1-4f51-976e-160fc087184b +0 -0
  137. package/avatars/873937df-8cb1-427a-92bf-f829f4259624 +0 -0
  138. package/avatars/8b45ffa5-3322-4109-bfa0-1be088336135 +0 -0
  139. package/avatars/8efaa426-22a9-42a2-b4ac-a275717b812f +0 -0
  140. package/avatars/903fd1a6-d6ea-431c-b98f-f21e424a2852 +0 -0
  141. package/avatars/943d8533-5174-4199-990f-1ec69e5d60c4 +0 -0
  142. package/avatars/952d014e-3804-4cd2-a4a0-ffe40a11e4ac +0 -0
  143. package/avatars/95d3acb0-724d-4413-b20f-edad55812d5d +0 -0
  144. package/avatars/9641a946-f613-471b-bedd-c1730b96b51e +0 -0
  145. package/avatars/9b01cbbf-f6b2-43fb-b569-589b6f2a8134 +0 -0
  146. package/avatars/9cd4424d-a34f-4467-acc0-93cf82703e0d +0 -0
  147. package/avatars/9d9ad3b0-e5a6-420a-a6c5-fb9085b70376 +0 -0
  148. package/avatars/9e9e34b5-4e63-4c4b-9722-c7f5674b47aa +0 -0
  149. package/avatars/a387d5c1-59eb-4a6b-80c1-a8982ed12c33 +0 -0
  150. package/avatars/a3e86d21-d881-4824-8ebf-45e3bf0f9186 +0 -0
  151. package/avatars/a8d5cc1c-3f42-4b7b-8d33-f9a9ef77f96b +0 -0
  152. package/avatars/a91d815f-badc-4604-a7be-6c7a44e6101d +0 -0
  153. package/avatars/aa8d0324-bcec-4737-a8c4-bdbff914148d +0 -0
  154. package/avatars/abb8a941-8b6b-47d7-a2f9-8b153ba44aa2 +0 -0
  155. package/avatars/b011bb38-1ef3-4d22-82fa-8bf60faf7b5d +0 -0
  156. package/avatars/b24bcbc1-11f0-473d-a8b9-ba8ce4ca127d +0 -0
  157. package/avatars/b2607346-af1c-4e98-b725-7650a766db2a +0 -0
  158. package/avatars/b6300f7c-cb37-459b-b1bf-8a0a0e797a52 +0 -0
  159. package/avatars/b7d3cff3-84dd-4547-93a1-de1aaa8aa34c +0 -0
  160. package/avatars/baa4b51c-e97f-4f51-bcb3-f27bc506cfaf +0 -0
  161. package/avatars/be7022e4-e292-4515-80d5-f9b61ebeb4ce +0 -0
  162. package/avatars/bed596a3-7569-4854-9e76-f52d33c0a541 +0 -0
  163. package/avatars/bf69992f-3f72-4930-99bb-0ffe17f3aebf +0 -0
  164. package/avatars/ca00c250-c6d4-464d-a6de-1c8467a18fe8 +0 -0
  165. package/avatars/ca19d78f-c0bc-4bd5-b26f-6923cb19996d +0 -0
  166. package/avatars/cda4d6e1-e0a4-4024-ac95-6de98e713b98 +0 -0
  167. package/avatars/cf72c30d-2da8-4e81-aa71-735b9e714274 +0 -0
  168. package/avatars/d5a35b78-99b3-4564-b6b9-b2ccab28c470 +0 -0
  169. package/avatars/dadb38c1-2a9d-47a3-8d92-b56b6166973c +0 -0
  170. package/avatars/e68705c7-375d-4423-9a86-29a16bd3ee0e +0 -0
  171. package/avatars/e9af3e4c-1f62-4302-8b99-b68ce93b7a86 +0 -0
  172. package/avatars/ea7e7331-e845-4189-8248-5f5b1d63f5e3 +0 -0
  173. package/avatars/ef4f8dcb-ef6c-4e7a-9be1-0476161bfce5 +0 -0
  174. package/avatars/ef7d0917-a206-4f88-8b60-93f8253774dc +0 -0
  175. package/avatars/f1a554d6-1db3-4ff5-b0dc-b607d6c3b4ff +0 -0
  176. package/avatars/f1fecc21-c81f-49a8-88f3-f942a0a679f6 +0 -0
  177. package/avatars/f30a2427-1755-4053-813e-129a179e1dd3 +0 -0
  178. package/avatars/f5370717-5109-46a5-a8d7-e1dd996d0615 +0 -0
  179. package/avatars/f6dd7126-1144-4998-bbd3-d4e0fbee2e95 +0 -0
  180. package/avatars/f83413d5-0003-4756-9ece-745fd61cc468 +0 -0
  181. package/avatars/f9b3149e-7ec8-4bb3-a9b9-dcbe66dac197 +0 -0
  182. package/avatars/fa41d70b-857e-4423-bd7d-26ddcddc13b9 +0 -0
  183. package/avatars/fb551ee8-99c7-400b-8e1d-322ce4619998 +0 -0
  184. package/avatars/fe83a6d4-abb0-4ab0-b61d-76a7cc08be84 +0 -0
  185. package/dist/migrations/20210103192527_users.d.ts +0 -3
  186. package/dist/migrations/20210103192527_users.js +0 -30
  187. package/dist/migrations/20210103193502_mail.d.ts +0 -3
  188. package/dist/migrations/20210103193502_mail.js +0 -35
  189. package/dist/migrations/20210103193525_preKeys.d.ts +0 -3
  190. package/dist/migrations/20210103193525_preKeys.js +0 -30
  191. package/dist/migrations/20210103193553_oneTimeKeys.d.ts +0 -3
  192. package/dist/migrations/20210103193553_oneTimeKeys.js +0 -30
  193. package/dist/migrations/20210103193615_servers.d.ts +0 -3
  194. package/dist/migrations/20210103193615_servers.js +0 -28
  195. package/dist/migrations/20210103193729_channels.d.ts +0 -3
  196. package/dist/migrations/20210103193729_channels.js +0 -28
  197. package/dist/migrations/20210103193749_permissions.d.ts +0 -3
  198. package/dist/migrations/20210103193749_permissions.js +0 -30
  199. package/dist/migrations/20210103193801_files.d.ts +0 -3
  200. package/dist/migrations/20210103193801_files.js +0 -28
  201. package/emoji/04d98632-2c86-421b-a407-17f14fe86f8f +0 -0
  202. package/emoji/1160ed6e-1163-4043-9808-4029e863ed30 +0 -0
  203. package/emoji/1547ab18-1635-4a80-a82d-ebbb767b9932 +0 -0
  204. package/emoji/16922521-f6cb-4de4-860c-27916b22c6ba +0 -0
  205. package/emoji/198a9432-0e41-4866-994a-448d4775afcb +0 -0
  206. package/emoji/1be886b3-c9c5-4593-b516-f357ed931f96 +0 -0
  207. package/emoji/1c2b3d1d-637f-4103-b066-4bc4511a3ad7 +0 -0
  208. package/emoji/1efd27e7-b15f-475c-8b32-9159d26b169d +0 -0
  209. package/emoji/270b9409-0ea5-4be2-a239-a8dce13f9c31 +0 -0
  210. package/emoji/27812f76-fee2-49dd-a217-363de6d159dc +0 -0
  211. package/emoji/297ec202-8c24-44c6-aead-689d6d461883 +0 -0
  212. package/emoji/2bf06d86-17cb-4f40-a5ef-bd75d239a1a3 +0 -0
  213. package/emoji/31a75163-1cce-4dc1-b0a2-ecad6a4c500b +0 -0
  214. package/emoji/35235635-fdbd-4273-8428-f3cb3e1e8fd3 +0 -0
  215. package/emoji/3690fff2-6824-4403-a6e3-16a6a54979a9 +0 -0
  216. package/emoji/391014c2-59e0-46a8-85ec-7a7fdaca1d2d +0 -0
  217. package/emoji/3b383dcb-6e76-4e85-8e16-7c68040c06c2 +0 -0
  218. package/emoji/42d617a7-b104-42f5-9618-473181f752cf +0 -0
  219. package/emoji/482495d3-cce9-4f88-bf2a-f6003f03a9b5 +0 -0
  220. package/emoji/48390e06-0efb-404c-89bd-5f2be241bd50 +0 -0
  221. package/emoji/4b808d8d-3248-4149-b919-71b108391bcf +0 -0
  222. package/emoji/4bc13544-d82a-4e32-bd17-a70592274314 +0 -0
  223. package/emoji/4fcebf70-8623-4343-8243-67c8547b2edd +0 -0
  224. package/emoji/509d09aa-1214-459c-8081-50918a17b9af +0 -0
  225. package/emoji/5272abd8-d4d7-4b90-acd2-bf30e6c27243 +0 -0
  226. package/emoji/53c272ce-48bd-4d7e-bfb8-a6482b88be54 +0 -0
  227. package/emoji/5b279e65-06f7-4b26-8b4c-d1b48fba728d +0 -0
  228. package/emoji/5bd141f9-4394-4108-9376-66ebbc2c2bc1 +0 -0
  229. package/emoji/5c769156-f9cb-40bd-ab89-4edeece613fd +0 -0
  230. package/emoji/5c85fba9-8ba7-4fc9-b1b2-48dc30d24a1b +0 -0
  231. package/emoji/61a5e565-d20b-40ba-a139-b0c73a6027f3 +0 -0
  232. package/emoji/6913f43d-dd45-456c-9641-a126104d9ae5 +0 -0
  233. package/emoji/6957e74e-9622-492d-a950-242db3752260 +0 -0
  234. package/emoji/6a14bab5-26af-4bfa-9c17-be7c2511976d +0 -0
  235. package/emoji/6be09439-509e-4095-a30a-b1c7c573895d +0 -0
  236. package/emoji/6cc435b1-fe53-433c-b5a9-2b2019053997 +0 -0
  237. package/emoji/74f1b2af-bc7e-4a0f-802a-64ded185d5e2 +0 -0
  238. package/emoji/7890ba09-f02f-428e-807d-006d03d51d4a +0 -0
  239. package/emoji/7dc69179-6b3c-4f40-b20b-0ff573deea2d +0 -0
  240. package/emoji/7dd1b6b1-439d-4279-916a-995408863172 +0 -0
  241. package/emoji/820498ad-a2c8-43a2-ab83-d26f9c2246d4 +0 -0
  242. package/emoji/8319469c-2787-44e5-91a6-c8c39810dd7c +0 -0
  243. package/emoji/86745d1d-9e59-4607-b2b0-46c741079be1 +0 -0
  244. package/emoji/887b3cff-ae9e-4b5f-ad00-3ca9fc72f689 +0 -0
  245. package/emoji/8c6cf621-71d6-4fca-abe6-e19f4dd7f883 +0 -0
  246. package/emoji/8ca6d32e-a1ef-4956-a416-d8d0d680f085 +0 -0
  247. package/emoji/8d979f5e-38a2-4dd1-b3e4-80938bbe499b +0 -0
  248. package/emoji/99f68ea0-e3fe-4f03-9cc3-5f7f5315404d +0 -0
  249. package/emoji/9ad3aedc-7f79-4d68-a144-82e5b5dc3033 +0 -0
  250. package/emoji/9e418c2f-1f0f-46c4-be39-3bda38a28545 +0 -0
  251. package/emoji/a1f616bf-7402-4e24-9111-18acaebabb48 +0 -0
  252. package/emoji/a25ed9c1-3f9c-4e5f-ade2-7b159fb9fbf4 +0 -0
  253. package/emoji/a5176bc2-39a8-467b-8c75-6fbbc81b59c7 +0 -0
  254. package/emoji/a584215c-6547-438b-8ae8-dd490b51890e +0 -0
  255. package/emoji/a739895f-cf61-4b7c-b350-8e8283aaf751 +0 -0
  256. package/emoji/aaa10dd2-02a2-499e-9e17-c83787436508 +0 -0
  257. package/emoji/ae90baf2-a0ef-4d4d-9cc4-94f8ddd60f45 +0 -0
  258. package/emoji/b0564c48-feae-431a-95f9-df597c6c124c +0 -0
  259. package/emoji/b218bb93-e69c-4793-a669-83316650c4e7 +0 -0
  260. package/emoji/b2998c27-85d5-4598-ab41-469aa8e0fcad +0 -0
  261. package/emoji/b3da08ba-4179-4b5d-826f-5fc15e1a3ad2 +0 -0
  262. package/emoji/b840eb6a-a917-4bb2-854b-8f1022e7904b +0 -0
  263. package/emoji/b84baa76-b4b0-4b83-bc83-78661cb4f1d4 +0 -0
  264. package/emoji/baf69d80-8b1d-4032-855f-605cf0d489c3 +0 -0
  265. package/emoji/bb4d372c-ccd0-4a47-b157-b6a3b9f763e2 +0 -0
  266. package/emoji/bdbd1627-c81d-42d9-b3f5-8979e2ab74dc +0 -0
  267. package/emoji/c257388f-8b85-450b-b168-ebdf8d8c3026 +0 -0
  268. package/emoji/c573fde1-faa9-4c1d-a172-e283645afcfd +0 -0
  269. package/emoji/c8e27810-e8ea-47ce-b7ec-cef0a6becb28 +0 -0
  270. package/emoji/cdeef182-b220-4850-9ecf-5d7c472fd754 +0 -0
  271. package/emoji/d59c1aa9-6f81-4c07-96dd-9953401ff211 +0 -0
  272. package/emoji/dd407dbf-a077-40ba-957f-337b3c5efdc7 +0 -0
  273. package/emoji/e01f6e06-5728-4e2c-90fb-314a5827b766 +0 -0
  274. package/emoji/e1f9ed12-a2ce-433d-b454-b833438a1f9c +0 -0
  275. package/emoji/e697e5c4-acd6-41cd-a43e-edee8da3ab7b +0 -0
  276. package/emoji/e8182220-3464-4e31-8c08-466baead7bfc +0 -0
  277. package/emoji/eb0f3fd5-abc9-4abf-b816-d8458aeb7ec8 +0 -0
  278. package/emoji/eb38ecf7-0d13-4c51-b96a-3777f79321c4 +0 -0
  279. package/emoji/ee515c85-b7ce-4493-a427-994cc0af0d59 +0 -0
  280. package/emoji/f485cef2-d3fa-4d59-88af-b79a3105cacf +0 -0
  281. package/emoji/ff0dab2a-7015-4e8c-b0d0-3569058359dc +0 -0
  282. package/files/01087968-07b6-4fdb-9aeb-fa9dc061be94 +0 -0
  283. package/files/030455b3-17cc-415f-b3b3-2bb56c92ee8b +0 -0
  284. package/files/06129f52-a858-4031-ad85-4d7f2bb793af +0 -0
  285. package/files/0a6155a9-069f-45e9-8a06-56992fe55187 +0 -0
  286. package/files/12b9dda5-feb1-4f20-a987-ad422db5ba73 +0 -0
  287. package/files/13c8fa1e-8821-4628-b607-9e4fa4510df7 +0 -0
  288. package/files/159b0ad3-1a30-419d-af22-28a1096ce825 +0 -0
  289. package/files/1699c563-6769-431b-8041-99a6a4386f25 +0 -0
  290. package/files/176b916c-0dd9-4b93-bfdc-b8ccabf15a96 +0 -0
  291. package/files/1a27dad9-8cc9-4a0c-9d4e-7bde63adda60 +0 -0
  292. package/files/1d29832c-059a-4190-bca6-b83ac77540d9 +0 -0
  293. package/files/1dcde013-8833-4369-8726-81236e4eb30e +0 -0
  294. package/files/2080fc9f-d2c4-4fbc-af84-232fe4900a4f +0 -0
  295. package/files/20889623-6869-46a2-999b-c07708c12521 +0 -0
  296. package/files/2107e243-c378-418d-a183-7df13873c65b +0 -0
  297. package/files/225ed61e-8f8c-4d17-b675-9a9f9918d5b4 +0 -0
  298. package/files/29ffba15-5acc-4ef8-b5a4-5ce61d3f6e85 +0 -0
  299. package/files/2a69434c-1d8a-4e9d-90d4-569aaeaae7e3 +0 -0
  300. package/files/2aec8fcf-25bf-478e-b2f6-fe67ad753071 +0 -0
  301. package/files/2c64490e-c7de-4cb2-b22e-70e2ac69d88f +0 -0
  302. package/files/2d80b4f4-6389-4f5d-8d32-f0ed93820907 +0 -0
  303. package/files/2f9dd26e-363f-4445-8ee5-28548007a33a +0 -0
  304. package/files/360dd8f5-76c4-4e86-ba22-9025dc7ca2a4 +0 -0
  305. package/files/3a0a7f5b-45a5-4340-ba7c-6a0fa2c63871 +0 -0
  306. package/files/3a8fa6be-8acf-4b7b-b653-25edc6b28cdc +0 -5
  307. package/files/3d9e191b-2c15-42aa-9928-c2cdbb5e14ca +0 -0
  308. package/files/3dd0f2ef-0d4e-4837-bffc-22aa645cbe85 +0 -0
  309. package/files/402c4b9b-adbf-4ab6-a21d-c17369b48abf +0 -0
  310. package/files/4685d988-33eb-4902-872f-3b824f497c8b +0 -0
  311. package/files/495f6c55-07f8-4713-a444-a2261a789b94 +0 -0
  312. package/files/4acba8e3-567b-4062-b81b-340e205a01de +0 -0
  313. package/files/4c0a10cc-395b-474a-8140-677ed607da89 +0 -0
  314. package/files/537584d9-25ad-4830-808a-a1e3d63e2a52 +0 -0
  315. package/files/5519f10d-745e-4474-8ca8-6c111693704c +0 -0
  316. package/files/5563cf92-a5e3-4be3-867d-9647a02298d4 +0 -0
  317. package/files/55b70d9b-fd58-4800-832c-d6b4521ba6d0 +0 -0
  318. package/files/5623cff3-ce9b-403b-9915-50d2bcbc981c +0 -0
  319. package/files/576314ba-2a1a-4753-8d77-2a9e04f509d1 +0 -0
  320. package/files/58ed97ae-0eac-4e04-add1-76646effa2d5 +0 -0
  321. package/files/68638efd-5389-481d-a841-36164c62c078 +0 -0
  322. package/files/6936d34b-a1f8-4a9d-b005-5544dbdcf5e5 +0 -0
  323. package/files/6bda9e88-3a28-47f7-994e-900ede6bf984 +0 -0
  324. package/files/7024a3b9-a863-4618-9dbd-fa6502017ae0 +0 -0
  325. package/files/707caccb-3780-456d-9056-c20bfdfc0e5b +0 -0
  326. package/files/74b8c73c-032b-4640-a5b3-d30dd270cbcb +0 -0
  327. package/files/767cb262-2974-40f8-8182-f7770b431923 +0 -0
  328. package/files/7a669935-cb48-4349-b26b-7705f8a04fbc +0 -0
  329. package/files/7bc8f678-6ee7-454c-a1f8-bb9358b89c95 +0 -0
  330. package/files/7ccc2699-07ec-4177-a9ce-ee7dc952fda1 +0 -0
  331. package/files/7e0eabf4-b334-4683-8156-ab8d949a0532 +0 -0
  332. package/files/7f0644ab-02a3-4121-adfc-29a7c55cf804 +0 -0
  333. package/files/7ff3266d-f103-48ce-90dd-95b9dfe5fcc9 +0 -0
  334. package/files/836e2e8e-aefd-4b4a-a9b9-bf7436158a8c +0 -0
  335. package/files/875f7ae5-fa23-4fc5-b04a-8433a7f7089c +0 -0
  336. package/files/8ca62da9-f204-49e3-b418-9451661b2904 +0 -0
  337. package/files/9283054c-107f-474d-b61e-f1d0061bcb86 +0 -0
  338. package/files/93b1ce7f-9566-452b-b4be-30f87d3de150 +0 -0
  339. package/files/93fb51c7-e19b-4ac1-9dc3-aeb8da0672ed +0 -0
  340. package/files/9b54a3a1-534a-4ed5-b016-3c74ed4c9edd +0 -0
  341. package/files/9b5beb6f-712d-4969-b127-fd66c9b2a9c6 +0 -0
  342. package/files/9e964fbf-d063-498e-b2e3-79f8d6afcf5f +0 -0
  343. package/files/a66b7a9f-58c2-47a8-a429-a6f0647c6fe9 +0 -0
  344. package/files/a7cbda7d-81ba-40f7-a997-51146af63e5f +0 -0
  345. package/files/ac01e83a-e572-41b6-81ab-c992cff7c170 +0 -0
  346. package/files/ad11a58f-f963-4233-bd29-1658b6b7e600 +0 -0
  347. package/files/ae60a4a8-08b9-4521-a0a3-d015a8b3ed08 +0 -0
  348. package/files/b1d3cf27-8d76-4cf9-aa51-d7c6bfd1b3bf +0 -0
  349. package/files/b2c68863-8554-4ac6-8e99-821f0267cf91 +0 -0
  350. package/files/b3043e01-a771-44af-bf19-5327646ff929 +0 -0
  351. package/files/b6d01b89-def5-4c7c-8e97-a3d88617f8f4 +0 -0
  352. package/files/b8760b32-bb1e-4cd7-a9d6-29c6e0b071bc +0 -0
  353. package/files/ba5e0470-44f7-47b3-bcb5-eeab3ca8c292 +0 -0
  354. package/files/c3969b5f-43ae-43f3-9bdd-d3959c79ca01 +0 -0
  355. package/files/caecd488-dbe6-4a30-a400-bced2ba8dae6 +0 -0
  356. package/files/d7d865b8-3a05-4ed1-b95d-b93dc1ebb9a9 +0 -0
  357. package/files/dbca5f31-cf38-4c1e-83b0-5ec8473196fc +0 -0
  358. package/files/dbf07e82-fff6-4985-bebe-d62c0458bfd0 +0 -0
  359. package/files/dd759c20-eead-4e57-9e74-4d3a2b978e91 +0 -5
  360. package/files/de0b2cf1-981b-4e4a-a04e-ac185d1620cd +0 -0
  361. package/files/de6837fe-5aa2-4ff9-a067-2646a008c780 +0 -0
  362. package/files/e2fde852-91cb-4e01-88a2-ee086b5f227c +0 -0
  363. package/files/e391d1ce-8d39-460c-a462-791730131f7f +0 -0
  364. package/files/e6d9b60f-2b1b-4c6f-ba6d-02f6de90d40f +0 -0
  365. package/files/f693c62d-2ac8-49fd-aa38-30904e013e3c +0 -0
  366. package/files/f749fdf5-193e-41f7-b643-5696f67c6402 +0 -0
  367. package/files/f8910384-e75c-4d65-825f-52a6748f6475 +0 -0
  368. package/files/fad8826b-e952-4acb-a509-3e6543b94d61 +0 -0
  369. package/jest.config.js +0 -13
  370. package/spire.sqlite +0 -0
package/dist/Database.js CHANGED
@@ -1,394 +1,424 @@
1
- "use strict";
2
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
- if (k2 === undefined) k2 = k;
4
- Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
5
- }) : (function(o, m, k, k2) {
6
- if (k2 === undefined) k2 = k;
7
- o[k2] = m[k];
8
- }));
9
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
10
- Object.defineProperty(o, "default", { enumerable: true, value: v });
11
- }) : function(o, v) {
12
- o["default"] = v;
13
- });
14
- var __importStar = (this && this.__importStar) || function (mod) {
15
- if (mod && mod.__esModule) return mod;
16
- var result = {};
17
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
18
- __setModuleDefault(result, mod);
19
- return result;
20
- };
21
- var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
22
- function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
23
- return new (P || (P = Promise))(function (resolve, reject) {
24
- function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
25
- function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
26
- function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
27
- step((generator = generator.apply(thisArg, _arguments || [])).next());
28
- });
29
- };
30
- var __importDefault = (this && this.__importDefault) || function (mod) {
31
- return (mod && mod.__esModule) ? mod : { "default": mod };
1
+ var __rewriteRelativeImportExtension = (this && this.__rewriteRelativeImportExtension) || function (path, preserveJsx) {
2
+ if (typeof path === "string" && /^\.\.?\//.test(path)) {
3
+ return path.replace(/\.(tsx)$|((?:\.d)?)((?:\.[^./]+?)?)\.([cm]?)ts$/i, function (m, tsx, d, ext, cm) {
4
+ return tsx ? preserveJsx ? ".jsx" : ".js" : d && (!ext || !cm) ? m : (d + ext + "." + cm.toLowerCase() + "js");
5
+ });
6
+ }
7
+ return path;
32
8
  };
33
- Object.defineProperty(exports, "__esModule", { value: true });
34
- exports.hashPassword = exports.Database = exports.ITERATIONS = void 0;
35
- const crypto_1 = require("@vex-chat/crypto");
36
- const events_1 = require("events");
37
- const knex_1 = __importDefault(require("knex"));
38
- const pbkdf2_1 = __importDefault(require("pbkdf2"));
39
- const uuid = __importStar(require("uuid"));
40
- const createLogger_1 = require("./utils/createLogger");
9
+ import { EventEmitter } from "events";
10
+ import { pbkdf2Sync } from "node:crypto";
11
+ import * as fs from "node:fs/promises";
12
+ import path from "node:path";
13
+ import { fileURLToPath, pathToFileURL } from "node:url";
14
+ import { xMakeNonce, XUtils } from "@vex-chat/crypto";
15
+ import { MailType } from "@vex-chat/types";
16
+ /**
17
+ * Narrow a plain integer from the `mailType` SQL column to the
18
+ * `MailType` union (0 = initial, 1 = subsequent). Throws if the
19
+ * database contains an unexpected value, catching row corruption
20
+ * at read time instead of propagating an invalid literal into
21
+ * application code.
22
+ */
23
+ function parseMailType(n) {
24
+ if (n === MailType.initial)
25
+ return MailType.initial;
26
+ if (n === MailType.subsequent)
27
+ return MailType.subsequent;
28
+ throw new Error(`Invalid mailType in database row: ${String(n)}`);
29
+ }
30
+ import BetterSqlite3 from "better-sqlite3";
31
+ import { Kysely, Migrator, sql, SqliteDialect } from "kysely";
32
+ import { stringify as uuidStringify, validate as uuidValidate } from "uuid";
33
+ import { createLogger } from "./utils/createLogger.js";
34
+ const __dirname = path.dirname(fileURLToPath(import.meta.url));
35
+ const migrationFolder = path.join(__dirname, "migrations");
36
+ /**
37
+ * Cross-platform Kysely migration provider.
38
+ *
39
+ * Replaces Kysely's built-in `FileMigrationProvider`, which on Windows
40
+ * fails with `ERR_UNSUPPORTED_ESM_URL_SCHEME` because it does
41
+ * `await import(joinedPath)` where `joinedPath` is a Windows absolute
42
+ * path like `D:\\spire\\src\\migrations\\schema.ts`. Node's ESM loader
43
+ * requires `file://` URLs for absolute paths on Windows.
44
+ *
45
+ * This implementation uses `pathToFileURL` to convert each migration
46
+ * file's absolute path to a `file://` URL before passing it to
47
+ * `import()`. Works on Linux, macOS, and Windows. Filters out `.d.ts`
48
+ * declaration files and accepts both `.ts` and `.js` source files for
49
+ * spire's `--experimental-strip-types` runtime.
50
+ */
51
+ class CrossPlatformMigrationProvider {
52
+ folder;
53
+ constructor(folder) {
54
+ this.folder = folder;
55
+ }
56
+ async getMigrations() {
57
+ const files = await fs.readdir(this.folder);
58
+ const migrations = {};
59
+ for (const file of files.sort()) {
60
+ if (file.endsWith(".d.ts"))
61
+ continue;
62
+ if (!file.endsWith(".ts") && !file.endsWith(".js"))
63
+ continue;
64
+ const fullPath = path.join(this.folder, file);
65
+ const fileUrl = pathToFileURL(fullPath).href;
66
+ const mod = await import(__rewriteRelativeImportExtension(fileUrl));
67
+ if (!isMigration(mod)) {
68
+ throw new Error(`Invalid migration ${file}: expected an exported \`up\` function`);
69
+ }
70
+ const name = file.replace(/\.(ts|js)$/, "");
71
+ migrations[name] = mod;
72
+ }
73
+ return migrations;
74
+ }
75
+ }
76
+ function isMigration(mod) {
77
+ return (typeof mod === "object" &&
78
+ mod !== null &&
79
+ "up" in mod &&
80
+ typeof mod.up === "function");
81
+ }
41
82
  const pubkeyRegex = /[0-9a-f]{64}/;
42
- exports.ITERATIONS = 1000;
43
- class Database extends events_1.EventEmitter {
83
+ export const ITERATIONS = 1000;
84
+ // ── Row-to-interface converters ─────────────────────────────────────────
85
+ // SQLite stores booleans as integers and dates as strings, but the
86
+ // @vex-chat/types interfaces expect boolean / Date.
87
+ export class Database extends EventEmitter {
88
+ db;
89
+ log;
44
90
  constructor(options) {
45
91
  super();
46
- this.log = createLogger_1.createLogger("spire-db", (options === null || options === void 0 ? void 0 : options.logLevel) || "error");
47
- switch ((options === null || options === void 0 ? void 0 : options.dbType) || "mysql") {
92
+ this.log = createLogger("spire-db", options?.logLevel || "error");
93
+ const dbType = options?.dbType || "sqlite3";
94
+ let filename;
95
+ switch (dbType) {
96
+ case "sqlite":
48
97
  case "sqlite3":
49
- this.db = knex_1.default({
50
- client: "sqlite3",
51
- connection: {
52
- filename: "spire.sqlite",
53
- },
54
- useNullAsDefault: true,
55
- });
98
+ filename = "spire.sqlite";
56
99
  break;
57
100
  case "sqlite3mem":
58
- this.db = knex_1.default({
59
- client: "sqlite3",
60
- connection: {
61
- filename: ":memory:",
62
- },
63
- useNullAsDefault: true,
64
- });
101
+ filename = ":memory:";
65
102
  break;
66
- case "mysql":
67
103
  default:
68
- this.db = knex_1.default({
69
- client: "mysql",
70
- connection: {
71
- host: process.env.SQL_HOST,
72
- user: process.env.SQL_USER,
73
- password: process.env.SQL_PASSWORD,
74
- database: process.env.SQL_DB_NAME,
75
- },
76
- });
104
+ filename = "spire.sqlite";
77
105
  break;
78
106
  }
79
- this.init();
80
- }
81
- saveOTK(userID, deviceID, otks) {
82
- return __awaiter(this, void 0, void 0, function* () {
83
- for (const otk of otks) {
84
- const newOTK = {
85
- keyID: uuid.v4(),
86
- userID,
87
- deviceID: otk.deviceID,
88
- publicKey: crypto_1.XUtils.encodeHex(otk.publicKey),
89
- signature: crypto_1.XUtils.encodeHex(otk.signature),
90
- index: otk.index,
91
- };
92
- yield this.db("oneTimeKeys").insert(newOTK);
93
- }
94
- });
95
- }
96
- getPreKeys(deviceID) {
97
- return __awaiter(this, void 0, void 0, function* () {
98
- const rows = yield this.db
99
- .from("preKeys")
100
- .select()
101
- .where({
102
- deviceID,
103
- });
104
- if (rows.length === 0) {
105
- return null;
106
- }
107
- const [preKeyInfo] = rows;
108
- const preKey = {
109
- index: preKeyInfo.index,
110
- publicKey: crypto_1.XUtils.decodeHex(preKeyInfo.publicKey),
111
- signature: crypto_1.XUtils.decodeHex(preKeyInfo.signature),
112
- deviceID: preKeyInfo.deviceID,
113
- };
114
- return preKey;
115
- });
116
- }
117
- retrieveUsers() {
118
- return __awaiter(this, void 0, void 0, function* () {
119
- return this.db.from("users").select();
120
- });
121
- }
122
- getKeyBundle(deviceID) {
123
- return __awaiter(this, void 0, void 0, function* () {
124
- const device = yield this.retrieveDevice(deviceID);
125
- if (!device) {
126
- throw new Error("DeviceID not found.");
127
- }
128
- const otk = (yield this.getOTK(deviceID)) || undefined;
129
- const preKey = yield this.getPreKeys(deviceID);
130
- if (!preKey) {
131
- throw new Error("Failed to get prekey.");
132
- }
133
- const keyBundle = {
134
- signKey: crypto_1.XUtils.decodeHex(device.signKey),
135
- preKey,
136
- otk,
137
- };
138
- return keyBundle;
139
- });
140
- }
141
- createDevice(owner, payload) {
142
- return __awaiter(this, void 0, void 0, function* () {
143
- const device = {
144
- owner,
145
- signKey: payload.signKey,
146
- deviceID: uuid.v4(),
147
- name: payload.deviceName,
148
- lastLogin: new Date(Date.now()).toString(),
149
- deleted: false,
150
- };
151
- yield this.db("devices").insert(device);
152
- const medPreKeys = {
153
- keyID: uuid.v4(),
154
- userID: owner,
155
- deviceID: device.deviceID,
156
- publicKey: payload.preKey,
157
- signature: payload.preKeySignature,
158
- index: payload.preKeyIndex,
107
+ const sqliteDb = new BetterSqlite3(filename);
108
+ sqliteDb.pragma("journal_mode = WAL");
109
+ sqliteDb.pragma("synchronous = NORMAL");
110
+ sqliteDb.pragma("busy_timeout = 5000");
111
+ sqliteDb.pragma("cache_size = -64000");
112
+ sqliteDb.pragma("temp_store = memory");
113
+ sqliteDb.pragma("foreign_keys = ON");
114
+ this.db = new Kysely({
115
+ dialect: new SqliteDialect({ database: sqliteDb }),
116
+ });
117
+ void this.init();
118
+ }
119
+ async close() {
120
+ this.log.info("Closing database.");
121
+ await this.db.destroy();
122
+ }
123
+ async createChannel(name, serverID) {
124
+ const channel = {
125
+ channelID: crypto.randomUUID(),
126
+ name,
127
+ serverID,
128
+ };
129
+ await this.db.insertInto("channels").values(channel).execute();
130
+ return channel;
131
+ }
132
+ async createDevice(owner, payload) {
133
+ const device = {
134
+ deleted: 0,
135
+ deviceID: crypto.randomUUID(),
136
+ lastLogin: new Date().toISOString(),
137
+ name: payload.deviceName,
138
+ owner,
139
+ signKey: payload.signKey,
140
+ };
141
+ await this.db.insertInto("devices").values(device).execute();
142
+ const medPreKeys = {
143
+ deviceID: device.deviceID,
144
+ index: payload.preKeyIndex,
145
+ keyID: crypto.randomUUID(),
146
+ publicKey: payload.preKey,
147
+ signature: payload.preKeySignature,
148
+ userID: owner,
149
+ };
150
+ await this.db.insertInto("preKeys").values(medPreKeys).execute();
151
+ return toDevice(device);
152
+ }
153
+ async createEmoji(emoji) {
154
+ await this.db.insertInto("emojis").values(emoji).execute();
155
+ }
156
+ async createFile(file) {
157
+ await this.db.insertInto("files").values(file).execute();
158
+ }
159
+ async createInvite(inviteID, serverID, ownerID, expiration) {
160
+ const invite = {
161
+ expiration,
162
+ inviteID,
163
+ owner: ownerID,
164
+ serverID,
165
+ };
166
+ await this.db.insertInto("invites").values(invite).execute();
167
+ return invite;
168
+ }
169
+ async createPermission(userID, resourceType, resourceID, powerLevel) {
170
+ const permissionID = crypto.randomUUID();
171
+ // check if it already exists
172
+ const checkPermission = await this.db
173
+ .selectFrom("permissions")
174
+ .selectAll()
175
+ .where("userID", "=", userID)
176
+ .where("resourceID", "=", resourceID)
177
+ .execute();
178
+ const existing = checkPermission[0];
179
+ if (existing) {
180
+ return existing;
181
+ }
182
+ const permission = {
183
+ permissionID,
184
+ powerLevel,
185
+ resourceID,
186
+ resourceType,
187
+ userID,
188
+ };
189
+ await this.db.insertInto("permissions").values(permission).execute();
190
+ return permission;
191
+ }
192
+ async createServer(name, ownerID) {
193
+ // create the server
194
+ const server = {
195
+ name,
196
+ serverID: crypto.randomUUID(),
197
+ };
198
+ await this.db
199
+ .insertInto("servers")
200
+ .values({
201
+ icon: server.icon ?? null,
202
+ name: server.name,
203
+ serverID: server.serverID,
204
+ })
205
+ .execute();
206
+ // create the admin permission
207
+ await this.createPermission(ownerID, "server", server.serverID, 100);
208
+ // create the general channel
209
+ await this.createChannel("general", server.serverID);
210
+ return server;
211
+ }
212
+ async createUser(regKey, regPayload) {
213
+ try {
214
+ const salt = xMakeNonce();
215
+ const passwordHash = hashPassword(regPayload.password, salt);
216
+ const user = {
217
+ lastSeen: new Date().toISOString(),
218
+ passwordHash: passwordHash.toString("hex"),
219
+ passwordSalt: XUtils.encodeHex(salt),
220
+ userID: uuidStringify(regKey),
221
+ username: regPayload.username,
159
222
  };
160
- yield this.db("preKeys").insert(medPreKeys);
161
- return device;
162
- });
163
- }
164
- deleteDevice(deviceID) {
165
- return __awaiter(this, void 0, void 0, function* () {
166
- yield this.db
167
- .from("preKeys")
168
- .where({ deviceID })
169
- .del();
170
- yield this.db
171
- .from("oneTimeKeys")
172
- .where({ deviceID })
173
- .del();
174
- return this.db
175
- .from("devices")
176
- .where({ deviceID })
177
- .update({ deleted: true });
178
- });
223
+ await this.db
224
+ .insertInto("users")
225
+ .values({
226
+ ...user,
227
+ lastSeen: user.lastSeen,
228
+ })
229
+ .execute();
230
+ await this.createDevice(user.userID, regPayload);
231
+ return [user, null];
232
+ }
233
+ catch (err) {
234
+ return [null, err instanceof Error ? err : new Error(String(err))];
235
+ }
179
236
  }
180
- retrieveDevice(deviceID) {
181
- return __awaiter(this, void 0, void 0, function* () {
182
- if (uuid.validate(deviceID)) {
183
- const rows = yield this.db
184
- .from("devices")
185
- .select()
186
- .where({ deviceID, deleted: false });
187
- if (rows.length === 0) {
188
- return null;
189
- }
190
- const [device] = rows;
191
- return device;
192
- }
193
- if (pubkeyRegex.test(deviceID)) {
194
- const rows = yield this.db
195
- .from("devices")
196
- .select()
197
- .where({ signKey: deviceID, deleted: false });
198
- if (rows.length === 0) {
199
- return null;
200
- }
201
- const [device] = rows;
202
- return device;
203
- }
237
+ async deleteChannel(channelID) {
238
+ await this.deletePermissions(channelID);
239
+ await this.db
240
+ .deleteFrom("mail")
241
+ .where("group", "=", channelID)
242
+ .execute();
243
+ await this.db
244
+ .deleteFrom("channels")
245
+ .where("channelID", "=", channelID)
246
+ .execute();
247
+ }
248
+ async deleteDevice(deviceID) {
249
+ await this.db
250
+ .deleteFrom("preKeys")
251
+ .where("deviceID", "=", deviceID)
252
+ .execute();
253
+ await this.db
254
+ .deleteFrom("oneTimeKeys")
255
+ .where("deviceID", "=", deviceID)
256
+ .execute();
257
+ await this.db
258
+ .updateTable("devices")
259
+ .set({ deleted: 1 })
260
+ .where("deviceID", "=", deviceID)
261
+ .execute();
262
+ }
263
+ async deleteEmoji(emojiID) {
264
+ await this.db
265
+ .deleteFrom("emojis")
266
+ .where("emojiID", "=", emojiID)
267
+ .execute();
268
+ }
269
+ async deleteInvite(inviteID) {
270
+ await this.db
271
+ .deleteFrom("invites")
272
+ .where("inviteID", "=", inviteID)
273
+ .execute();
274
+ }
275
+ async deleteMail(nonce, userID) {
276
+ await this.db
277
+ .deleteFrom("mail")
278
+ .where("nonce", "=", XUtils.encodeHex(nonce))
279
+ .where("recipient", "=", userID)
280
+ .execute();
281
+ }
282
+ async deletePermission(permissionID) {
283
+ await this.db
284
+ .deleteFrom("permissions")
285
+ .where("permissionID", "=", permissionID)
286
+ .execute();
287
+ }
288
+ async deletePermissions(resourceID) {
289
+ await this.db
290
+ .deleteFrom("permissions")
291
+ .where("resourceID", "=", resourceID)
292
+ .execute();
293
+ }
294
+ async deleteServer(serverID) {
295
+ await this.deletePermissions(serverID);
296
+ const channels = await this.retrieveChannels(serverID);
297
+ for (const channel of channels) {
298
+ await this.deleteChannel(channel.channelID);
299
+ }
300
+ await this.db
301
+ .deleteFrom("servers")
302
+ .where("serverID", "=", serverID)
303
+ .execute();
304
+ }
305
+ async getKeyBundle(deviceID) {
306
+ const device = await this.retrieveDevice(deviceID);
307
+ if (!device) {
308
+ throw new Error("DeviceID not found.");
309
+ }
310
+ const otk = (await this.getOTK(deviceID)) || undefined;
311
+ const preKey = await this.getPreKeys(deviceID);
312
+ if (!preKey) {
313
+ throw new Error("Failed to get prekey.");
314
+ }
315
+ const keyBundle = {
316
+ otk,
317
+ preKey,
318
+ signKey: XUtils.decodeHex(device.signKey),
319
+ };
320
+ return keyBundle;
321
+ }
322
+ async getOTK(deviceID) {
323
+ const rows = await this.db
324
+ .selectFrom("oneTimeKeys")
325
+ .selectAll()
326
+ .where("deviceID", "=", deviceID)
327
+ .orderBy("index")
328
+ .limit(1)
329
+ .execute();
330
+ const otkInfo = rows[0];
331
+ if (!otkInfo) {
204
332
  return null;
205
- });
206
- }
207
- retrieveUserDeviceList(userIDs) {
208
- return __awaiter(this, void 0, void 0, function* () {
209
- return this.db
210
- .from("devices")
211
- .select()
212
- .whereIn("owner", userIDs)
213
- .andWhere({ deleted: false });
214
- });
215
- }
216
- getOTK(deviceID) {
217
- return __awaiter(this, void 0, void 0, function* () {
218
- const rows = yield this.db("oneTimeKeys")
219
- .select()
220
- .where({ deviceID })
221
- .limit(1)
222
- .orderBy("index");
223
- if (rows.length === 0) {
224
- return null;
225
- }
226
- const [otkInfo] = rows;
227
- const otk = {
228
- publicKey: crypto_1.XUtils.decodeHex(otkInfo.publicKey),
229
- signature: crypto_1.XUtils.decodeHex(otkInfo.signature),
230
- index: otkInfo.index,
231
- deviceID: otkInfo.deviceID,
232
- };
233
- try {
234
- // delete the otk
235
- yield this.db
236
- .from("oneTimeKeys")
237
- .delete()
238
- .where({ deviceID, index: otk.index });
239
- return otk;
240
- }
241
- catch (err) {
242
- throw err;
243
- }
244
- });
245
- }
246
- getOTKCount(deviceID) {
247
- return __awaiter(this, void 0, void 0, function* () {
248
- const keys = yield this.db
249
- .from("oneTimeKeys")
250
- .select()
251
- .where({ deviceID });
252
- return keys.length;
253
- });
254
- }
255
- createPermission(userID, resourceType, resourceID, powerLevel) {
256
- return __awaiter(this, void 0, void 0, function* () {
257
- const permissionID = uuid.v4();
258
- // check if it already exists
259
- const checkPermission = yield this.db
260
- .from("permissions")
261
- .select()
262
- .where({ userID, resourceID });
263
- if (checkPermission.length > 0) {
264
- return checkPermission[0];
265
- }
266
- const permission = {
267
- permissionID,
268
- userID,
269
- resourceType,
270
- resourceID,
271
- powerLevel,
272
- };
273
- yield this.db("permissions").insert(permission);
274
- return permission;
275
- });
276
- }
277
- retrieveInvite(inviteID) {
278
- return __awaiter(this, void 0, void 0, function* () {
279
- const rows = yield this.db
280
- .from("invites")
281
- .select()
282
- .where({ inviteID });
283
- if (rows.length === 0) {
284
- return null;
285
- }
286
- return rows[0];
287
- });
288
- }
289
- retrieveServerInvites(serverID) {
290
- return __awaiter(this, void 0, void 0, function* () {
291
- const rows = yield this.db
292
- .from("invites")
293
- .select()
294
- .where({ serverID });
295
- return rows.filter((invite) => {
296
- const valid = new Date(Date.now()).getTime() <
297
- new Date(invite.expiration).getTime();
298
- if (!valid) {
299
- this.deleteInvite(invite.inviteID);
300
- }
301
- return valid;
302
- });
303
- });
304
- }
305
- deleteInvite(inviteID) {
306
- return __awaiter(this, void 0, void 0, function* () {
307
- yield this.db
308
- .from("invites")
309
- .where({ inviteID })
310
- .delete();
311
- });
312
- }
313
- createInvite(inviteID, serverID, ownerID, expiration) {
314
- return __awaiter(this, void 0, void 0, function* () {
315
- const invite = {
316
- inviteID,
317
- serverID,
318
- owner: ownerID,
319
- expiration,
320
- };
321
- yield this.db("invites").insert(invite);
322
- return invite;
323
- });
324
- }
325
- retrieveGroupMembers(channelID) {
326
- return __awaiter(this, void 0, void 0, function* () {
327
- const channel = yield this.retrieveChannel(channelID);
328
- if (!channel) {
329
- return [];
330
- }
331
- const permissions = yield this.db
332
- .from("permissions")
333
- .select()
334
- .where({ resourceID: channel.serverID });
335
- const groupMembers = [];
336
- for (const permission of permissions) {
337
- const user = yield this.retrieveUser(permission.userID);
338
- if (user) {
339
- groupMembers.push(user);
340
- }
341
- }
342
- return groupMembers;
343
- });
344
- }
345
- retrieveChannel(channelID) {
346
- return __awaiter(this, void 0, void 0, function* () {
347
- const channels = yield this.db
348
- .from("channels")
349
- .select()
350
- .where({ channelID })
351
- .limit(1);
352
- if (channels.length === 0) {
353
- return null;
354
- }
355
- return channels[0];
356
- });
333
+ }
334
+ const otk = {
335
+ deviceID: otkInfo.deviceID,
336
+ index: otkInfo.index,
337
+ publicKey: XUtils.decodeHex(otkInfo.publicKey),
338
+ signature: XUtils.decodeHex(otkInfo.signature),
339
+ };
340
+ // delete the otk
341
+ await this.db
342
+ .deleteFrom("oneTimeKeys")
343
+ .where("deviceID", "=", deviceID)
344
+ .where("index", "=", otk.index)
345
+ .execute();
346
+ return otk;
347
+ }
348
+ async getOTKCount(deviceID) {
349
+ const result = await this.db
350
+ .selectFrom("oneTimeKeys")
351
+ .select((eb) => eb.fn.countAll().as("count"))
352
+ .where("deviceID", "=", deviceID)
353
+ .executeTakeFirst();
354
+ return Number(result?.count ?? 0);
355
+ }
356
+ async getPreKeys(deviceID) {
357
+ const rows = await this.db
358
+ .selectFrom("preKeys")
359
+ .selectAll()
360
+ .where("deviceID", "=", deviceID)
361
+ .execute();
362
+ const preKeyInfo = rows[0];
363
+ if (!preKeyInfo) {
364
+ return null;
365
+ }
366
+ const preKey = {
367
+ deviceID: preKeyInfo.deviceID,
368
+ index: preKeyInfo.index,
369
+ publicKey: XUtils.decodeHex(preKeyInfo.publicKey),
370
+ signature: XUtils.decodeHex(preKeyInfo.signature),
371
+ };
372
+ return preKey;
373
+ }
374
+ async getRequestsTotal() {
375
+ const row = await this.db
376
+ .selectFrom("service_metrics")
377
+ .select("metric_value")
378
+ .where("metric_key", "=", "requests_total")
379
+ .executeTakeFirst();
380
+ const raw = row?.metric_value;
381
+ const count = Number(raw);
382
+ if (!Number.isFinite(count) || count < 0) {
383
+ return 0;
384
+ }
385
+ return count;
357
386
  }
358
- retrieveChannels(serverID) {
359
- return __awaiter(this, void 0, void 0, function* () {
360
- const channels = yield this.db
361
- .from("channels")
362
- .select()
363
- .where({ serverID });
364
- return channels;
365
- });
387
+ async incrementRequestsTotal(by = 1) {
388
+ if (!Number.isFinite(by) || by <= 0) {
389
+ return;
390
+ }
391
+ await this.db
392
+ .updateTable("service_metrics")
393
+ .set({
394
+ metric_value: sql `metric_value + ${Math.floor(by)}`,
395
+ })
396
+ .where("metric_key", "=", "requests_total")
397
+ .execute();
398
+ }
399
+ async isHealthy() {
400
+ try {
401
+ await sql `select 1 as ok`.execute(this.db);
402
+ return true;
403
+ }
404
+ catch (err) {
405
+ this.log.warn("Database health check failed: " + String(err));
406
+ return false;
407
+ }
366
408
  }
367
- createChannel(name, serverID) {
368
- return __awaiter(this, void 0, void 0, function* () {
369
- const channel = {
370
- channelID: uuid.v4(),
371
- serverID,
372
- name,
373
- };
374
- yield this.db("channels").insert(channel);
375
- return channel;
376
- });
409
+ async markDeviceLogin(device) {
410
+ await this.db
411
+ .updateTable("devices")
412
+ .set({ lastLogin: new Date().toISOString() })
413
+ .where("deviceID", "=", device.deviceID)
414
+ .execute();
377
415
  }
378
- createServer(name, ownerID) {
379
- return __awaiter(this, void 0, void 0, function* () {
380
- // create the server
381
- const server = {
382
- name,
383
- serverID: uuid.v4(),
384
- };
385
- yield this.db("servers").insert(server);
386
- // create the admin permission
387
- yield this.createPermission(ownerID, "server", server.serverID, 100);
388
- // create the general channel
389
- yield this.createChannel("general", server.serverID);
390
- return server;
391
- });
416
+ async markUserSeen(user) {
417
+ await this.db
418
+ .updateTable("users")
419
+ .set({ lastSeen: new Date().toISOString() })
420
+ .where("userID", "=", user.userID)
421
+ .execute();
392
422
  }
393
423
  /**
394
424
  * Retrives a list of users that should be notified when a specific resourceID
@@ -396,414 +426,309 @@ class Database extends events_1.EventEmitter {
396
426
  *
397
427
  * @param resourceID
398
428
  */
399
- retrieveAffectedUsers(resourceID) {
400
- return __awaiter(this, void 0, void 0, function* () {
401
- const permissionList = yield this.retrievePermissionsByResourceID(resourceID);
402
- const users = [];
403
- for (const permission of permissionList) {
404
- const user = yield this.retrieveUser(permission.userID);
405
- if (user) {
406
- users.push(user);
407
- }
429
+ async retrieveAffectedUsers(resourceID) {
430
+ const permissionList = await this.retrievePermissionsByResourceID(resourceID);
431
+ const users = [];
432
+ for (const permission of permissionList) {
433
+ const user = await this.retrieveUser(permission.userID);
434
+ if (user) {
435
+ users.push(user);
408
436
  }
409
- return users;
410
- });
411
- }
412
- retrievePermissionsByResourceID(resourceID) {
413
- return __awaiter(this, void 0, void 0, function* () {
414
- return this.db
415
- .from("permissions")
416
- .select()
417
- .where({ resourceID });
418
- });
419
- }
420
- retrievePermissions(userID, resourceType) {
421
- return __awaiter(this, void 0, void 0, function* () {
422
- if (resourceType === "all") {
423
- const sList = yield this.db
424
- .from("permissions")
425
- .select()
426
- .where({ userID });
427
- return sList;
428
- }
429
- const serverList = yield this.db
430
- .from("permissions")
431
- .select()
432
- .where({ userID, resourceType });
433
- return serverList;
434
- });
435
- }
436
- retrieveServer(serverID) {
437
- return __awaiter(this, void 0, void 0, function* () {
438
- const rows = yield this.db
439
- .from("servers")
440
- .select()
441
- .where({ serverID })
442
- .limit(1);
443
- if (rows.length === 0) {
444
- return null;
445
- }
446
- const server = rows[0];
447
- return server;
448
- });
449
- }
450
- deletePermissions(resourceID) {
451
- return __awaiter(this, void 0, void 0, function* () {
452
- yield this.db
453
- .from("permissions")
454
- .where({ resourceID })
455
- .delete();
456
- });
457
- }
458
- deletePermission(permissionID) {
459
- return __awaiter(this, void 0, void 0, function* () {
460
- yield this.db
461
- .from("permissions")
462
- .where({ permissionID })
463
- .delete();
464
- });
465
- }
466
- retrievePermission(permissionID) {
467
- return __awaiter(this, void 0, void 0, function* () {
468
- const rows = yield this.db
469
- .from("permissions")
470
- .where({ permissionID })
471
- .select();
472
- if (rows.length === 0) {
473
- return null;
474
- }
475
- return rows[0];
476
- });
477
- }
478
- deleteChannel(channelID) {
479
- return __awaiter(this, void 0, void 0, function* () {
480
- yield this.deletePermissions(channelID);
481
- yield this.db
482
- .from("mail")
483
- .where({ group: channelID })
484
- .delete();
485
- yield this.db
486
- .from("channels")
487
- .where({ channelID })
488
- .delete();
489
- });
490
- }
491
- createEmoji(emoji) {
492
- return __awaiter(this, void 0, void 0, function* () {
493
- yield this.db("emojis").insert(emoji);
494
- });
495
- }
496
- deleteEmoji(emojiID) {
497
- return __awaiter(this, void 0, void 0, function* () {
498
- yield this.db
499
- .from("emojis")
500
- .where({ emojiID })
501
- .del();
502
- });
503
- }
504
- retrieveEmojiList(userID) {
505
- return __awaiter(this, void 0, void 0, function* () {
506
- return this.db
507
- .from("emojis")
508
- .select()
509
- .where({ owner: userID });
510
- });
511
- }
512
- retrieveEmoji(emojiID) {
513
- return __awaiter(this, void 0, void 0, function* () {
514
- const rows = yield this.db
515
- .from("emojis")
516
- .select()
517
- .where({ emojiID });
518
- if (rows.length === 0) {
519
- return null;
520
- }
521
- return rows[0];
522
- });
523
- }
524
- deleteServer(serverID) {
525
- return __awaiter(this, void 0, void 0, function* () {
526
- yield this.deletePermissions(serverID);
527
- const channels = yield this.retrieveChannels(serverID);
528
- for (const channel of channels) {
529
- yield this.deleteChannel(channel.channelID);
530
- }
531
- yield this.db
532
- .from("servers")
533
- .where({ serverID })
534
- .delete();
535
- });
536
- }
537
- retrieveServers(userID) {
538
- return __awaiter(this, void 0, void 0, function* () {
539
- const serverPerms = yield this.retrievePermissions(userID, "server");
540
- if (!serverPerms) {
541
- return [];
542
- }
543
- const serverList = [];
544
- for (const perm of serverPerms) {
545
- const server = yield this.retrieveServer(perm.resourceID);
546
- if (server) {
547
- serverList.push(server);
548
- }
549
- }
550
- return serverList;
551
- });
552
- }
553
- createUser(regKey, regPayload) {
554
- return __awaiter(this, void 0, void 0, function* () {
555
- try {
556
- const salt = crypto_1.xMakeNonce();
557
- const passwordHash = exports.hashPassword(regPayload.password, salt);
558
- const user = {
559
- userID: uuid.stringify(regKey),
560
- username: regPayload.username,
561
- lastSeen: new Date(Date.now()),
562
- passwordHash: passwordHash.toString("hex"),
563
- passwordSalt: crypto_1.XUtils.encodeHex(salt),
564
- };
565
- yield this.db("users").insert(user);
566
- yield this.createDevice(user.userID, regPayload);
567
- return [user, null];
568
- }
569
- catch (err) {
570
- return [null, err];
571
- }
572
- });
573
- }
574
- createFile(file) {
575
- return __awaiter(this, void 0, void 0, function* () {
576
- return this.db("files").insert(file);
577
- });
578
- }
579
- retrieveFile(fileID) {
580
- return __awaiter(this, void 0, void 0, function* () {
581
- const file = yield this.db
582
- .from("files")
583
- .select()
584
- .where({ fileID });
585
- if (file.length === 0) {
586
- return null;
587
- }
588
- return file[0];
589
- });
590
- }
591
- // the identifier can be username, public key, or userID
592
- retrieveUser(userIdentifier) {
593
- return __awaiter(this, void 0, void 0, function* () {
594
- let rows = [];
595
- if (uuid.validate(userIdentifier)) {
596
- rows = yield this.db
597
- .from("users")
598
- .select()
599
- .where({ userID: userIdentifier })
600
- .limit(1);
601
- }
602
- else {
603
- rows = yield this.db
604
- .from("users")
605
- .select()
606
- .where({ username: userIdentifier })
607
- .limit(1);
608
- }
609
- if (rows.length === 0) {
610
- return null;
437
+ }
438
+ return users;
439
+ }
440
+ async retrieveChannel(channelID) {
441
+ const channels = await this.db
442
+ .selectFrom("channels")
443
+ .selectAll()
444
+ .where("channelID", "=", channelID)
445
+ .limit(1)
446
+ .execute();
447
+ return channels[0] ?? null;
448
+ }
449
+ async retrieveChannels(serverID) {
450
+ const channels = await this.db
451
+ .selectFrom("channels")
452
+ .selectAll()
453
+ .where("serverID", "=", serverID)
454
+ .execute();
455
+ return channels;
456
+ }
457
+ async retrieveDevice(deviceID) {
458
+ if (uuidValidate(deviceID)) {
459
+ const rows = await this.db
460
+ .selectFrom("devices")
461
+ .selectAll()
462
+ .where("deviceID", "=", deviceID)
463
+ .where("deleted", "=", 0)
464
+ .execute();
465
+ const device = rows[0];
466
+ return device ? toDevice(device) : null;
467
+ }
468
+ if (pubkeyRegex.test(deviceID)) {
469
+ const rows = await this.db
470
+ .selectFrom("devices")
471
+ .selectAll()
472
+ .where("signKey", "=", deviceID)
473
+ .where("deleted", "=", 0)
474
+ .execute();
475
+ const device = rows[0];
476
+ return device ? toDevice(device) : null;
477
+ }
478
+ return null;
479
+ }
480
+ async retrieveEmoji(emojiID) {
481
+ const rows = await this.db
482
+ .selectFrom("emojis")
483
+ .selectAll()
484
+ .where("emojiID", "=", emojiID)
485
+ .execute();
486
+ return rows[0] ?? null;
487
+ }
488
+ async retrieveEmojiList(userID) {
489
+ return this.db
490
+ .selectFrom("emojis")
491
+ .selectAll()
492
+ .where("owner", "=", userID)
493
+ .execute();
494
+ }
495
+ async retrieveFile(fileID) {
496
+ const file = await this.db
497
+ .selectFrom("files")
498
+ .selectAll()
499
+ .where("fileID", "=", fileID)
500
+ .execute();
501
+ return file[0] ?? null;
502
+ }
503
+ async retrieveGroupMembers(channelID) {
504
+ const channel = await this.retrieveChannel(channelID);
505
+ if (!channel) {
506
+ return [];
507
+ }
508
+ const permissions = await this.db
509
+ .selectFrom("permissions")
510
+ .selectAll()
511
+ .where("resourceID", "=", channel.serverID)
512
+ .execute();
513
+ const groupMembers = [];
514
+ for (const permission of permissions) {
515
+ const user = await this.retrieveUser(permission.userID);
516
+ if (user) {
517
+ groupMembers.push(user);
611
518
  }
612
- const [user] = rows;
613
- return user;
614
- });
615
- }
616
- saveMail(mail, header, deviceID, userID) {
617
- return __awaiter(this, void 0, void 0, function* () {
618
- const entry = {
519
+ }
520
+ return groupMembers;
521
+ }
522
+ async retrieveInvite(inviteID) {
523
+ const rows = await this.db
524
+ .selectFrom("invites")
525
+ .selectAll()
526
+ .where("inviteID", "=", inviteID)
527
+ .execute();
528
+ return rows[0] ?? null;
529
+ }
530
+ async retrieveMail(deviceID) {
531
+ const rawRows = await this.db
532
+ .selectFrom("mail")
533
+ .selectAll()
534
+ .where("recipient", "=", deviceID)
535
+ .execute();
536
+ const rows = rawRows.map(toMailSQL);
537
+ const fixMail = (mail) => {
538
+ const msgb = {
539
+ authorID: mail.authorID,
540
+ cipher: XUtils.decodeHex(mail.cipher),
541
+ extra: XUtils.decodeHex(mail.extra),
542
+ forward: mail.forward,
543
+ group: mail.group ? XUtils.decodeHex(mail.group) : null,
619
544
  mailID: mail.mailID,
620
545
  mailType: mail.mailType,
621
- recipient: mail.recipient,
622
- sender: deviceID,
623
- cipher: crypto_1.XUtils.encodeHex(mail.cipher),
624
- nonce: crypto_1.XUtils.encodeHex(mail.nonce),
625
- extra: crypto_1.XUtils.encodeHex(mail.extra),
626
- header: crypto_1.XUtils.encodeHex(header),
627
- time: new Date(Date.now()),
628
- group: mail.group ? crypto_1.XUtils.encodeHex(mail.group) : null,
629
- forward: mail.forward,
630
- authorID: userID,
546
+ nonce: XUtils.decodeHex(mail.nonce),
631
547
  readerID: mail.readerID,
548
+ recipient: mail.recipient,
549
+ sender: mail.sender,
632
550
  };
633
- yield this.db("mail").insert(entry);
634
- });
551
+ const msgh = XUtils.decodeHex(mail.header);
552
+ return [msgh, msgb, mail.time];
553
+ };
554
+ const allMail = rows.map(fixMail);
555
+ return allMail;
556
+ }
557
+ async retrievePermission(permissionID) {
558
+ const rows = await this.db
559
+ .selectFrom("permissions")
560
+ .selectAll()
561
+ .where("permissionID", "=", permissionID)
562
+ .execute();
563
+ return rows[0] ?? null;
564
+ }
565
+ async retrievePermissions(userID, resourceType) {
566
+ if (resourceType === "all") {
567
+ const sList = await this.db
568
+ .selectFrom("permissions")
569
+ .selectAll()
570
+ .where("userID", "=", userID)
571
+ .execute();
572
+ return sList;
573
+ }
574
+ const serverList = await this.db
575
+ .selectFrom("permissions")
576
+ .selectAll()
577
+ .where("userID", "=", userID)
578
+ .where("resourceType", "=", resourceType)
579
+ .execute();
580
+ return serverList;
581
+ }
582
+ async retrievePermissionsByResourceID(resourceID) {
583
+ return this.db
584
+ .selectFrom("permissions")
585
+ .selectAll()
586
+ .where("resourceID", "=", resourceID)
587
+ .execute();
588
+ }
589
+ async retrieveServer(serverID) {
590
+ const rows = await this.db
591
+ .selectFrom("servers")
592
+ .selectAll()
593
+ .where("serverID", "=", serverID)
594
+ .limit(1)
595
+ .execute();
596
+ const row = rows[0];
597
+ return row ? toServer(row) : null;
598
+ }
599
+ async retrieveServerInvites(serverID) {
600
+ const rows = await this.db
601
+ .selectFrom("invites")
602
+ .selectAll()
603
+ .where("serverID", "=", serverID)
604
+ .execute();
605
+ return rows.filter((invite) => {
606
+ const valid = new Date(Date.now()).getTime() <
607
+ new Date(invite.expiration).getTime();
608
+ if (!valid) {
609
+ void this.deleteInvite(invite.inviteID);
610
+ }
611
+ return valid;
612
+ });
613
+ }
614
+ async retrieveServers(userID) {
615
+ const serverPerms = await this.retrievePermissions(userID, "server");
616
+ const serverList = [];
617
+ for (const perm of serverPerms) {
618
+ const server = await this.retrieveServer(perm.resourceID);
619
+ if (server) {
620
+ serverList.push(server);
621
+ }
622
+ }
623
+ return serverList;
635
624
  }
636
- retrieveMail(deviceID
637
- // tslint:disable-next-line: array-type
638
- ) {
639
- return __awaiter(this, void 0, void 0, function* () {
640
- const rows = yield this.db
641
- .from("mail")
642
- .select()
643
- .where({ recipient: deviceID });
644
- const fixMail = (mail) => {
645
- const msgb = {
646
- mailType: mail.mailType,
647
- mailID: mail.mailID,
648
- recipient: mail.recipient,
649
- cipher: crypto_1.XUtils.decodeHex(mail.cipher),
650
- nonce: crypto_1.XUtils.decodeHex(mail.nonce),
651
- extra: crypto_1.XUtils.decodeHex(mail.extra),
652
- sender: mail.sender,
653
- group: mail.group ? crypto_1.XUtils.decodeHex(mail.group) : null,
654
- forward: Boolean(mail.forward),
655
- authorID: mail.authorID,
656
- readerID: mail.readerID,
657
- };
658
- const msgh = crypto_1.XUtils.decodeHex(mail.header);
659
- return [msgh, msgb, new Date(mail.time)];
625
+ // the identifier can be username, public key, or userID
626
+ async retrieveUser(userIdentifier) {
627
+ let rows;
628
+ if (uuidValidate(userIdentifier)) {
629
+ rows = await this.db
630
+ .selectFrom("users")
631
+ .selectAll()
632
+ .where("userID", "=", userIdentifier)
633
+ .limit(1)
634
+ .execute();
635
+ }
636
+ else {
637
+ rows = await this.db
638
+ .selectFrom("users")
639
+ .selectAll()
640
+ .where("username", "=", userIdentifier)
641
+ .limit(1)
642
+ .execute();
643
+ }
644
+ const row = rows[0];
645
+ return row ? toUserRecord(row) : null;
646
+ }
647
+ async retrieveUserDeviceList(userIDs) {
648
+ const rows = await this.db
649
+ .selectFrom("devices")
650
+ .selectAll()
651
+ .where("owner", "in", userIDs)
652
+ .where("deleted", "=", 0)
653
+ .execute();
654
+ return rows.map(toDevice);
655
+ }
656
+ async retrieveUsers() {
657
+ const rows = await this.db.selectFrom("users").selectAll().execute();
658
+ return rows.map(toUserRecord);
659
+ }
660
+ async saveMail(mail, header, deviceID, userID) {
661
+ const entry = {
662
+ authorID: userID,
663
+ cipher: XUtils.encodeHex(mail.cipher),
664
+ extra: XUtils.encodeHex(mail.extra),
665
+ forward: mail.forward,
666
+ group: mail.group ? XUtils.encodeHex(mail.group) : null,
667
+ header: XUtils.encodeHex(header),
668
+ mailID: mail.mailID,
669
+ mailType: mail.mailType,
670
+ nonce: XUtils.encodeHex(mail.nonce),
671
+ readerID: mail.readerID,
672
+ recipient: mail.recipient,
673
+ sender: deviceID,
674
+ time: new Date().toISOString(),
675
+ };
676
+ await this.db
677
+ .insertInto("mail")
678
+ .values({
679
+ ...entry,
680
+ forward: entry.forward ? 1 : 0,
681
+ time: entry.time,
682
+ })
683
+ .execute();
684
+ }
685
+ async saveOTK(userID, deviceID, otks) {
686
+ for (const otk of otks) {
687
+ const newOTK = {
688
+ deviceID: otk.deviceID,
689
+ index: otk.index ?? 0,
690
+ keyID: crypto.randomUUID(),
691
+ publicKey: XUtils.encodeHex(otk.publicKey),
692
+ signature: XUtils.encodeHex(otk.signature),
693
+ userID,
660
694
  };
661
- const allMail = rows.map(fixMail);
662
- return allMail;
663
- });
664
- }
665
- deleteMail(nonce, userID) {
666
- return __awaiter(this, void 0, void 0, function* () {
667
- yield this.db
668
- .from("mail")
669
- .delete()
670
- .where({ nonce: crypto_1.XUtils.encodeHex(nonce), recipient: userID });
671
- });
672
- }
673
- markUserSeen(user) {
674
- return __awaiter(this, void 0, void 0, function* () {
675
- yield this.db("users")
676
- .where({ userID: user.userID })
677
- .update({
678
- lastSeen: new Date(Date.now()),
679
- });
680
- });
681
- }
682
- markDeviceLogin(device) {
683
- return __awaiter(this, void 0, void 0, function* () {
684
- yield this.db("devices")
685
- .where({ deviceID: device.deviceID })
686
- .update({
687
- lastLogin: new Date(Date.now()),
688
- });
689
- });
695
+ await this.db.insertInto("oneTimeKeys").values(newOTK).execute();
696
+ }
690
697
  }
691
- close() {
692
- return __awaiter(this, void 0, void 0, function* () {
693
- this.log.info("Closing database.");
694
- yield this.db.destroy();
695
- });
696
- }
697
- init() {
698
- return __awaiter(this, void 0, void 0, function* () {
699
- if (!(yield this.db.schema.hasTable("invites"))) {
700
- yield this.db.schema.createTable("invites", (table) => {
701
- table.string("inviteID").primary();
702
- table.string("serverID").index();
703
- table.string("owner");
704
- table.string("expiration");
705
- });
706
- }
707
- if (!(yield this.db.schema.hasTable("users"))) {
708
- yield this.db.schema.createTable("users", (table) => {
709
- table.string("userID").primary();
710
- table.string("username").unique();
711
- table.string("passwordHash");
712
- table.string("passwordSalt");
713
- table.dateTime("lastSeen");
714
- });
715
- }
716
- if (!(yield this.db.schema.hasTable("devices"))) {
717
- yield this.db.schema.createTable("devices", (table) => {
718
- table.string("deviceID").primary();
719
- table.string("signKey").unique();
720
- table.string("owner");
721
- table.string("name");
722
- table.string("lastLogin");
723
- table.boolean("deleted");
724
- });
725
- }
726
- if (!(yield this.db.schema.hasTable("mail"))) {
727
- yield this.db.schema.createTable("mail", (table) => {
728
- table.string("nonce").primary();
729
- table.string("recipient").index();
730
- table.string("mailID");
731
- table.string("sender");
732
- table.string("header");
733
- table.text("cipher", "mediumtext");
734
- table.string("group");
735
- table.text("extra");
736
- table.integer("mailType");
737
- table.dateTime("time");
738
- table.boolean("forward");
739
- table.string("authorID");
740
- table.string("readerID");
741
- });
742
- }
743
- if (!(yield this.db.schema.hasTable("preKeys"))) {
744
- yield this.db.schema.createTable("preKeys", (table) => {
745
- table.string("keyID").primary();
746
- table.string("userID").index();
747
- table
748
- .string("deviceID")
749
- .index()
750
- .unique();
751
- table.string("publicKey");
752
- table.string("signature");
753
- table.integer("index");
754
- });
755
- }
756
- if (!(yield this.db.schema.hasTable("oneTimeKeys"))) {
757
- yield this.db.schema.createTable("oneTimeKeys", (table) => {
758
- table.string("keyID").primary();
759
- table.string("userID").index();
760
- table.string("deviceID").index();
761
- table.string("publicKey");
762
- table.string("signature");
763
- table.integer("index");
764
- });
765
- }
766
- if (!(yield this.db.schema.hasTable("servers"))) {
767
- yield this.db.schema.createTable("servers", (table) => {
768
- table.string("serverID").primary();
769
- table.string("name");
770
- table.string("icon");
771
- });
772
- }
773
- if (!(yield this.db.schema.hasTable("channels"))) {
774
- yield this.db.schema.createTable("channels", (table) => {
775
- table.string("channelID").primary();
776
- table.string("serverID");
777
- table.string("name");
778
- });
779
- }
780
- if (!(yield this.db.schema.hasTable("permissions"))) {
781
- yield this.db.schema.createTable("permissions", (table) => {
782
- table.string("permissionID").primary();
783
- table.string("userID").index();
784
- table.string("resourceType");
785
- table.string("resourceID").index();
786
- table.integer("powerLevel");
787
- });
788
- }
789
- if (!(yield this.db.schema.hasTable("files"))) {
790
- yield this.db.schema.createTable("files", (table) => {
791
- table.string("fileID").primary();
792
- table.string("owner").index();
793
- table.string("nonce");
794
- });
795
- }
796
- if (!(yield this.db.schema.hasTable("emojis"))) {
797
- yield this.db.schema.createTable("emojis", (table) => {
798
- table.string("emojiID").primary();
799
- table.string("owner").index();
800
- table.string("name");
801
- });
802
- }
803
- this.emit("ready");
698
+ async init() {
699
+ const migrator = new Migrator({
700
+ db: this.db,
701
+ provider: new CrossPlatformMigrationProvider(migrationFolder),
804
702
  });
703
+ const { error } = await migrator.migrateToLatest();
704
+ if (error) {
705
+ this.emit("error", error);
706
+ return;
707
+ }
708
+ this.emit("ready");
805
709
  }
806
710
  }
807
- exports.Database = Database;
808
- const hashPassword = (password, salt) => pbkdf2_1.default.pbkdf2Sync(password, salt, exports.ITERATIONS, 32, "sha512");
809
- exports.hashPassword = hashPassword;
711
+ function toDevice(row) {
712
+ return { ...row, deleted: Boolean(row.deleted) };
713
+ }
714
+ function toMailSQL(row) {
715
+ return {
716
+ ...row,
717
+ extra: row.extra ?? "",
718
+ forward: Boolean(row.forward),
719
+ mailType: parseMailType(row.mailType),
720
+ time: row.time,
721
+ };
722
+ }
723
+ function toServer(row) {
724
+ return {
725
+ icon: row.icon ?? undefined,
726
+ name: row.name,
727
+ serverID: row.serverID,
728
+ };
729
+ }
730
+ function toUserRecord(row) {
731
+ return { ...row };
732
+ }
733
+ export const hashPassword = (password, salt) => pbkdf2Sync(password, salt, ITERATIONS, 32, "sha512");
734
+ //# sourceMappingURL=Database.js.map