@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
@@ -0,0 +1,167 @@
1
+ import type { SpireOptions } from "../Spire.ts";
2
+ import type { PreKeysWS } from "@vex-chat/types";
3
+
4
+ import { XUtils } from "@vex-chat/crypto";
5
+
6
+ import * as uuid from "uuid";
7
+ import { describe, expect, it, vi } from "vitest";
8
+ import winston from "winston";
9
+
10
+ import { Database } from "../Database.ts";
11
+
12
+ // vi.mock is hoisted above all imports automatically.
13
+ // Minimal stubs for uuid functions used by spire src: v4, parse, stringify.
14
+ vi.mock("uuid", () => ({
15
+ parse: (s: string) => {
16
+ const matches = s.replace(/-/g, "").match(/.{2}/g);
17
+ if (!matches) throw new Error("Invalid UUID");
18
+ return Uint8Array.from(matches.map((b) => parseInt(b, 16)));
19
+ },
20
+ stringify: (b: Uint8Array) => {
21
+ const hex = Array.from(b)
22
+ .map((x) => x.toString(16).padStart(2, "0"))
23
+ .join("");
24
+ return `${hex.slice(0, 8)}-${hex.slice(8, 12)}-${hex.slice(12, 16)}-${hex.slice(16, 20)}-${hex.slice(20, 32)}`;
25
+ },
26
+ v4: vi.fn(() => "93ce482b-a0f2-4f6e-b1df-3aed61073552"),
27
+ validate: () => true,
28
+ }));
29
+
30
+ /** Winston logger stub — Database.close() calls `.info`, and `{}` breaks that. */
31
+ function silentLogger(): winston.Logger {
32
+ const noop = vi.fn();
33
+ return {
34
+ debug: noop,
35
+ error: noop,
36
+ info: noop,
37
+ log: noop,
38
+ verbose: noop,
39
+ warn: noop,
40
+ } as unknown as winston.Logger;
41
+ }
42
+
43
+ describe("Database", () => {
44
+ // Reusable test data
45
+ const keyID = "de459e05-aa63-4dfa-97b4-ed43d5c7a5f7";
46
+ const userID = "4e67b90f-cbf8-44bc-8ce3-d3b248f033f1";
47
+ const deviceID = "23cb0b27-7d0c-43b2-87e1-c2b93e0095e5";
48
+
49
+ const publicKey = XUtils.decodeHex(
50
+ "30c2d0294c1cfdbb73c6b3bbe6010088c2dba8384b04ff2e2b92172431d66b5e",
51
+ );
52
+ const signature = XUtils.decodeHex(
53
+ "dd0665079426c3efcf4dce9b1487e4aca132f8147581b3294c3f23ddd2b4ba8240a10082bd06805d7eb320d91af971da3306e11b60073ccc3d829710f5036004000030c2d0294c1cfdbb73c6b3bbe6010088c2dba8384b04ff2e2b92172431d66b5e",
54
+ );
55
+
56
+ const testSQLPreKey = {
57
+ deviceID,
58
+ index: 1,
59
+ keyID,
60
+ publicKey:
61
+ "30c2d0294c1cfdbb73c6b3bbe6010088c2dba8384b04ff2e2b92172431d66b5e",
62
+ signature:
63
+ "dd0665079426c3efcf4dce9b1487e4aca132f8147581b3294c3f23ddd2b4ba8240a10082bd06805d7eb320d91af971da3306e11b60073ccc3d829710f5036004000030c2d0294c1cfdbb73c6b3bbe6010088c2dba8384b04ff2e2b92172431d66b5e",
64
+ userID,
65
+ };
66
+
67
+ const testWSPreKey: PreKeysWS = {
68
+ deviceID,
69
+ index: 1,
70
+ publicKey,
71
+ signature,
72
+ };
73
+
74
+ const options: SpireOptions = {
75
+ dbType: "sqlite3mem",
76
+ };
77
+
78
+ describe("saveOTK", () => {
79
+ it("takes a userId and one time key, adds a keyId and saves it to oneTimeKey table", async () => {
80
+ expect.assertions(1);
81
+
82
+ vi.mocked(uuid.v4).mockReturnValueOnce(keyID);
83
+ vi.spyOn(winston, "createLogger").mockReturnValueOnce(
84
+ silentLogger(),
85
+ );
86
+
87
+ const provider = new Database(options);
88
+ await new Promise<void>((resolve, reject) => {
89
+ provider.once("ready", () => {
90
+ void (async () => {
91
+ try {
92
+ await provider.saveOTK(
93
+ testSQLPreKey.userID,
94
+ testSQLPreKey.deviceID,
95
+ [
96
+ {
97
+ deviceID,
98
+ index: 1,
99
+ publicKey,
100
+ signature,
101
+ },
102
+ ],
103
+ );
104
+ const oneTimeKey = await provider.getOTK(deviceID);
105
+ expect(oneTimeKey).toEqual(testWSPreKey);
106
+ await provider.close();
107
+ resolve();
108
+ } catch (e: unknown) {
109
+ reject(
110
+ e instanceof Error ? e : new Error(String(e)),
111
+ );
112
+ }
113
+ })();
114
+ });
115
+ });
116
+ });
117
+ });
118
+
119
+ describe("getPreKeys", () => {
120
+ it("returns a preKey by deviceID if said preKey exists.", async () => {
121
+ expect.assertions(1);
122
+
123
+ const provider = new Database(options);
124
+ await new Promise<void>((resolve, reject) => {
125
+ provider.once("ready", () => {
126
+ void (async () => {
127
+ try {
128
+ await provider["db"]
129
+ .insertInto("preKeys")
130
+ .values(testSQLPreKey)
131
+ .execute();
132
+ const result = await provider.getPreKeys(deviceID);
133
+ expect(result).toEqual(testWSPreKey);
134
+ await provider.close();
135
+ resolve();
136
+ } catch (e: unknown) {
137
+ reject(
138
+ e instanceof Error ? e : new Error(String(e)),
139
+ );
140
+ }
141
+ })();
142
+ });
143
+ });
144
+ });
145
+
146
+ it("return null if there are no preKeys with deviceID param", async () => {
147
+ expect.assertions(1);
148
+ const provider = new Database(options);
149
+ await new Promise<void>((resolve, reject) => {
150
+ provider.once("ready", () => {
151
+ void (async () => {
152
+ try {
153
+ const result = await provider.getPreKeys(deviceID);
154
+ expect(result).toBeNull();
155
+ await provider.close();
156
+ resolve();
157
+ } catch (e: unknown) {
158
+ reject(
159
+ e instanceof Error ? e : new Error(String(e)),
160
+ );
161
+ }
162
+ })();
163
+ });
164
+ });
165
+ });
166
+ });
167
+ });
@@ -0,0 +1 @@
1
+ declare module "ed2curve";
@@ -0,0 +1,165 @@
1
+ import type { Insertable, Selectable, Updateable } from "kysely";
2
+
3
+ // ── Table interfaces ────────────────────────────────────────────────────
4
+
5
+ export type ChannelRow = Selectable<ChannelsTable>;
6
+
7
+ export interface ChannelsTable {
8
+ channelID: string;
9
+ name: string;
10
+ serverID: string;
11
+ }
12
+
13
+ export type ChannelUpdate = Updateable<ChannelsTable>;
14
+
15
+ export type DeviceRow = Selectable<DevicesTable>;
16
+
17
+ export interface DevicesTable {
18
+ deleted: number;
19
+ deviceID: string;
20
+ lastLogin: string;
21
+ name: string;
22
+ owner: string;
23
+ signKey: string;
24
+ }
25
+
26
+ export type DeviceUpdate = Updateable<DevicesTable>;
27
+
28
+ export type EmojiRow = Selectable<EmojisTable>;
29
+
30
+ export interface EmojisTable {
31
+ emojiID: string;
32
+ name: string;
33
+ owner: string;
34
+ }
35
+
36
+ export type EmojiUpdate = Updateable<EmojisTable>;
37
+
38
+ export type FileRow = Selectable<FilesTable>;
39
+
40
+ export interface FilesTable {
41
+ fileID: string;
42
+ nonce: string;
43
+ owner: string;
44
+ }
45
+
46
+ export type FileUpdate = Updateable<FilesTable>;
47
+
48
+ // ── Database schema ─────────────────────────────────────────────────────
49
+
50
+ export type InviteRow = Selectable<InvitesTable>;
51
+
52
+ // ── Row utility types ───────────────────────────────────────────────────
53
+
54
+ export interface InvitesTable {
55
+ expiration: string;
56
+ inviteID: string;
57
+ owner: string;
58
+ serverID: string;
59
+ }
60
+ export type InviteUpdate = Updateable<InvitesTable>;
61
+ export type MailRow = Selectable<MailTable>;
62
+
63
+ export interface MailTable {
64
+ authorID: string;
65
+ cipher: string;
66
+ extra: null | string;
67
+ forward: number;
68
+ group: null | string;
69
+ header: string;
70
+ mailID: string;
71
+ mailType: number;
72
+ nonce: string;
73
+ readerID: string;
74
+ recipient: string;
75
+ sender: string;
76
+ time: string;
77
+ }
78
+ export type MailUpdate = Updateable<MailTable>;
79
+ export type NewChannel = Insertable<ChannelsTable>;
80
+
81
+ export type NewDevice = Insertable<DevicesTable>;
82
+ export type NewEmoji = Insertable<EmojisTable>;
83
+ export type NewFile = Insertable<FilesTable>;
84
+
85
+ export type NewInvite = Insertable<InvitesTable>;
86
+ export type NewMail = Insertable<MailTable>;
87
+ export type NewOneTimeKey = Insertable<OneTimeKeysTable>;
88
+
89
+ export type NewPermission = Insertable<PermissionsTable>;
90
+ export type NewPreKey = Insertable<PreKeysTable>;
91
+ export type NewServer = Insertable<ServersTable>;
92
+
93
+ export type NewServiceMetric = Insertable<ServiceMetricsTable>;
94
+ export type NewUser = Insertable<UsersTable>;
95
+ export type OneTimeKeyRow = Selectable<OneTimeKeysTable>;
96
+
97
+ export interface OneTimeKeysTable {
98
+ deviceID: string;
99
+ index: number;
100
+ keyID: string;
101
+ publicKey: string;
102
+ signature: string;
103
+ userID: string;
104
+ }
105
+ export type OneTimeKeyUpdate = Updateable<OneTimeKeysTable>;
106
+ export type PermissionRow = Selectable<PermissionsTable>;
107
+
108
+ export interface PermissionsTable {
109
+ permissionID: string;
110
+ powerLevel: number;
111
+ resourceID: string;
112
+ resourceType: string;
113
+ userID: string;
114
+ }
115
+ export type PermissionUpdate = Updateable<PermissionsTable>;
116
+ export type PreKeyRow = Selectable<PreKeysTable>;
117
+
118
+ export interface PreKeysTable {
119
+ deviceID: string;
120
+ index: number;
121
+ keyID: string;
122
+ publicKey: string;
123
+ signature: string;
124
+ userID: string;
125
+ }
126
+ export type PreKeyUpdate = Updateable<PreKeysTable>;
127
+ export interface ServerDatabase {
128
+ channels: ChannelsTable;
129
+ devices: DevicesTable;
130
+ emojis: EmojisTable;
131
+ files: FilesTable;
132
+ invites: InvitesTable;
133
+ mail: MailTable;
134
+ oneTimeKeys: OneTimeKeysTable;
135
+ permissions: PermissionsTable;
136
+ preKeys: PreKeysTable;
137
+ servers: ServersTable;
138
+ service_metrics: ServiceMetricsTable;
139
+ users: UsersTable;
140
+ }
141
+
142
+ export type ServerRow = Selectable<ServersTable>;
143
+ export interface ServersTable {
144
+ icon: null | string;
145
+ name: string;
146
+ serverID: string;
147
+ }
148
+ export type ServerUpdate = Updateable<ServersTable>;
149
+
150
+ export type ServiceMetricRow = Selectable<ServiceMetricsTable>;
151
+ export interface ServiceMetricsTable {
152
+ metric_key: string;
153
+ metric_value: number;
154
+ }
155
+ export type ServiceMetricUpdate = Updateable<ServiceMetricsTable>;
156
+
157
+ export type UserRow = Selectable<UsersTable>;
158
+ export interface UsersTable {
159
+ lastSeen: string;
160
+ passwordHash: string;
161
+ passwordSalt: string;
162
+ userID: string;
163
+ username: string;
164
+ }
165
+ export type UserUpdate = Updateable<UsersTable>;
package/src/index.ts ADDED
@@ -0,0 +1,3 @@
1
+ import { Spire } from "./Spire.ts";
2
+
3
+ export { Spire };
@@ -0,0 +1,38 @@
1
+ import type { NextFunction, Request, Response } from "express";
2
+ import type { z } from "zod/v4";
3
+
4
+ /**
5
+ * Express middleware that validates req.body against a Zod schema.
6
+ * On failure, returns 400 with validation errors.
7
+ */
8
+ export function validateBody(schema: z.ZodType) {
9
+ return (req: Request, res: Response, next: NextFunction): void => {
10
+ const result = schema.safeParse(req.body);
11
+ if (!result.success) {
12
+ res.status(400).json({
13
+ error: "Validation failed",
14
+ issues: result.error.issues,
15
+ });
16
+ return;
17
+ }
18
+ req.body = result.data;
19
+ next();
20
+ };
21
+ }
22
+
23
+ /**
24
+ * Validates a single URL param as a non-empty string.
25
+ * For UUIDs, use validateUuidParam.
26
+ */
27
+ export function validateParam(name: string) {
28
+ return (req: Request, res: Response, next: NextFunction): void => {
29
+ const value = req.params[name];
30
+ if (!value || typeof value !== "string" || value.trim() === "") {
31
+ res.status(400).json({
32
+ error: `Missing or empty parameter: ${name}`,
33
+ });
34
+ return;
35
+ }
36
+ next();
37
+ };
38
+ }
@@ -0,0 +1,218 @@
1
+ import { type Kysely, sql } from "kysely";
2
+
3
+ export async function down(db: Kysely<unknown>): Promise<void> {
4
+ const tables = [
5
+ "service_metrics",
6
+ "invites",
7
+ "emojis",
8
+ "files",
9
+ "permissions",
10
+ "channels",
11
+ "servers",
12
+ "oneTimeKeys",
13
+ "preKeys",
14
+ "mail",
15
+ "devices",
16
+ "users",
17
+ ] as const;
18
+
19
+ for (const table of tables) {
20
+ await db.schema.dropTable(table).ifExists().execute();
21
+ }
22
+ }
23
+
24
+ export async function up(db: Kysely<unknown>): Promise<void> {
25
+ await db.schema
26
+ .createTable("users")
27
+ .ifNotExists()
28
+ .addColumn("userID", "varchar(255)", (cb) => cb.primaryKey())
29
+ .addColumn("username", "varchar(255)", (cb) => cb.unique())
30
+ .addColumn("passwordHash", "text")
31
+ .addColumn("passwordSalt", "text")
32
+ .addColumn("lastSeen", "text")
33
+ .execute();
34
+
35
+ await db.schema
36
+ .createTable("devices")
37
+ .ifNotExists()
38
+ .addColumn("deviceID", "varchar(255)", (cb) => cb.primaryKey())
39
+ .addColumn("signKey", "varchar(255)", (cb) => cb.unique())
40
+ .addColumn("owner", "varchar(255)")
41
+ .addColumn("name", "varchar(255)")
42
+ .addColumn("lastLogin", "text")
43
+ .addColumn("deleted", "integer", (cb) => cb.defaultTo(0))
44
+ .execute();
45
+
46
+ await db.schema
47
+ .createTable("mail")
48
+ .ifNotExists()
49
+ .addColumn("nonce", "varchar(255)", (cb) => cb.primaryKey())
50
+ .addColumn("recipient", "varchar(255)")
51
+ .addColumn("mailID", "varchar(255)")
52
+ .addColumn("sender", "varchar(255)")
53
+ .addColumn("header", "text")
54
+ .addColumn("cipher", "text")
55
+ .addColumn("group", "varchar(255)")
56
+ .addColumn("extra", "text")
57
+ .addColumn("mailType", "integer")
58
+ .addColumn("time", "text")
59
+ .addColumn("forward", "integer", (cb) => cb.defaultTo(0))
60
+ .addColumn("authorID", "varchar(255)")
61
+ .addColumn("readerID", "varchar(255)")
62
+ .execute();
63
+
64
+ await db.schema
65
+ .createIndex("mail_recipient_idx")
66
+ .ifNotExists()
67
+ .on("mail")
68
+ .column("recipient")
69
+ .execute();
70
+
71
+ await db.schema
72
+ .createTable("preKeys")
73
+ .ifNotExists()
74
+ .addColumn("keyID", "varchar(255)", (cb) => cb.primaryKey())
75
+ .addColumn("userID", "varchar(255)")
76
+ .addColumn("deviceID", "varchar(255)", (cb) => cb.unique())
77
+ .addColumn("publicKey", "text")
78
+ .addColumn("signature", "text")
79
+ .addColumn("index", "integer")
80
+ .execute();
81
+
82
+ await db.schema
83
+ .createIndex("preKeys_userID_idx")
84
+ .ifNotExists()
85
+ .on("preKeys")
86
+ .column("userID")
87
+ .execute();
88
+
89
+ await db.schema
90
+ .createIndex("preKeys_deviceID_idx")
91
+ .ifNotExists()
92
+ .on("preKeys")
93
+ .column("deviceID")
94
+ .execute();
95
+
96
+ await db.schema
97
+ .createTable("oneTimeKeys")
98
+ .ifNotExists()
99
+ .addColumn("keyID", "varchar(255)", (cb) => cb.primaryKey())
100
+ .addColumn("userID", "varchar(255)")
101
+ .addColumn("deviceID", "varchar(255)")
102
+ .addColumn("publicKey", "text")
103
+ .addColumn("signature", "text")
104
+ .addColumn("index", "integer")
105
+ .execute();
106
+
107
+ await db.schema
108
+ .createIndex("oneTimeKeys_userID_idx")
109
+ .ifNotExists()
110
+ .on("oneTimeKeys")
111
+ .column("userID")
112
+ .execute();
113
+
114
+ await db.schema
115
+ .createIndex("oneTimeKeys_deviceID_idx")
116
+ .ifNotExists()
117
+ .on("oneTimeKeys")
118
+ .column("deviceID")
119
+ .execute();
120
+
121
+ await db.schema
122
+ .createTable("servers")
123
+ .ifNotExists()
124
+ .addColumn("serverID", "varchar(255)", (cb) => cb.primaryKey())
125
+ .addColumn("name", "varchar(255)")
126
+ .addColumn("icon", "varchar(255)")
127
+ .execute();
128
+
129
+ await db.schema
130
+ .createTable("channels")
131
+ .ifNotExists()
132
+ .addColumn("channelID", "varchar(255)", (cb) => cb.primaryKey())
133
+ .addColumn("serverID", "varchar(255)")
134
+ .addColumn("name", "varchar(255)")
135
+ .execute();
136
+
137
+ await db.schema
138
+ .createTable("permissions")
139
+ .ifNotExists()
140
+ .addColumn("permissionID", "varchar(255)", (cb) => cb.primaryKey())
141
+ .addColumn("userID", "varchar(255)")
142
+ .addColumn("resourceType", "varchar(255)")
143
+ .addColumn("resourceID", "varchar(255)")
144
+ .addColumn("powerLevel", "integer")
145
+ .execute();
146
+
147
+ await db.schema
148
+ .createIndex("permissions_userID_idx")
149
+ .ifNotExists()
150
+ .on("permissions")
151
+ .column("userID")
152
+ .execute();
153
+
154
+ await db.schema
155
+ .createIndex("permissions_resourceID_idx")
156
+ .ifNotExists()
157
+ .on("permissions")
158
+ .column("resourceID")
159
+ .execute();
160
+
161
+ await db.schema
162
+ .createTable("files")
163
+ .ifNotExists()
164
+ .addColumn("fileID", "varchar(255)", (cb) => cb.primaryKey())
165
+ .addColumn("owner", "varchar(255)")
166
+ .addColumn("nonce", "varchar(255)")
167
+ .execute();
168
+
169
+ await db.schema
170
+ .createIndex("files_owner_idx")
171
+ .ifNotExists()
172
+ .on("files")
173
+ .column("owner")
174
+ .execute();
175
+
176
+ await db.schema
177
+ .createTable("emojis")
178
+ .ifNotExists()
179
+ .addColumn("emojiID", "varchar(255)", (cb) => cb.primaryKey())
180
+ .addColumn("owner", "varchar(255)")
181
+ .addColumn("name", "varchar(255)")
182
+ .execute();
183
+
184
+ await db.schema
185
+ .createIndex("emojis_owner_idx")
186
+ .ifNotExists()
187
+ .on("emojis")
188
+ .column("owner")
189
+ .execute();
190
+
191
+ await db.schema
192
+ .createTable("invites")
193
+ .ifNotExists()
194
+ .addColumn("inviteID", "varchar(255)", (cb) => cb.primaryKey())
195
+ .addColumn("serverID", "varchar(255)")
196
+ .addColumn("owner", "varchar(255)")
197
+ .addColumn("expiration", "text")
198
+ .execute();
199
+
200
+ await db.schema
201
+ .createIndex("invites_serverID_idx")
202
+ .ifNotExists()
203
+ .on("invites")
204
+ .column("serverID")
205
+ .execute();
206
+
207
+ await db.schema
208
+ .createTable("service_metrics")
209
+ .ifNotExists()
210
+ .addColumn("metric_key", "varchar(255)", (cb) => cb.primaryKey())
211
+ .addColumn("metric_value", "bigint", (cb) => cb.notNull().defaultTo(0))
212
+ .execute();
213
+
214
+ // Seed the requests_total metric row
215
+ await sql`INSERT OR IGNORE INTO service_metrics (metric_key, metric_value) VALUES ('requests_total', 0)`.execute(
216
+ db,
217
+ );
218
+ }
package/src/run.ts ADDED
@@ -0,0 +1,37 @@
1
+ import type { SpireOptions } from "./Spire.ts";
2
+
3
+ import { Spire } from "./Spire.ts";
4
+ import { loadEnv } from "./utils/loadEnv.ts";
5
+
6
+ function main() {
7
+ // load the environment variables — loadEnv() exits if required vars are missing
8
+ loadEnv();
9
+
10
+ const spk = process.env["SPK"];
11
+ if (!spk) {
12
+ throw new Error("SPK must be set (loadEnv should have caught this).");
13
+ }
14
+
15
+ const apiPort = process.env["API_PORT"];
16
+ const dbType = parseDbType(process.env["DB_TYPE"]);
17
+
18
+ new Spire(spk, {
19
+ ...(apiPort !== undefined ? { apiPort: Number(apiPort) } : {}),
20
+ ...(dbType !== undefined ? { dbType } : {}),
21
+ logLevel: "info",
22
+ });
23
+ }
24
+
25
+ function parseDbType(value: string | undefined): SpireOptions["dbType"] {
26
+ switch (value) {
27
+ case "mysql":
28
+ case "sqlite":
29
+ case "sqlite3":
30
+ case "sqlite3mem":
31
+ return value;
32
+ default:
33
+ return undefined;
34
+ }
35
+ }
36
+
37
+ main();