@digitaldefiance/node-express-suite 1.0.21 → 1.0.23

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 (633) hide show
  1. package/README.md +9 -0
  2. package/package.json +27 -32
  3. package/src/application-base.ts +492 -0
  4. package/src/application.ts +254 -0
  5. package/src/backup-code.ts +336 -0
  6. package/src/constants.ts +69 -0
  7. package/src/controllers/base.ts +440 -0
  8. package/{dist/controllers/index.d.ts → src/controllers/index.ts} +0 -1
  9. package/src/controllers/user.ts +1451 -0
  10. package/src/decorators/base-controller.ts +61 -0
  11. package/src/decorators/controller.ts +109 -0
  12. package/{dist/decorators/index.d.ts → src/decorators/index.ts} +0 -1
  13. package/src/decorators/zod-validation.ts +57 -0
  14. package/src/defaults.ts +94 -0
  15. package/src/documents/base.ts +7 -0
  16. package/src/documents/email-token.ts +14 -0
  17. package/{dist/documents/index.d.ts → src/documents/index.ts} +0 -1
  18. package/{dist/documents/mnemonic.d.ts → src/documents/mnemonic.ts} +5 -2
  19. package/{dist/documents/role.d.ts → src/documents/role.ts} +5 -2
  20. package/src/documents/used-direct-login-token.ts +7 -0
  21. package/{dist/documents/user-role.d.ts → src/documents/user-role.ts} +5 -2
  22. package/{dist/documents/user.d.ts → src/documents/user.ts} +4 -2
  23. package/src/enumerations/base-model-name.ts +41 -0
  24. package/{dist/enumerations/index.d.ts → src/enumerations/index.ts} +0 -1
  25. package/src/enumerations/length-encoding-type.ts +6 -0
  26. package/src/enumerations/schema-collection.ts +33 -0
  27. package/src/enumerations/symmetric-error-type.ts +4 -0
  28. package/src/environment.ts +770 -0
  29. package/src/errors/express-validation.ts +21 -0
  30. package/{dist/errors/index.d.ts → src/errors/index.ts} +0 -1
  31. package/src/errors/invalid-backup-code-version.ts +14 -0
  32. package/src/errors/invalid-jwt-token.ts +10 -0
  33. package/src/errors/invalid-model.ts +11 -0
  34. package/src/errors/invalid-new-password.ts +18 -0
  35. package/src/errors/invalid-password.ts +13 -0
  36. package/src/errors/missing-validated-data.ts +36 -0
  37. package/src/errors/mnemonic-or-password-required.ts +12 -0
  38. package/src/errors/model-not-registered.ts +11 -0
  39. package/src/errors/mongoose-validation.ts +34 -0
  40. package/src/errors/symmetric.ts +41 -0
  41. package/src/errors/token-expired.ts +10 -0
  42. package/src/get-language.ts +53 -0
  43. package/src/get-timezone.ts +45 -0
  44. package/{dist/index.d.ts → src/index.ts} +3 -2
  45. package/{dist/interfaces/api-error-response.d.ts → src/interfaces/api-error-response.ts} +2 -2
  46. package/src/interfaces/api-express-validation-error-response.ts +8 -0
  47. package/src/interfaces/api-message-response.ts +3 -0
  48. package/{dist/interfaces/api-mongo-validation-error-response.d.ts → src/interfaces/api-mongo-validation-error-response.ts} +2 -2
  49. package/{dist/interfaces/api-responses/backup-codes-response.d.ts → src/interfaces/api-responses/backup-codes-response.ts} +2 -2
  50. package/{dist/interfaces/api-responses/challenge-response.d.ts → src/interfaces/api-responses/challenge-response.ts} +3 -3
  51. package/{dist/interfaces/api-responses/code-count-response.d.ts → src/interfaces/api-responses/code-count-response.ts} +2 -2
  52. package/{dist/interfaces/api-responses/index.d.ts → src/interfaces/api-responses/index.ts} +0 -1
  53. package/{dist/interfaces/api-responses/login-response.d.ts → src/interfaces/api-responses/login-response.ts} +4 -4
  54. package/{dist/interfaces/api-responses/mnemonic-response.d.ts → src/interfaces/api-responses/mnemonic-response.ts} +2 -2
  55. package/{dist/interfaces/api-responses/registration-response.d.ts → src/interfaces/api-responses/registration-response.ts} +3 -3
  56. package/{dist/interfaces/api-responses/request-user-response.d.ts → src/interfaces/api-responses/request-user-response.ts} +2 -2
  57. package/{dist/interfaces/application.d.ts → src/interfaces/application.ts} +7 -7
  58. package/src/interfaces/backend-objects/email-token.ts +11 -0
  59. package/{dist/interfaces/backend-objects/index.d.ts → src/interfaces/backend-objects/index.ts} +0 -1
  60. package/{dist/interfaces/backend-objects/request-user.d.ts → src/interfaces/backend-objects/request-user.ts} +7 -2
  61. package/{dist/interfaces/backend-objects/role.d.ts → src/interfaces/backend-objects/role.ts} +1 -1
  62. package/src/interfaces/backend-objects/user.ts +9 -0
  63. package/src/interfaces/checksum-config.ts +4 -0
  64. package/src/interfaces/checksum-consts.ts +13 -0
  65. package/{dist/interfaces/constants.d.ts → src/interfaces/constants.ts} +5 -5
  66. package/src/interfaces/create-user-basics.ts +17 -0
  67. package/src/interfaces/csp-config.ts +35 -0
  68. package/src/interfaces/deep-partial.ts +3 -0
  69. package/{dist/interfaces/discriminator-collections.d.ts → src/interfaces/discriminator-collections.ts} +3 -3
  70. package/src/interfaces/email-service.ts +8 -0
  71. package/src/interfaces/environment-mongo.ts +76 -0
  72. package/src/interfaces/environment.ts +181 -0
  73. package/src/interfaces/failable-result.ts +6 -0
  74. package/src/interfaces/fec-consts.ts +4 -0
  75. package/src/interfaces/handleable-error-options.ts +6 -0
  76. package/{dist/interfaces/index.d.ts → src/interfaces/index.ts} +0 -1
  77. package/src/interfaces/jwt-consts.ts +23 -0
  78. package/src/interfaces/jwt-sign-response.ts +19 -0
  79. package/src/interfaces/mongo-errors.ts +5 -0
  80. package/src/interfaces/request-user.ts +50 -0
  81. package/src/interfaces/required-string-keys.ts +26 -0
  82. package/src/interfaces/schema.ts +31 -0
  83. package/src/interfaces/server-init-result.ts +37 -0
  84. package/src/interfaces/status-code-response.ts +7 -0
  85. package/src/interfaces/symmetric-encryption-results.d.ts +5 -0
  86. package/src/interfaces/symmetric-encryption-results.d.ts.map +1 -0
  87. package/src/interfaces/symmetric-encryption-results.js.map +1 -0
  88. package/src/interfaces/symmetric-encryption-results.ts +4 -0
  89. package/{dist/interfaces/token-response.d.ts → src/interfaces/token-response.ts} +2 -2
  90. package/src/middlewares/authenticate-crypto.ts +243 -0
  91. package/src/middlewares/authenticate-token.ts +152 -0
  92. package/src/middlewares/cleanup-crypto.ts +40 -0
  93. package/{dist/middlewares/index.d.ts → src/middlewares/index.ts} +0 -1
  94. package/src/middlewares/set-global-context-language.ts +24 -0
  95. package/src/middlewares.ts +120 -0
  96. package/src/model-registry.ts +75 -0
  97. package/src/models/email-token.ts +19 -0
  98. package/{dist/models/index.d.ts → src/models/index.ts} +0 -1
  99. package/src/models/mnemonic.ts +19 -0
  100. package/src/models/role.ts +19 -0
  101. package/src/models/used-direct-login-token.ts +23 -0
  102. package/src/models/user-role.ts +17 -0
  103. package/src/models/user.ts +19 -0
  104. package/src/registry/email-service-registry.ts +24 -0
  105. package/{dist/registry/index.d.ts → src/registry/index.ts} +0 -1
  106. package/src/routers/api.ts +151 -0
  107. package/src/routers/app.ts +258 -0
  108. package/src/routers/base.ts +17 -0
  109. package/{dist/routers/index.d.ts → src/routers/index.ts} +0 -1
  110. package/src/schemas/email-token.ts +91 -0
  111. package/{dist/schemas/index.d.ts → src/schemas/index.ts} +1 -2
  112. package/src/schemas/mnemonic.ts +37 -0
  113. package/src/schemas/role.ts +127 -0
  114. package/src/schemas/schema.ts +140 -0
  115. package/src/schemas/used-direct-login-token.ts +38 -0
  116. package/src/schemas/user-role.ts +75 -0
  117. package/src/schemas/user.ts +202 -0
  118. package/src/services/backup-code.ts +316 -0
  119. package/src/services/base.ts +33 -0
  120. package/src/services/checksum.ts +161 -0
  121. package/src/services/crc.ts +213 -0
  122. package/src/services/database-initialization.ts +1479 -0
  123. package/src/services/db-init-cache.d.ts +16 -0
  124. package/src/services/direct-login-token.ts +62 -0
  125. package/src/services/fec-usage-example.ts +102 -0
  126. package/src/services/fec.ts +296 -0
  127. package/{dist/services/index.d.ts → src/services/index.ts} +0 -1
  128. package/src/services/jwt.ts +134 -0
  129. package/src/services/key-wrapping.ts +434 -0
  130. package/src/services/mnemonic.ts +167 -0
  131. package/src/services/request-user.ts +62 -0
  132. package/src/services/role.ts +396 -0
  133. package/src/services/symmetric.ts +139 -0
  134. package/src/services/system-user.ts +82 -0
  135. package/src/services/user.ts +2137 -0
  136. package/src/services/xor.ts +34 -0
  137. package/src/types.d.ts +44 -0
  138. package/src/types.ts +128 -0
  139. package/src/utils.ts +1022 -0
  140. package/dist/application-base.d.ts +0 -112
  141. package/dist/application-base.d.ts.map +0 -1
  142. package/dist/application-base.js +0 -301
  143. package/dist/application-base.js.map +0 -1
  144. package/dist/application.d.ts +0 -23
  145. package/dist/application.d.ts.map +0 -1
  146. package/dist/application.js +0 -126
  147. package/dist/application.js.map +0 -1
  148. package/dist/backup-code.d.ts +0 -67
  149. package/dist/backup-code.d.ts.map +0 -1
  150. package/dist/backup-code.js +0 -270
  151. package/dist/backup-code.js.map +0 -1
  152. package/dist/constants.d.ts +0 -16
  153. package/dist/constants.d.ts.map +0 -1
  154. package/dist/constants.js +0 -54
  155. package/dist/constants.js.map +0 -1
  156. package/dist/controllers/base.d.ts +0 -63
  157. package/dist/controllers/base.d.ts.map +0 -1
  158. package/dist/controllers/base.js +0 -269
  159. package/dist/controllers/base.js.map +0 -1
  160. package/dist/controllers/index.d.ts.map +0 -1
  161. package/dist/controllers/index.js +0 -19
  162. package/dist/controllers/index.js.map +0 -1
  163. package/dist/controllers/user.d.ts +0 -45
  164. package/dist/controllers/user.d.ts.map +0 -1
  165. package/dist/controllers/user.js +0 -750
  166. package/dist/controllers/user.js.map +0 -1
  167. package/dist/decorators/base-controller.d.ts +0 -14
  168. package/dist/decorators/base-controller.d.ts.map +0 -1
  169. package/dist/decorators/base-controller.js +0 -49
  170. package/dist/decorators/base-controller.js.map +0 -1
  171. package/dist/decorators/controller.d.ts +0 -32
  172. package/dist/decorators/controller.d.ts.map +0 -1
  173. package/dist/decorators/controller.js +0 -67
  174. package/dist/decorators/controller.js.map +0 -1
  175. package/dist/decorators/index.d.ts.map +0 -1
  176. package/dist/decorators/index.js +0 -20
  177. package/dist/decorators/index.js.map +0 -1
  178. package/dist/decorators/zod-validation.d.ts +0 -5
  179. package/dist/decorators/zod-validation.d.ts.map +0 -1
  180. package/dist/decorators/zod-validation.js +0 -47
  181. package/dist/decorators/zod-validation.js.map +0 -1
  182. package/dist/defaults.d.ts +0 -7
  183. package/dist/defaults.d.ts.map +0 -1
  184. package/dist/defaults.js +0 -83
  185. package/dist/defaults.js.map +0 -1
  186. package/dist/documents/base.d.ts +0 -3
  187. package/dist/documents/base.d.ts.map +0 -1
  188. package/dist/documents/base.js +0 -3
  189. package/dist/documents/base.js.map +0 -1
  190. package/dist/documents/email-token.d.ts +0 -8
  191. package/dist/documents/email-token.d.ts.map +0 -1
  192. package/dist/documents/email-token.js +0 -3
  193. package/dist/documents/email-token.js.map +0 -1
  194. package/dist/documents/index.d.ts.map +0 -1
  195. package/dist/documents/index.js +0 -3
  196. package/dist/documents/index.js.map +0 -1
  197. package/dist/documents/mnemonic.d.ts.map +0 -1
  198. package/dist/documents/mnemonic.js +0 -3
  199. package/dist/documents/mnemonic.js.map +0 -1
  200. package/dist/documents/role.d.ts.map +0 -1
  201. package/dist/documents/role.js +0 -3
  202. package/dist/documents/role.js.map +0 -1
  203. package/dist/documents/used-direct-login-token.d.ts +0 -5
  204. package/dist/documents/used-direct-login-token.d.ts.map +0 -1
  205. package/dist/documents/used-direct-login-token.js +0 -3
  206. package/dist/documents/used-direct-login-token.js.map +0 -1
  207. package/dist/documents/user-role.d.ts.map +0 -1
  208. package/dist/documents/user-role.js +0 -3
  209. package/dist/documents/user-role.js.map +0 -1
  210. package/dist/documents/user.d.ts.map +0 -1
  211. package/dist/documents/user.js +0 -3
  212. package/dist/documents/user.js.map +0 -1
  213. package/dist/enumerations/base-model-name.d.ts +0 -38
  214. package/dist/enumerations/base-model-name.d.ts.map +0 -1
  215. package/dist/enumerations/base-model-name.js +0 -34
  216. package/dist/enumerations/base-model-name.js.map +0 -1
  217. package/dist/enumerations/index.d.ts.map +0 -1
  218. package/dist/enumerations/index.js +0 -21
  219. package/dist/enumerations/index.js.map +0 -1
  220. package/dist/enumerations/length-encoding-type.d.ts +0 -7
  221. package/dist/enumerations/length-encoding-type.d.ts.map +0 -1
  222. package/dist/enumerations/length-encoding-type.js +0 -11
  223. package/dist/enumerations/length-encoding-type.js.map +0 -1
  224. package/dist/enumerations/schema-collection.d.ts +0 -34
  225. package/dist/enumerations/schema-collection.d.ts.map +0 -1
  226. package/dist/enumerations/schema-collection.js +0 -38
  227. package/dist/enumerations/schema-collection.js.map +0 -1
  228. package/dist/enumerations/symmetric-error-type.d.ts +0 -5
  229. package/dist/enumerations/symmetric-error-type.d.ts.map +0 -1
  230. package/dist/enumerations/symmetric-error-type.js +0 -9
  231. package/dist/enumerations/symmetric-error-type.js.map +0 -1
  232. package/dist/environment.d.ts +0 -189
  233. package/dist/environment.d.ts.map +0 -1
  234. package/dist/environment.js +0 -618
  235. package/dist/environment.js.map +0 -1
  236. package/dist/errors/express-validation.d.ts +0 -9
  237. package/dist/errors/express-validation.d.ts.map +0 -1
  238. package/dist/errors/express-validation.js +0 -17
  239. package/dist/errors/express-validation.js.map +0 -1
  240. package/dist/errors/index.d.ts.map +0 -1
  241. package/dist/errors/index.js +0 -29
  242. package/dist/errors/index.js.map +0 -1
  243. package/dist/errors/invalid-backup-code-version.d.ts +0 -6
  244. package/dist/errors/invalid-backup-code-version.d.ts.map +0 -1
  245. package/dist/errors/invalid-backup-code-version.js +0 -14
  246. package/dist/errors/invalid-backup-code-version.js.map +0 -1
  247. package/dist/errors/invalid-jwt-token.d.ts +0 -5
  248. package/dist/errors/invalid-jwt-token.d.ts.map +0 -1
  249. package/dist/errors/invalid-jwt-token.js +0 -11
  250. package/dist/errors/invalid-jwt-token.js.map +0 -1
  251. package/dist/errors/invalid-model.d.ts +0 -6
  252. package/dist/errors/invalid-model.d.ts.map +0 -1
  253. package/dist/errors/invalid-model.js +0 -13
  254. package/dist/errors/invalid-model.js.map +0 -1
  255. package/dist/errors/invalid-new-password.d.ts +0 -5
  256. package/dist/errors/invalid-new-password.d.ts.map +0 -1
  257. package/dist/errors/invalid-new-password.js +0 -14
  258. package/dist/errors/invalid-new-password.js.map +0 -1
  259. package/dist/errors/invalid-password.d.ts +0 -5
  260. package/dist/errors/invalid-password.d.ts.map +0 -1
  261. package/dist/errors/invalid-password.js +0 -14
  262. package/dist/errors/invalid-password.js.map +0 -1
  263. package/dist/errors/missing-validated-data.d.ts +0 -7
  264. package/dist/errors/missing-validated-data.d.ts.map +0 -1
  265. package/dist/errors/missing-validated-data.js +0 -34
  266. package/dist/errors/missing-validated-data.js.map +0 -1
  267. package/dist/errors/mnemonic-or-password-required.d.ts +0 -5
  268. package/dist/errors/mnemonic-or-password-required.d.ts.map +0 -1
  269. package/dist/errors/mnemonic-or-password-required.js +0 -13
  270. package/dist/errors/mnemonic-or-password-required.js.map +0 -1
  271. package/dist/errors/model-not-registered.d.ts +0 -5
  272. package/dist/errors/model-not-registered.d.ts.map +0 -1
  273. package/dist/errors/model-not-registered.js +0 -12
  274. package/dist/errors/model-not-registered.js.map +0 -1
  275. package/dist/errors/mongoose-validation.d.ts +0 -11
  276. package/dist/errors/mongoose-validation.d.ts.map +0 -1
  277. package/dist/errors/mongoose-validation.js +0 -16
  278. package/dist/errors/mongoose-validation.js.map +0 -1
  279. package/dist/errors/symmetric.d.ts +0 -8
  280. package/dist/errors/symmetric.d.ts.map +0 -1
  281. package/dist/errors/symmetric.js +0 -23
  282. package/dist/errors/symmetric.js.map +0 -1
  283. package/dist/errors/token-expired.d.ts +0 -5
  284. package/dist/errors/token-expired.d.ts.map +0 -1
  285. package/dist/errors/token-expired.js +0 -11
  286. package/dist/errors/token-expired.js.map +0 -1
  287. package/dist/get-language.d.ts +0 -2
  288. package/dist/get-language.d.ts.map +0 -1
  289. package/dist/get-language.js +0 -30
  290. package/dist/get-language.js.map +0 -1
  291. package/dist/get-timezone.d.ts +0 -3
  292. package/dist/get-timezone.d.ts.map +0 -1
  293. package/dist/get-timezone.js +0 -31
  294. package/dist/get-timezone.js.map +0 -1
  295. package/dist/index.d.ts.map +0 -1
  296. package/dist/index.js +0 -40
  297. package/dist/index.js.map +0 -1
  298. package/dist/interfaces/api-error-response.d.ts.map +0 -1
  299. package/dist/interfaces/api-error-response.js +0 -3
  300. package/dist/interfaces/api-error-response.js.map +0 -1
  301. package/dist/interfaces/api-express-validation-error-response.d.ts +0 -7
  302. package/dist/interfaces/api-express-validation-error-response.d.ts.map +0 -1
  303. package/dist/interfaces/api-express-validation-error-response.js +0 -3
  304. package/dist/interfaces/api-express-validation-error-response.js.map +0 -1
  305. package/dist/interfaces/api-message-response.d.ts +0 -4
  306. package/dist/interfaces/api-message-response.d.ts.map +0 -1
  307. package/dist/interfaces/api-message-response.js +0 -3
  308. package/dist/interfaces/api-message-response.js.map +0 -1
  309. package/dist/interfaces/api-mongo-validation-error-response.d.ts.map +0 -1
  310. package/dist/interfaces/api-mongo-validation-error-response.js +0 -3
  311. package/dist/interfaces/api-mongo-validation-error-response.js.map +0 -1
  312. package/dist/interfaces/api-responses/backup-codes-response.d.ts.map +0 -1
  313. package/dist/interfaces/api-responses/backup-codes-response.js +0 -3
  314. package/dist/interfaces/api-responses/backup-codes-response.js.map +0 -1
  315. package/dist/interfaces/api-responses/challenge-response.d.ts.map +0 -1
  316. package/dist/interfaces/api-responses/challenge-response.js +0 -3
  317. package/dist/interfaces/api-responses/challenge-response.js.map +0 -1
  318. package/dist/interfaces/api-responses/code-count-response.d.ts.map +0 -1
  319. package/dist/interfaces/api-responses/code-count-response.js +0 -3
  320. package/dist/interfaces/api-responses/code-count-response.js.map +0 -1
  321. package/dist/interfaces/api-responses/index.d.ts.map +0 -1
  322. package/dist/interfaces/api-responses/index.js +0 -24
  323. package/dist/interfaces/api-responses/index.js.map +0 -1
  324. package/dist/interfaces/api-responses/login-response.d.ts.map +0 -1
  325. package/dist/interfaces/api-responses/login-response.js +0 -3
  326. package/dist/interfaces/api-responses/login-response.js.map +0 -1
  327. package/dist/interfaces/api-responses/mnemonic-response.d.ts.map +0 -1
  328. package/dist/interfaces/api-responses/mnemonic-response.js +0 -3
  329. package/dist/interfaces/api-responses/mnemonic-response.js.map +0 -1
  330. package/dist/interfaces/api-responses/registration-response.d.ts.map +0 -1
  331. package/dist/interfaces/api-responses/registration-response.js +0 -3
  332. package/dist/interfaces/api-responses/registration-response.js.map +0 -1
  333. package/dist/interfaces/api-responses/request-user-response.d.ts.map +0 -1
  334. package/dist/interfaces/api-responses/request-user-response.js +0 -3
  335. package/dist/interfaces/api-responses/request-user-response.js.map +0 -1
  336. package/dist/interfaces/application.d.ts.map +0 -1
  337. package/dist/interfaces/application.js +0 -3
  338. package/dist/interfaces/application.js.map +0 -1
  339. package/dist/interfaces/backend-objects/email-token.d.ts +0 -4
  340. package/dist/interfaces/backend-objects/email-token.d.ts.map +0 -1
  341. package/dist/interfaces/backend-objects/email-token.js +0 -3
  342. package/dist/interfaces/backend-objects/email-token.js.map +0 -1
  343. package/dist/interfaces/backend-objects/index.d.ts.map +0 -1
  344. package/dist/interfaces/backend-objects/index.js +0 -21
  345. package/dist/interfaces/backend-objects/index.js.map +0 -1
  346. package/dist/interfaces/backend-objects/request-user.d.ts.map +0 -1
  347. package/dist/interfaces/backend-objects/request-user.js +0 -3
  348. package/dist/interfaces/backend-objects/request-user.js.map +0 -1
  349. package/dist/interfaces/backend-objects/role.d.ts.map +0 -1
  350. package/dist/interfaces/backend-objects/role.js +0 -3
  351. package/dist/interfaces/backend-objects/role.js.map +0 -1
  352. package/dist/interfaces/backend-objects/user.d.ts +0 -4
  353. package/dist/interfaces/backend-objects/user.d.ts.map +0 -1
  354. package/dist/interfaces/backend-objects/user.js +0 -3
  355. package/dist/interfaces/backend-objects/user.js.map +0 -1
  356. package/dist/interfaces/checksum-config.d.ts +0 -5
  357. package/dist/interfaces/checksum-config.d.ts.map +0 -1
  358. package/dist/interfaces/checksum-config.js +0 -3
  359. package/dist/interfaces/checksum-config.js.map +0 -1
  360. package/dist/interfaces/checksum-consts.d.ts +0 -11
  361. package/dist/interfaces/checksum-consts.d.ts.map +0 -1
  362. package/dist/interfaces/checksum-consts.js +0 -3
  363. package/dist/interfaces/checksum-consts.js.map +0 -1
  364. package/dist/interfaces/constants.d.ts.map +0 -1
  365. package/dist/interfaces/constants.js +0 -3
  366. package/dist/interfaces/constants.js.map +0 -1
  367. package/dist/interfaces/create-user-basics.d.ts +0 -18
  368. package/dist/interfaces/create-user-basics.d.ts.map +0 -1
  369. package/dist/interfaces/create-user-basics.js +0 -3
  370. package/dist/interfaces/create-user-basics.js.map +0 -1
  371. package/dist/interfaces/csp-config.d.ts +0 -14
  372. package/dist/interfaces/csp-config.d.ts.map +0 -1
  373. package/dist/interfaces/csp-config.js +0 -3
  374. package/dist/interfaces/csp-config.js.map +0 -1
  375. package/dist/interfaces/deep-partial.d.ts +0 -4
  376. package/dist/interfaces/deep-partial.d.ts.map +0 -1
  377. package/dist/interfaces/deep-partial.js +0 -3
  378. package/dist/interfaces/deep-partial.js.map +0 -1
  379. package/dist/interfaces/discriminator-collections.d.ts.map +0 -1
  380. package/dist/interfaces/discriminator-collections.js +0 -3
  381. package/dist/interfaces/discriminator-collections.js.map +0 -1
  382. package/dist/interfaces/email-service.d.ts +0 -4
  383. package/dist/interfaces/email-service.d.ts.map +0 -1
  384. package/dist/interfaces/email-service.js +0 -3
  385. package/dist/interfaces/email-service.js.map +0 -1
  386. package/dist/interfaces/environment-mongo.d.ts +0 -76
  387. package/dist/interfaces/environment-mongo.d.ts.map +0 -1
  388. package/dist/interfaces/environment-mongo.js +0 -3
  389. package/dist/interfaces/environment-mongo.js.map +0 -1
  390. package/dist/interfaces/environment.d.ts +0 -181
  391. package/dist/interfaces/environment.d.ts.map +0 -1
  392. package/dist/interfaces/environment.js +0 -3
  393. package/dist/interfaces/environment.js.map +0 -1
  394. package/dist/interfaces/failable-result.d.ts +0 -7
  395. package/dist/interfaces/failable-result.d.ts.map +0 -1
  396. package/dist/interfaces/failable-result.js +0 -3
  397. package/dist/interfaces/failable-result.js.map +0 -1
  398. package/dist/interfaces/fec-consts.d.ts +0 -5
  399. package/dist/interfaces/fec-consts.d.ts.map +0 -1
  400. package/dist/interfaces/fec-consts.js +0 -3
  401. package/dist/interfaces/fec-consts.js.map +0 -1
  402. package/dist/interfaces/handleable-error-options.d.ts +0 -7
  403. package/dist/interfaces/handleable-error-options.d.ts.map +0 -1
  404. package/dist/interfaces/handleable-error-options.js +0 -3
  405. package/dist/interfaces/handleable-error-options.js.map +0 -1
  406. package/dist/interfaces/index.d.ts.map +0 -1
  407. package/dist/interfaces/index.js +0 -46
  408. package/dist/interfaces/index.js.map +0 -1
  409. package/dist/interfaces/jwt-consts.d.ts +0 -11
  410. package/dist/interfaces/jwt-consts.d.ts.map +0 -1
  411. package/dist/interfaces/jwt-consts.js +0 -3
  412. package/dist/interfaces/jwt-consts.js.map +0 -1
  413. package/dist/interfaces/jwt-sign-response.d.ts +0 -11
  414. package/dist/interfaces/jwt-sign-response.d.ts.map +0 -1
  415. package/dist/interfaces/jwt-sign-response.js +0 -3
  416. package/dist/interfaces/jwt-sign-response.js.map +0 -1
  417. package/dist/interfaces/mongo-errors.d.ts +0 -5
  418. package/dist/interfaces/mongo-errors.d.ts.map +0 -1
  419. package/dist/interfaces/mongo-errors.js +0 -3
  420. package/dist/interfaces/mongo-errors.js.map +0 -1
  421. package/dist/interfaces/request-user.d.ts +0 -42
  422. package/dist/interfaces/request-user.d.ts.map +0 -1
  423. package/dist/interfaces/request-user.js +0 -3
  424. package/dist/interfaces/request-user.js.map +0 -1
  425. package/dist/interfaces/required-string-keys.d.ts +0 -22
  426. package/dist/interfaces/required-string-keys.d.ts.map +0 -1
  427. package/dist/interfaces/required-string-keys.js +0 -3
  428. package/dist/interfaces/required-string-keys.js.map +0 -1
  429. package/dist/interfaces/schema.d.ts +0 -29
  430. package/dist/interfaces/schema.d.ts.map +0 -1
  431. package/dist/interfaces/schema.js +0 -3
  432. package/dist/interfaces/schema.js.map +0 -1
  433. package/dist/interfaces/server-init-result.d.ts +0 -35
  434. package/dist/interfaces/server-init-result.d.ts.map +0 -1
  435. package/dist/interfaces/server-init-result.js +0 -3
  436. package/dist/interfaces/server-init-result.js.map +0 -1
  437. package/dist/interfaces/status-code-response.d.ts +0 -7
  438. package/dist/interfaces/status-code-response.d.ts.map +0 -1
  439. package/dist/interfaces/status-code-response.js +0 -3
  440. package/dist/interfaces/status-code-response.js.map +0 -1
  441. package/dist/interfaces/symmetric-encryption-results.d.ts +0 -5
  442. package/dist/interfaces/symmetric-encryption-results.d.ts.map +0 -1
  443. package/dist/interfaces/symmetric-encryption-results.js.map +0 -1
  444. package/dist/interfaces/token-response.d.ts.map +0 -1
  445. package/dist/interfaces/token-response.js +0 -3
  446. package/dist/interfaces/token-response.js.map +0 -1
  447. package/dist/middlewares/authenticate-crypto.d.ts +0 -13
  448. package/dist/middlewares/authenticate-crypto.d.ts.map +0 -1
  449. package/dist/middlewares/authenticate-crypto.js +0 -146
  450. package/dist/middlewares/authenticate-crypto.js.map +0 -1
  451. package/dist/middlewares/authenticate-token.d.ts +0 -24
  452. package/dist/middlewares/authenticate-token.d.ts.map +0 -1
  453. package/dist/middlewares/authenticate-token.js +0 -102
  454. package/dist/middlewares/authenticate-token.js.map +0 -1
  455. package/dist/middlewares/cleanup-crypto.d.ts +0 -7
  456. package/dist/middlewares/cleanup-crypto.d.ts.map +0 -1
  457. package/dist/middlewares/cleanup-crypto.js +0 -32
  458. package/dist/middlewares/cleanup-crypto.js.map +0 -1
  459. package/dist/middlewares/index.d.ts.map +0 -1
  460. package/dist/middlewares/index.js +0 -21
  461. package/dist/middlewares/index.js.map +0 -1
  462. package/dist/middlewares/set-global-context-language.d.ts +0 -3
  463. package/dist/middlewares/set-global-context-language.d.ts.map +0 -1
  464. package/dist/middlewares/set-global-context-language.js +0 -14
  465. package/dist/middlewares/set-global-context-language.js.map +0 -1
  466. package/dist/middlewares.d.ts +0 -18
  467. package/dist/middlewares.d.ts.map +0 -1
  468. package/dist/middlewares.js +0 -76
  469. package/dist/middlewares.js.map +0 -1
  470. package/dist/model-registry.d.ts +0 -23
  471. package/dist/model-registry.d.ts.map +0 -1
  472. package/dist/model-registry.js +0 -47
  473. package/dist/model-registry.js.map +0 -1
  474. package/dist/models/email-token.d.ts +0 -11
  475. package/dist/models/email-token.d.ts.map +0 -1
  476. package/dist/models/email-token.js +0 -11
  477. package/dist/models/email-token.js.map +0 -1
  478. package/dist/models/index.d.ts.map +0 -1
  479. package/dist/models/index.js +0 -23
  480. package/dist/models/index.js.map +0 -1
  481. package/dist/models/mnemonic.d.ts +0 -11
  482. package/dist/models/mnemonic.d.ts.map +0 -1
  483. package/dist/models/mnemonic.js +0 -11
  484. package/dist/models/mnemonic.js.map +0 -1
  485. package/dist/models/role.d.ts +0 -11
  486. package/dist/models/role.d.ts.map +0 -1
  487. package/dist/models/role.js +0 -11
  488. package/dist/models/role.js.map +0 -1
  489. package/dist/models/used-direct-login-token.d.ts +0 -11
  490. package/dist/models/used-direct-login-token.d.ts.map +0 -1
  491. package/dist/models/used-direct-login-token.js +0 -11
  492. package/dist/models/used-direct-login-token.js.map +0 -1
  493. package/dist/models/user-role.d.ts +0 -6
  494. package/dist/models/user-role.d.ts.map +0 -1
  495. package/dist/models/user-role.js +0 -10
  496. package/dist/models/user-role.js.map +0 -1
  497. package/dist/models/user.d.ts +0 -7
  498. package/dist/models/user.d.ts.map +0 -1
  499. package/dist/models/user.js +0 -11
  500. package/dist/models/user.js.map +0 -1
  501. package/dist/registry/email-service-registry.d.ts +0 -9
  502. package/dist/registry/email-service-registry.d.ts.map +0 -1
  503. package/dist/registry/email-service-registry.js +0 -17
  504. package/dist/registry/email-service-registry.js.map +0 -1
  505. package/dist/registry/index.d.ts.map +0 -1
  506. package/dist/registry/index.js +0 -6
  507. package/dist/registry/index.js.map +0 -1
  508. package/dist/routers/api.d.ts +0 -27
  509. package/dist/routers/api.d.ts.map +0 -1
  510. package/dist/routers/api.js +0 -44
  511. package/dist/routers/api.js.map +0 -1
  512. package/dist/routers/app.d.ts +0 -28
  513. package/dist/routers/app.d.ts.map +0 -1
  514. package/dist/routers/app.js +0 -182
  515. package/dist/routers/app.js.map +0 -1
  516. package/dist/routers/base.d.ts +0 -12
  517. package/dist/routers/base.d.ts.map +0 -1
  518. package/dist/routers/base.js +0 -12
  519. package/dist/routers/base.js.map +0 -1
  520. package/dist/routers/index.d.ts.map +0 -1
  521. package/dist/routers/index.js +0 -20
  522. package/dist/routers/index.js.map +0 -1
  523. package/dist/schemas/email-token.d.ts +0 -38
  524. package/dist/schemas/email-token.d.ts.map +0 -1
  525. package/dist/schemas/email-token.js +0 -56
  526. package/dist/schemas/email-token.js.map +0 -1
  527. package/dist/schemas/index.d.ts.map +0 -1
  528. package/dist/schemas/index.js +0 -24
  529. package/dist/schemas/index.js.map +0 -1
  530. package/dist/schemas/mnemonic.d.ts +0 -20
  531. package/dist/schemas/mnemonic.d.ts.map +0 -1
  532. package/dist/schemas/mnemonic.js +0 -30
  533. package/dist/schemas/mnemonic.js.map +0 -1
  534. package/dist/schemas/role.d.ts +0 -32
  535. package/dist/schemas/role.d.ts.map +0 -1
  536. package/dist/schemas/role.js +0 -86
  537. package/dist/schemas/role.js.map +0 -1
  538. package/dist/schemas/schema.d.ts +0 -40
  539. package/dist/schemas/schema.d.ts.map +0 -1
  540. package/dist/schemas/schema.js +0 -64
  541. package/dist/schemas/schema.js.map +0 -1
  542. package/dist/schemas/used-direct-login-token.d.ts +0 -27
  543. package/dist/schemas/used-direct-login-token.d.ts.map +0 -1
  544. package/dist/schemas/used-direct-login-token.js +0 -23
  545. package/dist/schemas/used-direct-login-token.js.map +0 -1
  546. package/dist/schemas/user-role.d.ts +0 -29
  547. package/dist/schemas/user-role.d.ts.map +0 -1
  548. package/dist/schemas/user-role.js +0 -54
  549. package/dist/schemas/user-role.js.map +0 -1
  550. package/dist/schemas/user.d.ts +0 -21
  551. package/dist/schemas/user.d.ts.map +0 -1
  552. package/dist/schemas/user.js +0 -178
  553. package/dist/schemas/user.js.map +0 -1
  554. package/dist/services/backup-code.d.ts +0 -78
  555. package/dist/services/backup-code.d.ts.map +0 -1
  556. package/dist/services/backup-code.js +0 -180
  557. package/dist/services/backup-code.js.map +0 -1
  558. package/dist/services/base.d.ts +0 -13
  559. package/dist/services/base.d.ts.map +0 -1
  560. package/dist/services/base.js +0 -14
  561. package/dist/services/base.js.map +0 -1
  562. package/dist/services/checksum.d.ts +0 -67
  563. package/dist/services/checksum.d.ts.map +0 -1
  564. package/dist/services/checksum.js +0 -175
  565. package/dist/services/checksum.js.map +0 -1
  566. package/dist/services/crc.d.ts +0 -87
  567. package/dist/services/crc.d.ts.map +0 -1
  568. package/dist/services/crc.js +0 -198
  569. package/dist/services/crc.js.map +0 -1
  570. package/dist/services/database-initialization.d.ts +0 -105
  571. package/dist/services/database-initialization.d.ts.map +0 -1
  572. package/dist/services/database-initialization.js +0 -779
  573. package/dist/services/database-initialization.js.map +0 -1
  574. package/dist/services/direct-login-token.d.ts +0 -9
  575. package/dist/services/direct-login-token.d.ts.map +0 -1
  576. package/dist/services/direct-login-token.js +0 -41
  577. package/dist/services/direct-login-token.js.map +0 -1
  578. package/dist/services/fec-usage-example.d.ts +0 -38
  579. package/dist/services/fec-usage-example.d.ts.map +0 -1
  580. package/dist/services/fec-usage-example.js +0 -77
  581. package/dist/services/fec-usage-example.js.map +0 -1
  582. package/dist/services/fec.d.ts +0 -46
  583. package/dist/services/fec.d.ts.map +0 -1
  584. package/dist/services/fec.js +0 -192
  585. package/dist/services/fec.js.map +0 -1
  586. package/dist/services/index.d.ts.map +0 -1
  587. package/dist/services/index.js +0 -35
  588. package/dist/services/index.js.map +0 -1
  589. package/dist/services/jwt.d.ts +0 -33
  590. package/dist/services/jwt.d.ts.map +0 -1
  591. package/dist/services/jwt.js +0 -90
  592. package/dist/services/jwt.js.map +0 -1
  593. package/dist/services/key-wrapping.d.ts +0 -60
  594. package/dist/services/key-wrapping.d.ts.map +0 -1
  595. package/dist/services/key-wrapping.js +0 -311
  596. package/dist/services/key-wrapping.js.map +0 -1
  597. package/dist/services/mnemonic.d.ts +0 -61
  598. package/dist/services/mnemonic.d.ts.map +0 -1
  599. package/dist/services/mnemonic.js +0 -112
  600. package/dist/services/mnemonic.js.map +0 -1
  601. package/dist/services/request-user.d.ts +0 -20
  602. package/dist/services/request-user.d.ts.map +0 -1
  603. package/dist/services/request-user.js +0 -50
  604. package/dist/services/request-user.js.map +0 -1
  605. package/dist/services/role.d.ts +0 -88
  606. package/dist/services/role.d.ts.map +0 -1
  607. package/dist/services/role.js +0 -263
  608. package/dist/services/role.js.map +0 -1
  609. package/dist/services/symmetric.d.ts +0 -42
  610. package/dist/services/symmetric.d.ts.map +0 -1
  611. package/dist/services/symmetric.js +0 -101
  612. package/dist/services/symmetric.js.map +0 -1
  613. package/dist/services/system-user.d.ts +0 -17
  614. package/dist/services/system-user.d.ts.map +0 -1
  615. package/dist/services/system-user.js +0 -46
  616. package/dist/services/system-user.js.map +0 -1
  617. package/dist/services/user.d.ts +0 -320
  618. package/dist/services/user.d.ts.map +0 -1
  619. package/dist/services/user.js +0 -1374
  620. package/dist/services/user.js.map +0 -1
  621. package/dist/services/xor.d.ts +0 -24
  622. package/dist/services/xor.d.ts.map +0 -1
  623. package/dist/services/xor.js +0 -37
  624. package/dist/services/xor.js.map +0 -1
  625. package/dist/types.d.ts +0 -70
  626. package/dist/types.d.ts.map +0 -1
  627. package/dist/types.js +0 -14
  628. package/dist/types.js.map +0 -1
  629. package/dist/utils.d.ts +0 -202
  630. package/dist/utils.d.ts.map +0 -1
  631. package/dist/utils.js +0 -786
  632. package/dist/utils.js.map +0 -1
  633. /package/{dist → src}/interfaces/symmetric-encryption-results.js +0 -0
@@ -0,0 +1,254 @@
1
+ import { HandleableError } from '@digitaldefiance/i18n-lib';
2
+ import {
3
+ Constants,
4
+ getSuiteCoreI18nEngine,
5
+ SuiteCoreComponentId,
6
+ SuiteCoreStringKey,
7
+ TranslatableSuiteError,
8
+ } from '@digitaldefiance/suite-core-lib';
9
+ import express, {
10
+ Application as ExpressApplication,
11
+ NextFunction,
12
+ Request,
13
+ Response,
14
+ } from 'express';
15
+ import { readFileSync } from 'fs';
16
+ import { Server } from 'http';
17
+ import { createServer } from 'https';
18
+ import mongoose, { Types } from 'mongoose';
19
+ import { isAbsolute, normalize, resolve } from 'path';
20
+ import { BaseApplication } from './application-base';
21
+ import { IBaseDocument } from './documents/base';
22
+ import { Environment } from './environment';
23
+ import { IApplication, ICSPConfig, IFailableResult } from './interfaces';
24
+ import { IConstants } from './interfaces/constants';
25
+ import { Middlewares } from './middlewares';
26
+ import { AppRouter } from './routers/app';
27
+ import { BaseRouter } from './routers/base';
28
+ import { SchemaMap } from './types';
29
+ import { debugLog, handleError, sendApiMessageResponse } from './utils';
30
+
31
+ /**
32
+ * Application class
33
+ */
34
+ type ServerWithOptionalClose = Server & { closeAllConnections?: () => void };
35
+
36
+ export class Application<
37
+ T,
38
+ I extends Types.ObjectId | string,
39
+ TInitResults,
40
+ TModelDocs extends Record<string, IBaseDocument<any>>,
41
+ TBaseDocument extends IBaseDocument<T, I> = IBaseDocument<T, I>,
42
+ TEnvironment extends Environment = Environment,
43
+ TConstants extends IConstants = IConstants,
44
+ >
45
+ extends BaseApplication<TModelDocs, TInitResults, TConstants>
46
+ implements IApplication<T, I, TBaseDocument, TEnvironment, TConstants>
47
+ {
48
+ public readonly expressApp: ExpressApplication;
49
+ private server: ServerWithOptionalClose | null = null;
50
+ private readonly _cspConfig: ICSPConfig;
51
+ private readonly _apiRouter: BaseRouter;
52
+
53
+ public override get environment(): TEnvironment {
54
+ return super.environment as TEnvironment;
55
+ }
56
+
57
+ constructor(
58
+ environment: TEnvironment,
59
+ apiRouter: BaseRouter,
60
+ schemaMapFactory: (
61
+ connection: mongoose.Connection,
62
+ ) => SchemaMap<TModelDocs>,
63
+ databaseInitFunction: (
64
+ application: BaseApplication<TModelDocs, TInitResults>,
65
+ ) => Promise<IFailableResult<TInitResults>>,
66
+ initResultHashFunction: (initResults: TInitResults) => string,
67
+ cspConfig: ICSPConfig = {
68
+ corsWhitelist: [],
69
+ csp: {
70
+ defaultSrc: [],
71
+ imgSrc: [],
72
+ connectSrc: [],
73
+ scriptSrc: [],
74
+ styleSrc: [],
75
+ fontSrc: [],
76
+ frameSrc: [],
77
+ },
78
+ },
79
+ constants: TConstants = Constants as TConstants,
80
+ ) {
81
+ super(
82
+ environment,
83
+ schemaMapFactory,
84
+ databaseInitFunction,
85
+ initResultHashFunction,
86
+ constants,
87
+ );
88
+ this._apiRouter = apiRouter;
89
+ this.expressApp = express();
90
+ this.server = null;
91
+ this._cspConfig = cspConfig;
92
+ }
93
+
94
+ public override async start(mongoUri?: string): Promise<void> {
95
+ const engine = getSuiteCoreI18nEngine();
96
+ await super.start(mongoUri, true);
97
+ try {
98
+ Middlewares.init(
99
+ this.expressApp,
100
+ this._cspConfig.corsWhitelist,
101
+ this._cspConfig.csp,
102
+ );
103
+ const appRouter = new AppRouter(this._apiRouter);
104
+
105
+ appRouter.init(this.expressApp);
106
+ this.expressApp.use(
107
+ (
108
+ err: HandleableError | Error,
109
+ req: Request,
110
+ res: Response,
111
+ next: NextFunction,
112
+ ) => {
113
+ const handleableError =
114
+ err instanceof HandleableError
115
+ ? err
116
+ : new HandleableError(
117
+ new Error(
118
+ err.message ||
119
+ engine.translate(
120
+ SuiteCoreComponentId,
121
+ SuiteCoreStringKey.Common_UnexpectedError,
122
+ ),
123
+ ),
124
+ { cause: err },
125
+ );
126
+ handleError(handleableError, res, sendApiMessageResponse, next);
127
+ },
128
+ );
129
+
130
+ const serversReady: Promise<void>[] = [];
131
+ serversReady.push(
132
+ new Promise<void>((resolve) => {
133
+ this.server = this.expressApp.listen(
134
+ this.environment.port,
135
+ this.environment.host,
136
+ () => {
137
+ debugLog(
138
+ this.environment.debug,
139
+ 'log',
140
+ `[ ${engine.translate(
141
+ SuiteCoreComponentId,
142
+ SuiteCoreStringKey.Common_Ready,
143
+ )} ] http://${this.environment.host}:${this.environment.port}`,
144
+ );
145
+ resolve();
146
+ },
147
+ ) as ServerWithOptionalClose;
148
+ }),
149
+ );
150
+
151
+ if (this.environment.httpsDevCertRoot) {
152
+ try {
153
+ const certRoot = normalize(this.environment.httpsDevCertRoot);
154
+ if (!isAbsolute(certRoot) || certRoot.includes('..')) {
155
+ throw new TranslatableSuiteError(
156
+ SuiteCoreStringKey.Error_InvalidCertificatePathMustBeAbsolute,
157
+ );
158
+ }
159
+ const certPath = normalize(resolve(certRoot + '.pem'));
160
+ const keyPath = normalize(resolve(certRoot + '-key.pem'));
161
+ if (certPath.includes('..') || keyPath.includes('..')) {
162
+ throw new TranslatableSuiteError(
163
+ SuiteCoreStringKey.Error_InvalidCertificatePathAfterResolution,
164
+ );
165
+ }
166
+ const options = {
167
+ // amazonq-ignore-next-line fixed above
168
+ key: readFileSync(keyPath),
169
+ // amazonq-ignore-next-line fixed above
170
+ cert: readFileSync(certPath),
171
+ };
172
+
173
+ serversReady.push(
174
+ new Promise<void>((resolve) => {
175
+ createServer(options, this.expressApp).listen(
176
+ this.environment.httpsDevPort,
177
+ () => {
178
+ console.log(
179
+ `[ ${engine.translate(
180
+ SuiteCoreComponentId,
181
+ SuiteCoreStringKey.Common_Ready,
182
+ )} ] https://${this.environment.host}:${
183
+ this.environment.httpsDevPort
184
+ }`,
185
+ );
186
+ resolve();
187
+ },
188
+ );
189
+ }),
190
+ );
191
+ } catch (err) {
192
+ console.error('Failed to start HTTPS server:', err);
193
+ }
194
+ }
195
+
196
+ await Promise.all(serversReady);
197
+ this._ready = true;
198
+ } catch (err) {
199
+ console.error(
200
+ engine.translate(
201
+ SuiteCoreComponentId,
202
+ SuiteCoreStringKey.Error_FailedToStartApplication,
203
+ ),
204
+ err,
205
+ );
206
+ if (process.env['NODE_ENV'] === 'test') {
207
+ throw err;
208
+ }
209
+ process.exit(1);
210
+ }
211
+ }
212
+
213
+ public override async stop(): Promise<void> {
214
+ const engine = getSuiteCoreI18nEngine();
215
+ if (this.server) {
216
+ debugLog(
217
+ this.environment.debug,
218
+ 'log',
219
+ `[ ${engine.translate(
220
+ SuiteCoreComponentId,
221
+ SuiteCoreStringKey.Common_Stopping,
222
+ )} ] ${engine.translate(
223
+ SuiteCoreComponentId,
224
+ SuiteCoreStringKey.Common_ApplicationAndDatabase,
225
+ )}`,
226
+ );
227
+ await new Promise<void>((resolve, reject) => {
228
+ this.server!.closeAllConnections?.();
229
+ this.server!.close((err) => {
230
+ if (err) {
231
+ reject(err);
232
+ } else {
233
+ resolve();
234
+ }
235
+ });
236
+ });
237
+ this.server = null;
238
+ }
239
+
240
+ await super.stop();
241
+ this._ready = false;
242
+ debugLog(
243
+ this.environment.debug,
244
+ 'log',
245
+ `[ ${engine.translate(
246
+ SuiteCoreComponentId,
247
+ SuiteCoreStringKey.Common_Stopped,
248
+ )} ] ${engine.translate(
249
+ SuiteCoreComponentId,
250
+ SuiteCoreStringKey.Common_ApplicationAndDatabase,
251
+ )}`,
252
+ );
253
+ }
254
+ }
@@ -0,0 +1,336 @@
1
+ import { MemberType } from '@digitaldefiance/ecies-lib';
2
+ import {
3
+ Constants as ApiConstants,
4
+ Member as BackendMember,
5
+ } from '@digitaldefiance/node-ecies-lib';
6
+ import {
7
+ BackupCodeString,
8
+ IBackupCode,
9
+ InvalidBackupCodeError,
10
+ PrivateKeyRequiredError,
11
+ } from '@digitaldefiance/suite-core-lib';
12
+ import * as argon2 from 'argon2';
13
+ import { createHmac, randomBytes, timingSafeEqual } from 'crypto';
14
+ import { Constants } from './constants';
15
+ import { InvalidBackupCodeVersionError } from './errors/invalid-backup-code-version';
16
+ import { IConstants } from './interfaces';
17
+ import { SymmetricService } from './services/symmetric';
18
+
19
+ /**
20
+ * Class representing a backup code string with associated operations.
21
+ *
22
+ * v1 scheme:
23
+ * - Code: 32 lowercase alphanumerics (a–z0–9), displayed as 8 groups of 4: xxxx-xxxx-xxxx-xxxx-xxxx-xxxx-xxxx-xxxx
24
+ * - Checksum/tag: HKDF-SHA256(codeUtf8, salt, "backup-checksum") → 32 bytes (stored as hex)
25
+ * - KDF for encryption key: Argon2id(codeUtf8, salt) → 32 bytes
26
+ * - Encryption: SymmetricService AEAD (encryptedData must embed IV + authTag + ciphertext)
27
+ * - Wrapping: AEAD blob wrapped with system user's asymmetric key (ECIES)
28
+ */
29
+ export class BackupCode extends BackupCodeString {
30
+ /** Current backup code scheme version implemented by this service. */
31
+ public static readonly BackupCodeVersion = '1.0.0';
32
+ // Centralized Argon2id parameters (tunable)
33
+ private static readonly Argon2Params = {
34
+ type: argon2.argon2id,
35
+ hashLength: 32, // derive AES-256 key
36
+ timeCost: 3,
37
+ memoryCost: 65536, // 64 MiB
38
+ parallelism: 1,
39
+ raw: true as const,
40
+ } as const;
41
+
42
+ constructor(code: string) {
43
+ super(code);
44
+ }
45
+
46
+ /**
47
+ * Generate the configured number of backup codes.
48
+ * Note: If generation alphabet/length is controlled elsewhere, prefer that path.
49
+ */
50
+ public static override generateBackupCodes(
51
+ constants: IConstants = Constants,
52
+ ): Array<BackupCode> {
53
+ const codes: Array<BackupCode> = [];
54
+ for (let i = 0; i < constants.BACKUP_CODES.Count; i++) {
55
+ codes.push(new BackupCode(BackupCode.generateBackupCode()));
56
+ }
57
+ return codes;
58
+ }
59
+
60
+ /**
61
+ * HKDF-Extract-and-Expand using HMAC-SHA-256.
62
+ *
63
+ * PRK = HMAC(salt, ikm)
64
+ * T(0) = empty
65
+ * T(i) = HMAC(PRK, T(i-1) || info || i)
66
+ * OKM = first 'length' bytes of T(1) || T(2) || ...
67
+ */
68
+ public static hkdfSha256(
69
+ ikm: Buffer,
70
+ salt: Buffer,
71
+ info: Buffer,
72
+ length: number,
73
+ ): Buffer {
74
+ if (length === 0) {
75
+ return Buffer.alloc(0);
76
+ }
77
+
78
+ // HKDF-Extract: PRK = HMAC-Hash(salt, IKM)
79
+ // If salt is empty, use a string of HashLen zeros
80
+ const actualSalt = salt.length === 0 ? Buffer.alloc(32, 0) : salt;
81
+ const prk = createHmac('sha256', actualSalt).update(ikm).digest();
82
+
83
+ // HKDF-Expand
84
+ const blocks: Buffer[] = [];
85
+ let prev = Buffer.alloc(0);
86
+ const n = Math.ceil(length / 32);
87
+
88
+ for (let i = 1; i <= n; i++) {
89
+ const hmac = createHmac('sha256', prk);
90
+ hmac.update(prev);
91
+ hmac.update(info);
92
+ hmac.update(Buffer.from([i]));
93
+ prev = Buffer.from(hmac.digest());
94
+ blocks.push(prev);
95
+ }
96
+
97
+ return Buffer.concat(blocks).subarray(0, length);
98
+ }
99
+
100
+ /**
101
+ * v1: Derive a 32-byte encryption key from a normalized backup code using Argon2id and the per-code salt.
102
+ * Uses UTF-8 bytes of the normalized code (not hex).
103
+ */
104
+ public static async getBackupKeyV1(
105
+ checksumSaltHex: string,
106
+ normalizedCode: string,
107
+ constants: IConstants = Constants,
108
+ ): Promise<Buffer> {
109
+ if (!constants.BACKUP_CODES.NormalizedHexRegex.test(normalizedCode)) {
110
+ throw new InvalidBackupCodeError();
111
+ }
112
+ const codeBytes = Buffer.from(normalizedCode, 'utf8');
113
+ const checksumSalt = Buffer.from(checksumSaltHex, 'hex');
114
+ try {
115
+ const key = (await argon2.hash(codeBytes, {
116
+ ...BackupCode.Argon2Params,
117
+ salt: checksumSalt,
118
+ })) as unknown as Buffer;
119
+ return key; // 32-byte Buffer
120
+ } finally {
121
+ codeBytes.fill(0);
122
+ }
123
+ }
124
+
125
+ /**
126
+ * v1: Compute a 32-byte checksum/tag for a normalized code using HKDF-SHA256(codeUtf8, salt, "backup-checksum").
127
+ */
128
+ private static computeChecksumV1(
129
+ normalizedCode: string,
130
+ checksumSalt: Buffer,
131
+ ): Buffer {
132
+ const codeBytes = Buffer.from(normalizedCode, 'utf8');
133
+ try {
134
+ return BackupCode.hkdfSha256(
135
+ codeBytes,
136
+ checksumSalt,
137
+ Buffer.from('backup-checksum'),
138
+ 32,
139
+ );
140
+ } finally {
141
+ codeBytes.fill(0);
142
+ }
143
+ }
144
+
145
+ public async encrypt(
146
+ backupUser: BackendMember,
147
+ systemUser: BackendMember,
148
+ constants: IConstants = Constants,
149
+ ): Promise<IBackupCode> {
150
+ if (!backupUser.hasPrivateKey) {
151
+ throw new PrivateKeyRequiredError();
152
+ }
153
+ if (systemUser.type !== MemberType.System) {
154
+ throw new Error('System user must be of MemberType.System');
155
+ }
156
+ const raw = this.value ?? '';
157
+ const normalized = BackupCode.normalizeCode(raw);
158
+ if (
159
+ !(
160
+ constants.BACKUP_CODES.DisplayRegex.test(raw) ||
161
+ constants.BACKUP_CODES.NormalizedHexRegex.test(normalized)
162
+ )
163
+ ) {
164
+ throw new InvalidBackupCodeError();
165
+ }
166
+
167
+ const checksumSalt = randomBytes(ApiConstants.PBKDF2.SALT_BYTES);
168
+ const checksumBuf = BackupCode.computeChecksumV1(normalized, checksumSalt);
169
+ const encryptionKey = await BackupCode.getBackupKeyV1(
170
+ checksumSalt.toString('hex'),
171
+ normalized,
172
+ );
173
+
174
+ try {
175
+ const sealed = SymmetricService.encryptBuffer(
176
+ Buffer.from(backupUser.privateKey!.value),
177
+ encryptionKey,
178
+ );
179
+ const wrappedEncryptedPrivateKey = systemUser
180
+ .encryptData(sealed.encryptedData)
181
+ .toString('hex');
182
+
183
+ return {
184
+ version: BackupCode.BackupCodeVersion,
185
+ checksumSalt: checksumSalt.toString('hex'),
186
+ checksum: checksumBuf.toString('hex'),
187
+ encrypted: wrappedEncryptedPrivateKey,
188
+ } as IBackupCode;
189
+ } finally {
190
+ encryptionKey.fill(0);
191
+ checksumBuf.fill(0);
192
+ }
193
+ }
194
+
195
+ /**
196
+ * v1: Encrypt and wrap backup codes for a user.
197
+ * - Validates code format (display or normalized)
198
+ * - Computes HKDF checksum/tag
199
+ * - Derives Argon2id encryption key (32 bytes) from UTF-8 code
200
+ * - Encrypts the private key with AEAD and wraps with system user
201
+ */
202
+ public static async encryptBackupCodesV1(
203
+ backupUser: BackendMember,
204
+ systemUser: BackendMember,
205
+ codes: Array<BackupCode>,
206
+ ): Promise<Array<IBackupCode>> {
207
+ const encryptedCodes: Array<IBackupCode> = [];
208
+ for (const code of codes) {
209
+ encryptedCodes.push(await code.encrypt(backupUser, systemUser));
210
+ }
211
+ return encryptedCodes;
212
+ }
213
+
214
+ /** Delegate to current version. */
215
+ public static encryptBackupCodes(
216
+ backupUser: BackendMember,
217
+ systemUser: BackendMember,
218
+ codes: Array<BackupCode>,
219
+ ): Promise<Array<IBackupCode>> {
220
+ return BackupCode.encryptBackupCodesV1(backupUser, systemUser, codes);
221
+ }
222
+
223
+ /**
224
+ * v1: Validate whether a backup code exists (unused) in the provided collection.
225
+ * Uses constant-time comparison of binary checksums (codeUtf8 + salt).
226
+ */
227
+ public static validateBackupCodeV1(
228
+ encryptedBackupCodes: Array<IBackupCode>,
229
+ backupCode: string,
230
+ constants: IConstants = Constants,
231
+ ): boolean {
232
+ const normalizedCode = BackupCodeString.normalizeCode(backupCode);
233
+ if (!constants.BACKUP_CODES.NormalizedHexRegex.test(normalizedCode)) {
234
+ return false;
235
+ }
236
+ const codeBytes = Buffer.from(normalizedCode, 'utf8');
237
+ try {
238
+ for (const code of encryptedBackupCodes) {
239
+ if (code.version !== BackupCode.BackupCodeVersion) continue;
240
+ const checksumSalt = Buffer.from(code.checksumSalt, 'hex');
241
+ const expected = BackupCode.hkdfSha256(
242
+ codeBytes,
243
+ checksumSalt,
244
+ Buffer.from('backup-checksum'),
245
+ 32,
246
+ );
247
+ if (
248
+ code.checksum.length === expected.length * 2 &&
249
+ timingSafeEqual(Buffer.from(code.checksum, 'hex'), expected)
250
+ ) {
251
+ return true;
252
+ }
253
+ }
254
+ return false;
255
+ } finally {
256
+ codeBytes.fill(0);
257
+ }
258
+ }
259
+
260
+ /**
261
+ * Validate a backup code against any supported version present in the collection.
262
+ */
263
+ public static validateBackupCode(
264
+ encryptedBackupCodes: Array<IBackupCode>,
265
+ backupCode: string,
266
+ constants: IConstants = Constants,
267
+ ): boolean {
268
+ const normalizedCode = BackupCodeString.normalizeCode(backupCode);
269
+ if (!constants.BACKUP_CODES.NormalizedHexRegex.test(normalizedCode)) {
270
+ return false;
271
+ }
272
+ if (
273
+ encryptedBackupCodes.some(
274
+ (c) => c.version === BackupCode.BackupCodeVersion,
275
+ )
276
+ ) {
277
+ return this.validateBackupCodeV1(
278
+ encryptedBackupCodes.filter(
279
+ (c) => c.version === BackupCode.BackupCodeVersion,
280
+ ),
281
+ normalizedCode,
282
+ );
283
+ }
284
+ return false;
285
+ }
286
+
287
+ /**
288
+ * Detect the version by matching checksum against stored codes; returns the matched version.
289
+ */
290
+ public static detectBackupCodeVersion(
291
+ encryptedBackupCodes: Array<IBackupCode>,
292
+ backupCode: string,
293
+ constants: IConstants = Constants,
294
+ ): string {
295
+ const normalizedCode = BackupCodeString.normalizeCode(backupCode);
296
+ if (!constants.BACKUP_CODES.NormalizedHexRegex.test(normalizedCode)) {
297
+ throw new InvalidBackupCodeError();
298
+ }
299
+
300
+ const v1Set = encryptedBackupCodes.filter(
301
+ (c) => c.version === BackupCode.BackupCodeVersion,
302
+ );
303
+ if (v1Set.length) {
304
+ const codeBytes = Buffer.from(normalizedCode, 'utf8');
305
+ try {
306
+ for (const c of v1Set) {
307
+ const checksumSalt = Buffer.from(c.checksumSalt, 'hex');
308
+ const expected = BackupCode.hkdfSha256(
309
+ codeBytes,
310
+ checksumSalt,
311
+ Buffer.from('backup-checksum'),
312
+ 32,
313
+ );
314
+ if (
315
+ c.checksum.length === expected.length * 2 &&
316
+ timingSafeEqual(Buffer.from(c.checksum, 'hex'), expected)
317
+ ) {
318
+ return c.version;
319
+ }
320
+ }
321
+ } finally {
322
+ // zeroize
323
+ codeBytes.fill(0);
324
+ }
325
+ }
326
+
327
+ const versionsInSet = new Set(encryptedBackupCodes.map((c) => c.version));
328
+ if (
329
+ versionsInSet.size > 0 &&
330
+ !versionsInSet.has(BackupCode.BackupCodeVersion)
331
+ ) {
332
+ throw new InvalidBackupCodeVersionError([...versionsInSet][0]);
333
+ }
334
+ throw new InvalidBackupCodeError();
335
+ }
336
+ }
@@ -0,0 +1,69 @@
1
+ import { ECIES as ECIESDefaults } from '@digitaldefiance/ecies-lib';
2
+ import { createConstants } from '@digitaldefiance/suite-core-lib';
3
+ import { IFECConsts } from './interfaces';
4
+ import { IChecksumConsts } from './interfaces/checksum-consts';
5
+ import { IConstants } from './interfaces/constants';
6
+ import { IJwtConsts } from './interfaces/jwt-consts';
7
+
8
+ /**
9
+ * Constants for checksum operations
10
+ * These values are critical for data integrity and MUST NOT be changed
11
+ * in an already established system as it will break all existing checksums.
12
+ */
13
+ export const CHECKSUM: IChecksumConsts = Object.freeze({
14
+ /** Default hash bits for SHA3 */
15
+ SHA3_DEFAULT_HASH_BITS: 512 as const,
16
+
17
+ /** Length of a SHA3 checksum buffer in bytes */
18
+ SHA3_BUFFER_LENGTH: 64 as const,
19
+
20
+ /** algorithm to use for checksum */
21
+ ALGORITHM: 'sha3-512' as const,
22
+
23
+ /** encoding to use for checksum */
24
+ ENCODING: 'hex' as const,
25
+ } as const);
26
+
27
+ export const JWT: IJwtConsts = {
28
+ /**
29
+ * Algorithm to use for JWT
30
+ */
31
+ ALGORITHM: 'HS256' as const,
32
+
33
+ /**
34
+ * The expiration time for a JWT token in seconds
35
+ */
36
+ EXPIRATION_SEC: 86400 as const,
37
+ } as const;
38
+
39
+ export const FEC: IFECConsts = {
40
+ /**
41
+ * Maximum size of a single shard
42
+ */
43
+ MAX_SHARD_SIZE: 1048576 as const,
44
+ } as const;
45
+
46
+ // use defaults from ecies-lib
47
+ export const ECIES = Object.freeze(ECIESDefaults);
48
+
49
+ export const createExpressConstants = (
50
+ siteDomain: string,
51
+ overrides?: Partial<IConstants>,
52
+ ): IConstants => {
53
+ return Object.freeze({
54
+ ...createConstants(siteDomain, overrides),
55
+ CHECKSUM: CHECKSUM,
56
+ JWT: JWT,
57
+ FEC: FEC,
58
+ ECIES: ECIES,
59
+ } as const);
60
+ };
61
+
62
+ export const Constants: IConstants = createExpressConstants('localhost');
63
+
64
+ if (
65
+ CHECKSUM.SHA3_BUFFER_LENGTH !== CHECKSUM.SHA3_DEFAULT_HASH_BITS / 8 ||
66
+ CHECKSUM.SHA3_BUFFER_LENGTH !== CHECKSUM.SHA3_DEFAULT_HASH_BITS / 8
67
+ ) {
68
+ throw new Error('Invalid checksum constants');
69
+ }