@vandenberghinc/volt 1.2.4 → 1.2.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (489) hide show
  1. package/frontend/assets/admin/admin.png +0 -0
  2. package/frontend/assets/admin/password.webp +0 -0
  3. package/frontend/assets/icons/arrow.v1.webp +0 -0
  4. package/frontend/assets/icons/copy.webp +0 -0
  5. package/frontend/assets/payments/arrow.long.webp +0 -0
  6. package/frontend/assets/payments/arrow.long2.webp +0 -0
  7. package/frontend/assets/payments/cancelled.webp +0 -0
  8. package/frontend/assets/payments/check.sign.webp +0 -0
  9. package/frontend/assets/payments/check.webp +0 -0
  10. package/frontend/assets/payments/close.webp +0 -0
  11. package/frontend/assets/payments/error.webp +0 -0
  12. package/frontend/assets/payments/exclamation.webp +0 -0
  13. package/frontend/assets/payments/minus.webp +0 -0
  14. package/frontend/assets/payments/party.webp +0 -0
  15. package/frontend/assets/payments/plus.webp +0 -0
  16. package/frontend/assets/payments/shopping_cart.webp +0 -0
  17. package/frontend/assets/payments/trash.webp +0 -0
  18. package/package.json +6 -2
  19. package/.libris/config.json +0 -82
  20. package/backend/dist/cjs/backend/src/blacklist.d.ts +0 -12
  21. package/backend/dist/cjs/backend/src/blacklist.js +0 -78
  22. package/backend/dist/cjs/backend/src/cli.d.ts +0 -2
  23. package/backend/dist/cjs/backend/src/cli.js +0 -198
  24. package/backend/dist/cjs/backend/src/database/collection.d.ts +0 -1765
  25. package/backend/dist/cjs/backend/src/database/collection.js +0 -3301
  26. package/backend/dist/cjs/backend/src/database/database.d.ts +0 -92
  27. package/backend/dist/cjs/backend/src/database/database.js +0 -170
  28. package/backend/dist/cjs/backend/src/database/document.d.ts +0 -1
  29. package/backend/dist/cjs/backend/src/database/document.js +0 -15
  30. package/backend/dist/cjs/backend/src/database/filters/filters.d.ts +0 -6
  31. package/backend/dist/cjs/backend/src/database/filters/filters.js +0 -15
  32. package/backend/dist/cjs/backend/src/database/filters/strict_filter.d.ts +0 -223
  33. package/backend/dist/cjs/backend/src/database/filters/strict_filter.js +0 -15
  34. package/backend/dist/cjs/backend/src/database/filters/strict_filter_test.d.ts +0 -1
  35. package/backend/dist/cjs/backend/src/database/filters/strict_filter_test.js +0 -443
  36. package/backend/dist/cjs/backend/src/database/filters/strict_filter_test_v0.d.ts +0 -1
  37. package/backend/dist/cjs/backend/src/database/filters/strict_filter_test_v0.js +0 -15
  38. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v0.d.ts +0 -50
  39. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v0.js +0 -15
  40. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v1.d.ts +0 -76
  41. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v1.js +0 -15
  42. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v2.d.ts +0 -75
  43. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v2.js +0 -15
  44. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v3.d.ts +0 -219
  45. package/backend/dist/cjs/backend/src/database/filters/strict_filter_v3.js +0 -15
  46. package/backend/dist/cjs/backend/src/database/filters/strict_update_filter.d.ts +0 -165
  47. package/backend/dist/cjs/backend/src/database/filters/strict_update_filter.js +0 -15
  48. package/backend/dist/cjs/backend/src/database/filters/strict_update_filter_test.d.ts +0 -5
  49. package/backend/dist/cjs/backend/src/database/filters/strict_update_filter_test.js +0 -355
  50. package/backend/dist/cjs/backend/src/database/flatten.d.ts +0 -78
  51. package/backend/dist/cjs/backend/src/database/flatten.js +0 -53
  52. package/backend/dist/cjs/backend/src/database/flatten_test.d.ts +0 -1
  53. package/backend/dist/cjs/backend/src/database/flatten_test.js +0 -175
  54. package/backend/dist/cjs/backend/src/database/quota/quoata_v2.d.ts +0 -533
  55. package/backend/dist/cjs/backend/src/database/quota/quoata_v2.js +0 -1046
  56. package/backend/dist/cjs/backend/src/database/quota/quota.d.ts +0 -551
  57. package/backend/dist/cjs/backend/src/database/quota/quota.js +0 -1108
  58. package/backend/dist/cjs/backend/src/database/quota/quota_v1.d.ts +0 -534
  59. package/backend/dist/cjs/backend/src/database/quota/quota_v1.js +0 -1087
  60. package/backend/dist/cjs/backend/src/database/quota/safe_int.d.ts +0 -412
  61. package/backend/dist/cjs/backend/src/database/quota/safe_int.js +0 -745
  62. package/backend/dist/cjs/backend/src/endpoint.d.ts +0 -346
  63. package/backend/dist/cjs/backend/src/endpoint.js +0 -468
  64. package/backend/dist/cjs/backend/src/errors/index.d.ts +0 -7
  65. package/backend/dist/cjs/backend/src/errors/index.js +0 -25
  66. package/backend/dist/cjs/backend/src/errors/internal_external.d.ts +0 -52
  67. package/backend/dist/cjs/backend/src/errors/internal_external.js +0 -95
  68. package/backend/dist/cjs/backend/src/errors/invalid_usage_error.d.ts +0 -41
  69. package/backend/dist/cjs/backend/src/errors/invalid_usage_error.js +0 -47
  70. package/backend/dist/cjs/backend/src/errors/system_error.d.ts +0 -261
  71. package/backend/dist/cjs/backend/src/errors/system_error.js +0 -436
  72. package/backend/dist/cjs/backend/src/events.d.ts +0 -97
  73. package/backend/dist/cjs/backend/src/events.js +0 -15
  74. package/backend/dist/cjs/backend/src/frontend.d.ts +0 -11
  75. package/backend/dist/cjs/backend/src/frontend.js +0 -37
  76. package/backend/dist/cjs/backend/src/image_endpoint.d.ts +0 -44
  77. package/backend/dist/cjs/backend/src/image_endpoint.js +0 -185
  78. package/backend/dist/cjs/backend/src/index.d.ts +0 -23
  79. package/backend/dist/cjs/backend/src/index.js +0 -70
  80. package/backend/dist/cjs/backend/src/logger.d.ts +0 -5
  81. package/backend/dist/cjs/backend/src/logger.js +0 -15
  82. package/backend/dist/cjs/backend/src/meta.d.ts +0 -112
  83. package/backend/dist/cjs/backend/src/meta.js +0 -181
  84. package/backend/dist/cjs/backend/src/payments/paddle.d.ts +0 -329
  85. package/backend/dist/cjs/backend/src/payments/paddle.js +0 -1996
  86. package/backend/dist/cjs/backend/src/payments/stripe/checkout.d.ts +0 -113
  87. package/backend/dist/cjs/backend/src/payments/stripe/checkout.js +0 -295
  88. package/backend/dist/cjs/backend/src/payments/stripe/customers.d.ts +0 -17
  89. package/backend/dist/cjs/backend/src/payments/stripe/customers.js +0 -164
  90. package/backend/dist/cjs/backend/src/payments/stripe/error.d.ts +0 -74
  91. package/backend/dist/cjs/backend/src/payments/stripe/error.js +0 -64
  92. package/backend/dist/cjs/backend/src/payments/stripe/events.d.ts +0 -155
  93. package/backend/dist/cjs/backend/src/payments/stripe/events.js +0 -15
  94. package/backend/dist/cjs/backend/src/payments/stripe/meters.d.ts +0 -105
  95. package/backend/dist/cjs/backend/src/payments/stripe/meters.js +0 -230
  96. package/backend/dist/cjs/backend/src/payments/stripe/payment_methods.d.ts +0 -58
  97. package/backend/dist/cjs/backend/src/payments/stripe/payment_methods.js +0 -109
  98. package/backend/dist/cjs/backend/src/payments/stripe/products.d.ts +0 -519
  99. package/backend/dist/cjs/backend/src/payments/stripe/products.js +0 -650
  100. package/backend/dist/cjs/backend/src/payments/stripe/stripe.d.ts +0 -215
  101. package/backend/dist/cjs/backend/src/payments/stripe/stripe.js +0 -468
  102. package/backend/dist/cjs/backend/src/payments/stripe/subscriptions.d.ts +0 -172
  103. package/backend/dist/cjs/backend/src/payments/stripe/subscriptions.js +0 -557
  104. package/backend/dist/cjs/backend/src/payments/stripe/utils.d.ts +0 -63
  105. package/backend/dist/cjs/backend/src/payments/stripe/utils.js +0 -118
  106. package/backend/dist/cjs/backend/src/payments/stripe/webhooks.d.ts +0 -105
  107. package/backend/dist/cjs/backend/src/payments/stripe/webhooks.js +0 -627
  108. package/backend/dist/cjs/backend/src/plugins/browser.d.ts +0 -1
  109. package/backend/dist/cjs/backend/src/plugins/browser.js +0 -15
  110. package/backend/dist/cjs/backend/src/plugins/communication.d.ts +0 -70
  111. package/backend/dist/cjs/backend/src/plugins/communication.js +0 -196
  112. package/backend/dist/cjs/backend/src/plugins/mail/mail.d.ts +0 -255
  113. package/backend/dist/cjs/backend/src/plugins/mail/mail.js +0 -381
  114. package/backend/dist/cjs/backend/src/plugins/mail/ui.d.ts +0 -297
  115. package/backend/dist/cjs/backend/src/plugins/mail/ui.js +0 -1370
  116. package/backend/dist/cjs/backend/src/plugins/pdf.d.ts +0 -1
  117. package/backend/dist/cjs/backend/src/plugins/pdf.js +0 -1456
  118. package/backend/dist/cjs/backend/src/plugins/thread_monitor.d.ts +0 -18
  119. package/backend/dist/cjs/backend/src/plugins/thread_monitor.js +0 -116
  120. package/backend/dist/cjs/backend/src/rate_limit.d.ts +0 -148
  121. package/backend/dist/cjs/backend/src/rate_limit.js +0 -543
  122. package/backend/dist/cjs/backend/src/route.d.ts +0 -39
  123. package/backend/dist/cjs/backend/src/route.js +0 -172
  124. package/backend/dist/cjs/backend/src/server.d.ts +0 -502
  125. package/backend/dist/cjs/backend/src/server.js +0 -1713
  126. package/backend/dist/cjs/backend/src/server.old.d.ts +0 -594
  127. package/backend/dist/cjs/backend/src/server.old.js +0 -2058
  128. package/backend/dist/cjs/backend/src/splash_screen.d.ts +0 -93
  129. package/backend/dist/cjs/backend/src/splash_screen.js +0 -119
  130. package/backend/dist/cjs/backend/src/status.d.ts +0 -89
  131. package/backend/dist/cjs/backend/src/status.js +0 -211
  132. package/backend/dist/cjs/backend/src/stream.d.ts +0 -494
  133. package/backend/dist/cjs/backend/src/stream.js +0 -1370
  134. package/backend/dist/cjs/backend/src/users.d.ts +0 -926
  135. package/backend/dist/cjs/backend/src/users.js +0 -2223
  136. package/backend/dist/cjs/backend/src/utils.d.ts +0 -22
  137. package/backend/dist/cjs/backend/src/utils.js +0 -626
  138. package/backend/dist/cjs/backend/src/view.d.ts +0 -115
  139. package/backend/dist/cjs/backend/src/view.js +0 -519
  140. package/backend/dist/cjs/backend/src/vinc.d.ts +0 -6
  141. package/backend/dist/cjs/backend/src/vinc.js +0 -40
  142. package/backend/dist/cjs/backend/src/volt.d.ts +0 -24
  143. package/backend/dist/cjs/backend/src/volt.js +0 -72
  144. package/backend/dist/cjs/frontend/src/modules/request.d.ts +0 -70
  145. package/backend/dist/cjs/frontend/src/modules/request.js +0 -99
  146. package/backend/dist/cjs/package.json +0 -1
  147. package/backend/dist/esm/backend/src/blacklist.d.ts +0 -12
  148. package/backend/dist/esm/backend/src/blacklist.js +0 -52
  149. package/backend/dist/esm/backend/src/cli.d.ts +0 -2
  150. package/backend/dist/esm/backend/src/cli.js +0 -211
  151. package/backend/dist/esm/backend/src/database/collection.d.ts +0 -1765
  152. package/backend/dist/esm/backend/src/database/collection.js +0 -3779
  153. package/backend/dist/esm/backend/src/database/database.d.ts +0 -92
  154. package/backend/dist/esm/backend/src/database/database.js +0 -214
  155. package/backend/dist/esm/backend/src/database/document.d.ts +0 -1
  156. package/backend/dist/esm/backend/src/database/document.js +0 -558
  157. package/backend/dist/esm/backend/src/database/filters/filters.d.ts +0 -6
  158. package/backend/dist/esm/backend/src/database/filters/filters.js +0 -1
  159. package/backend/dist/esm/backend/src/database/filters/strict_filter.d.ts +0 -223
  160. package/backend/dist/esm/backend/src/database/filters/strict_filter.js +0 -3
  161. package/backend/dist/esm/backend/src/database/filters/strict_filter_test.d.ts +0 -1
  162. package/backend/dist/esm/backend/src/database/filters/strict_filter_test.js +0 -505
  163. package/backend/dist/esm/backend/src/database/filters/strict_filter_test_v0.d.ts +0 -1
  164. package/backend/dist/esm/backend/src/database/filters/strict_filter_test_v0.js +0 -712
  165. package/backend/dist/esm/backend/src/database/filters/strict_filter_v0.d.ts +0 -50
  166. package/backend/dist/esm/backend/src/database/filters/strict_filter_v0.js +0 -5
  167. package/backend/dist/esm/backend/src/database/filters/strict_filter_v1.d.ts +0 -76
  168. package/backend/dist/esm/backend/src/database/filters/strict_filter_v1.js +0 -44
  169. package/backend/dist/esm/backend/src/database/filters/strict_filter_v2.d.ts +0 -75
  170. package/backend/dist/esm/backend/src/database/filters/strict_filter_v2.js +0 -5
  171. package/backend/dist/esm/backend/src/database/filters/strict_filter_v3.d.ts +0 -219
  172. package/backend/dist/esm/backend/src/database/filters/strict_filter_v3.js +0 -1
  173. package/backend/dist/esm/backend/src/database/filters/strict_update_filter.d.ts +0 -165
  174. package/backend/dist/esm/backend/src/database/filters/strict_update_filter.js +0 -5
  175. package/backend/dist/esm/backend/src/database/filters/strict_update_filter_test.d.ts +0 -5
  176. package/backend/dist/esm/backend/src/database/filters/strict_update_filter_test.js +0 -415
  177. package/backend/dist/esm/backend/src/database/flatten.d.ts +0 -78
  178. package/backend/dist/esm/backend/src/database/flatten.js +0 -22
  179. package/backend/dist/esm/backend/src/database/flatten_test.d.ts +0 -1
  180. package/backend/dist/esm/backend/src/database/flatten_test.js +0 -174
  181. package/backend/dist/esm/backend/src/database/quota/quoata_v2.d.ts +0 -533
  182. package/backend/dist/esm/backend/src/database/quota/quoata_v2.js +0 -1155
  183. package/backend/dist/esm/backend/src/database/quota/quota.d.ts +0 -551
  184. package/backend/dist/esm/backend/src/database/quota/quota.js +0 -1219
  185. package/backend/dist/esm/backend/src/database/quota/quota_v1.d.ts +0 -534
  186. package/backend/dist/esm/backend/src/database/quota/quota_v1.js +0 -1242
  187. package/backend/dist/esm/backend/src/database/quota/safe_int.d.ts +0 -412
  188. package/backend/dist/esm/backend/src/database/quota/safe_int.js +0 -810
  189. package/backend/dist/esm/backend/src/endpoint.d.ts +0 -346
  190. package/backend/dist/esm/backend/src/endpoint.js +0 -479
  191. package/backend/dist/esm/backend/src/errors/index.d.ts +0 -7
  192. package/backend/dist/esm/backend/src/errors/index.js +0 -7
  193. package/backend/dist/esm/backend/src/errors/internal_external.d.ts +0 -52
  194. package/backend/dist/esm/backend/src/errors/internal_external.js +0 -86
  195. package/backend/dist/esm/backend/src/errors/invalid_usage_error.d.ts +0 -41
  196. package/backend/dist/esm/backend/src/errors/invalid_usage_error.js +0 -33
  197. package/backend/dist/esm/backend/src/errors/system_error.d.ts +0 -261
  198. package/backend/dist/esm/backend/src/errors/system_error.js +0 -444
  199. package/backend/dist/esm/backend/src/events.d.ts +0 -97
  200. package/backend/dist/esm/backend/src/events.js +0 -5
  201. package/backend/dist/esm/backend/src/frontend.d.ts +0 -11
  202. package/backend/dist/esm/backend/src/frontend.js +0 -12
  203. package/backend/dist/esm/backend/src/image_endpoint.d.ts +0 -44
  204. package/backend/dist/esm/backend/src/image_endpoint.js +0 -196
  205. package/backend/dist/esm/backend/src/index.d.ts +0 -23
  206. package/backend/dist/esm/backend/src/index.js +0 -26
  207. package/backend/dist/esm/backend/src/logger.d.ts +0 -5
  208. package/backend/dist/esm/backend/src/logger.js +0 -8
  209. package/backend/dist/esm/backend/src/meta.d.ts +0 -112
  210. package/backend/dist/esm/backend/src/meta.js +0 -152
  211. package/backend/dist/esm/backend/src/payments/paddle.d.ts +0 -329
  212. package/backend/dist/esm/backend/src/payments/paddle.js +0 -2276
  213. package/backend/dist/esm/backend/src/payments/stripe/checkout.d.ts +0 -113
  214. package/backend/dist/esm/backend/src/payments/stripe/checkout.js +0 -356
  215. package/backend/dist/esm/backend/src/payments/stripe/customers.d.ts +0 -17
  216. package/backend/dist/esm/backend/src/payments/stripe/customers.js +0 -193
  217. package/backend/dist/esm/backend/src/payments/stripe/error.d.ts +0 -74
  218. package/backend/dist/esm/backend/src/payments/stripe/error.js +0 -51
  219. package/backend/dist/esm/backend/src/payments/stripe/events.d.ts +0 -155
  220. package/backend/dist/esm/backend/src/payments/stripe/events.js +0 -5
  221. package/backend/dist/esm/backend/src/payments/stripe/meters.d.ts +0 -105
  222. package/backend/dist/esm/backend/src/payments/stripe/meters.js +0 -318
  223. package/backend/dist/esm/backend/src/payments/stripe/payment_methods.d.ts +0 -58
  224. package/backend/dist/esm/backend/src/payments/stripe/payment_methods.js +0 -135
  225. package/backend/dist/esm/backend/src/payments/stripe/products.d.ts +0 -519
  226. package/backend/dist/esm/backend/src/payments/stripe/products.js +0 -896
  227. package/backend/dist/esm/backend/src/payments/stripe/stripe.d.ts +0 -215
  228. package/backend/dist/esm/backend/src/payments/stripe/stripe.js +0 -464
  229. package/backend/dist/esm/backend/src/payments/stripe/subscriptions.d.ts +0 -172
  230. package/backend/dist/esm/backend/src/payments/stripe/subscriptions.js +0 -754
  231. package/backend/dist/esm/backend/src/payments/stripe/utils.d.ts +0 -63
  232. package/backend/dist/esm/backend/src/payments/stripe/utils.js +0 -131
  233. package/backend/dist/esm/backend/src/payments/stripe/webhooks.d.ts +0 -105
  234. package/backend/dist/esm/backend/src/payments/stripe/webhooks.js +0 -752
  235. package/backend/dist/esm/backend/src/plugins/browser.d.ts +0 -1
  236. package/backend/dist/esm/backend/src/plugins/browser.js +0 -170
  237. package/backend/dist/esm/backend/src/plugins/communication.d.ts +0 -70
  238. package/backend/dist/esm/backend/src/plugins/communication.js +0 -169
  239. package/backend/dist/esm/backend/src/plugins/mail/mail.d.ts +0 -255
  240. package/backend/dist/esm/backend/src/plugins/mail/mail.js +0 -396
  241. package/backend/dist/esm/backend/src/plugins/mail/ui.d.ts +0 -297
  242. package/backend/dist/esm/backend/src/plugins/mail/ui.js +0 -1400
  243. package/backend/dist/esm/backend/src/plugins/pdf.d.ts +0 -1
  244. package/backend/dist/esm/backend/src/plugins/pdf.js +0 -1694
  245. package/backend/dist/esm/backend/src/plugins/thread_monitor.d.ts +0 -18
  246. package/backend/dist/esm/backend/src/plugins/thread_monitor.js +0 -120
  247. package/backend/dist/esm/backend/src/rate_limit.d.ts +0 -148
  248. package/backend/dist/esm/backend/src/rate_limit.js +0 -667
  249. package/backend/dist/esm/backend/src/route.d.ts +0 -39
  250. package/backend/dist/esm/backend/src/route.js +0 -222
  251. package/backend/dist/esm/backend/src/server.d.ts +0 -502
  252. package/backend/dist/esm/backend/src/server.js +0 -2034
  253. package/backend/dist/esm/backend/src/server.old.d.ts +0 -594
  254. package/backend/dist/esm/backend/src/server.old.js +0 -2630
  255. package/backend/dist/esm/backend/src/splash_screen.d.ts +0 -93
  256. package/backend/dist/esm/backend/src/splash_screen.js +0 -156
  257. package/backend/dist/esm/backend/src/status.d.ts +0 -89
  258. package/backend/dist/esm/backend/src/status.js +0 -213
  259. package/backend/dist/esm/backend/src/stream.d.ts +0 -494
  260. package/backend/dist/esm/backend/src/stream.js +0 -1611
  261. package/backend/dist/esm/backend/src/users.d.ts +0 -926
  262. package/backend/dist/esm/backend/src/users.js +0 -2423
  263. package/backend/dist/esm/backend/src/utils.d.ts +0 -22
  264. package/backend/dist/esm/backend/src/utils.js +0 -463
  265. package/backend/dist/esm/backend/src/view.d.ts +0 -115
  266. package/backend/dist/esm/backend/src/view.js +0 -584
  267. package/backend/dist/esm/backend/src/vinc.d.ts +0 -6
  268. package/backend/dist/esm/backend/src/vinc.js +0 -6
  269. package/backend/dist/esm/backend/src/volt.d.ts +0 -24
  270. package/backend/dist/esm/backend/src/volt.js +0 -27
  271. package/backend/dist/esm/frontend/src/modules/request.d.ts +0 -70
  272. package/backend/dist/esm/frontend/src/modules/request.js +0 -117
  273. package/backend/old/file_watcher.ts +0 -359
  274. package/backend/old/request.deprc.js +0 -626
  275. package/backend/old/response.deprc.js +0 -354
  276. package/frontend/dist/backend/src/database/collection.d.ts +0 -1765
  277. package/frontend/dist/backend/src/database/collection.js +0 -3779
  278. package/frontend/dist/backend/src/database/database.d.ts +0 -92
  279. package/frontend/dist/backend/src/database/database.js +0 -214
  280. package/frontend/dist/backend/src/database/filters/filters.d.ts +0 -6
  281. package/frontend/dist/backend/src/database/filters/filters.js +0 -1
  282. package/frontend/dist/backend/src/database/filters/strict_filter.d.ts +0 -223
  283. package/frontend/dist/backend/src/database/filters/strict_filter.js +0 -3
  284. package/frontend/dist/backend/src/database/filters/strict_update_filter.d.ts +0 -165
  285. package/frontend/dist/backend/src/database/filters/strict_update_filter.js +0 -5
  286. package/frontend/dist/backend/src/database/flatten.d.ts +0 -78
  287. package/frontend/dist/backend/src/database/flatten.js +0 -22
  288. package/frontend/dist/backend/src/endpoint.d.ts +0 -346
  289. package/frontend/dist/backend/src/endpoint.js +0 -479
  290. package/frontend/dist/backend/src/errors/index.d.ts +0 -7
  291. package/frontend/dist/backend/src/errors/index.js +0 -7
  292. package/frontend/dist/backend/src/errors/internal_external.d.ts +0 -52
  293. package/frontend/dist/backend/src/errors/internal_external.js +0 -86
  294. package/frontend/dist/backend/src/errors/invalid_usage_error.d.ts +0 -41
  295. package/frontend/dist/backend/src/errors/invalid_usage_error.js +0 -33
  296. package/frontend/dist/backend/src/errors/system_error.d.ts +0 -261
  297. package/frontend/dist/backend/src/errors/system_error.js +0 -444
  298. package/frontend/dist/backend/src/events.d.ts +0 -97
  299. package/frontend/dist/backend/src/events.js +0 -5
  300. package/frontend/dist/backend/src/frontend.d.ts +0 -11
  301. package/frontend/dist/backend/src/frontend.js +0 -12
  302. package/frontend/dist/backend/src/image_endpoint.d.ts +0 -44
  303. package/frontend/dist/backend/src/image_endpoint.js +0 -196
  304. package/frontend/dist/backend/src/meta.d.ts +0 -112
  305. package/frontend/dist/backend/src/meta.js +0 -152
  306. package/frontend/dist/backend/src/payments/paddle.d.ts +0 -329
  307. package/frontend/dist/backend/src/payments/paddle.js +0 -2276
  308. package/frontend/dist/backend/src/payments/stripe/checkout.d.ts +0 -113
  309. package/frontend/dist/backend/src/payments/stripe/checkout.js +0 -356
  310. package/frontend/dist/backend/src/payments/stripe/customers.d.ts +0 -17
  311. package/frontend/dist/backend/src/payments/stripe/customers.js +0 -193
  312. package/frontend/dist/backend/src/payments/stripe/error.d.ts +0 -74
  313. package/frontend/dist/backend/src/payments/stripe/error.js +0 -51
  314. package/frontend/dist/backend/src/payments/stripe/events.d.ts +0 -155
  315. package/frontend/dist/backend/src/payments/stripe/events.js +0 -5
  316. package/frontend/dist/backend/src/payments/stripe/meters.d.ts +0 -105
  317. package/frontend/dist/backend/src/payments/stripe/meters.js +0 -318
  318. package/frontend/dist/backend/src/payments/stripe/payment_methods.d.ts +0 -58
  319. package/frontend/dist/backend/src/payments/stripe/payment_methods.js +0 -135
  320. package/frontend/dist/backend/src/payments/stripe/products.d.ts +0 -519
  321. package/frontend/dist/backend/src/payments/stripe/products.js +0 -896
  322. package/frontend/dist/backend/src/payments/stripe/stripe.d.ts +0 -215
  323. package/frontend/dist/backend/src/payments/stripe/stripe.js +0 -464
  324. package/frontend/dist/backend/src/payments/stripe/subscriptions.d.ts +0 -172
  325. package/frontend/dist/backend/src/payments/stripe/subscriptions.js +0 -754
  326. package/frontend/dist/backend/src/payments/stripe/utils.d.ts +0 -63
  327. package/frontend/dist/backend/src/payments/stripe/utils.js +0 -131
  328. package/frontend/dist/backend/src/payments/stripe/webhooks.d.ts +0 -105
  329. package/frontend/dist/backend/src/payments/stripe/webhooks.js +0 -752
  330. package/frontend/dist/backend/src/plugins/mail/mail.d.ts +0 -255
  331. package/frontend/dist/backend/src/plugins/mail/mail.js +0 -396
  332. package/frontend/dist/backend/src/plugins/mail/ui.d.ts +0 -297
  333. package/frontend/dist/backend/src/plugins/mail/ui.js +0 -1400
  334. package/frontend/dist/backend/src/rate_limit.d.ts +0 -148
  335. package/frontend/dist/backend/src/rate_limit.js +0 -667
  336. package/frontend/dist/backend/src/route.d.ts +0 -39
  337. package/frontend/dist/backend/src/route.js +0 -222
  338. package/frontend/dist/backend/src/server.d.ts +0 -502
  339. package/frontend/dist/backend/src/server.js +0 -2034
  340. package/frontend/dist/backend/src/splash_screen.d.ts +0 -93
  341. package/frontend/dist/backend/src/splash_screen.js +0 -156
  342. package/frontend/dist/backend/src/status.d.ts +0 -89
  343. package/frontend/dist/backend/src/status.js +0 -213
  344. package/frontend/dist/backend/src/stream.d.ts +0 -494
  345. package/frontend/dist/backend/src/stream.js +0 -1611
  346. package/frontend/dist/backend/src/users.d.ts +0 -926
  347. package/frontend/dist/backend/src/users.js +0 -2423
  348. package/frontend/dist/backend/src/utils.d.ts +0 -22
  349. package/frontend/dist/backend/src/utils.js +0 -463
  350. package/frontend/dist/backend/src/view.d.ts +0 -115
  351. package/frontend/dist/backend/src/view.js +0 -584
  352. package/frontend/dist/frontend/src/elements/base.d.ts +0 -3743
  353. package/frontend/dist/frontend/src/elements/base.js +0 -12151
  354. package/frontend/dist/frontend/src/elements/module.d.ts +0 -95
  355. package/frontend/dist/frontend/src/elements/module.js +0 -216
  356. package/frontend/dist/frontend/src/elements/register_element.d.ts +0 -3
  357. package/frontend/dist/frontend/src/elements/register_element.js +0 -22
  358. package/frontend/dist/frontend/src/elements/resize_query_manager.d.ts +0 -0
  359. package/frontend/dist/frontend/src/elements/resize_query_manager.js +0 -150
  360. package/frontend/dist/frontend/src/elements/types.d.ts +0 -52
  361. package/frontend/dist/frontend/src/elements/types.js +0 -5
  362. package/frontend/dist/frontend/src/index.d.ts +0 -21
  363. package/frontend/dist/frontend/src/index.js +0 -29
  364. package/frontend/dist/frontend/src/modules/attachment.d.ts +0 -126
  365. package/frontend/dist/frontend/src/modules/attachment.js +0 -306
  366. package/frontend/dist/frontend/src/modules/auth.d.ts +0 -44
  367. package/frontend/dist/frontend/src/modules/auth.js +0 -80
  368. package/frontend/dist/frontend/src/modules/color.d.ts +0 -160
  369. package/frontend/dist/frontend/src/modules/color.js +0 -316
  370. package/frontend/dist/frontend/src/modules/compression.d.ts +0 -39
  371. package/frontend/dist/frontend/src/modules/compression.js +0 -102
  372. package/frontend/dist/frontend/src/modules/cookies.d.ts +0 -44
  373. package/frontend/dist/frontend/src/modules/cookies.js +0 -143
  374. package/frontend/dist/frontend/src/modules/events.d.ts +0 -31
  375. package/frontend/dist/frontend/src/modules/events.js +0 -79
  376. package/frontend/dist/frontend/src/modules/google.d.ts +0 -23
  377. package/frontend/dist/frontend/src/modules/google.js +0 -52
  378. package/frontend/dist/frontend/src/modules/meta.d.ts +0 -14
  379. package/frontend/dist/frontend/src/modules/meta.js +0 -48
  380. package/frontend/dist/frontend/src/modules/paddle.d.ts +0 -1207
  381. package/frontend/dist/frontend/src/modules/paddle.js +0 -2594
  382. package/frontend/dist/frontend/src/modules/request.d.ts +0 -70
  383. package/frontend/dist/frontend/src/modules/request.js +0 -117
  384. package/frontend/dist/frontend/src/modules/settings.d.ts +0 -3
  385. package/frontend/dist/frontend/src/modules/settings.js +0 -5
  386. package/frontend/dist/frontend/src/modules/statics.d.ts +0 -21
  387. package/frontend/dist/frontend/src/modules/statics.js +0 -43
  388. package/frontend/dist/frontend/src/modules/stripe/cart.d.ts +0 -112
  389. package/frontend/dist/frontend/src/modules/stripe/cart.js +0 -321
  390. package/frontend/dist/frontend/src/modules/stripe/checkout.d.ts +0 -7
  391. package/frontend/dist/frontend/src/modules/stripe/checkout.js +0 -37
  392. package/frontend/dist/frontend/src/modules/stripe/index.m.d.ts +0 -6
  393. package/frontend/dist/frontend/src/modules/stripe/index.m.js +0 -6
  394. package/frontend/dist/frontend/src/modules/stripe/payments.d.ts +0 -58
  395. package/frontend/dist/frontend/src/modules/stripe/payments.js +0 -92
  396. package/frontend/dist/frontend/src/modules/support.d.ts +0 -30
  397. package/frontend/dist/frontend/src/modules/support.js +0 -53
  398. package/frontend/dist/frontend/src/modules/theme.d.ts +0 -133
  399. package/frontend/dist/frontend/src/modules/theme.js +0 -406
  400. package/frontend/dist/frontend/src/modules/themes.d.ts +0 -12
  401. package/frontend/dist/frontend/src/modules/themes.js +0 -22
  402. package/frontend/dist/frontend/src/modules/user.d.ts +0 -164
  403. package/frontend/dist/frontend/src/modules/user.js +0 -270
  404. package/frontend/dist/frontend/src/modules/utils.d.ts +0 -176
  405. package/frontend/dist/frontend/src/modules/utils.js +0 -569
  406. package/frontend/dist/frontend/src/types/gradient.d.ts +0 -29
  407. package/frontend/dist/frontend/src/types/gradient.js +0 -79
  408. package/frontend/dist/frontend/src/ui/border_button.d.ts +0 -94
  409. package/frontend/dist/frontend/src/ui/border_button.js +0 -228
  410. package/frontend/dist/frontend/src/ui/button.d.ts +0 -241
  411. package/frontend/dist/frontend/src/ui/button.js +0 -682
  412. package/frontend/dist/frontend/src/ui/canvas.d.ts +0 -138
  413. package/frontend/dist/frontend/src/ui/canvas.js +0 -444
  414. package/frontend/dist/frontend/src/ui/checkbox.d.ts +0 -74
  415. package/frontend/dist/frontend/src/ui/checkbox.js +0 -321
  416. package/frontend/dist/frontend/src/ui/code.d.ts +0 -235
  417. package/frontend/dist/frontend/src/ui/code.js +0 -1007
  418. package/frontend/dist/frontend/src/ui/context_menu.d.ts +0 -36
  419. package/frontend/dist/frontend/src/ui/context_menu.js +0 -205
  420. package/frontend/dist/frontend/src/ui/css.d.ts +0 -16
  421. package/frontend/dist/frontend/src/ui/css.js +0 -48
  422. package/frontend/dist/frontend/src/ui/divider.d.ts +0 -15
  423. package/frontend/dist/frontend/src/ui/divider.js +0 -78
  424. package/frontend/dist/frontend/src/ui/dropdown.d.ts +0 -176
  425. package/frontend/dist/frontend/src/ui/dropdown.js +0 -481
  426. package/frontend/dist/frontend/src/ui/for_each.d.ts +0 -37
  427. package/frontend/dist/frontend/src/ui/for_each.js +0 -92
  428. package/frontend/dist/frontend/src/ui/form.d.ts +0 -34
  429. package/frontend/dist/frontend/src/ui/form.js +0 -233
  430. package/frontend/dist/frontend/src/ui/frame_modes.d.ts +0 -37
  431. package/frontend/dist/frontend/src/ui/frame_modes.js +0 -108
  432. package/frontend/dist/frontend/src/ui/google_map.d.ts +0 -24
  433. package/frontend/dist/frontend/src/ui/google_map.js +0 -106
  434. package/frontend/dist/frontend/src/ui/gradient.d.ts +0 -25
  435. package/frontend/dist/frontend/src/ui/gradient.js +0 -131
  436. package/frontend/dist/frontend/src/ui/image.d.ts +0 -111
  437. package/frontend/dist/frontend/src/ui/image.js +0 -576
  438. package/frontend/dist/frontend/src/ui/input.d.ts +0 -392
  439. package/frontend/dist/frontend/src/ui/input.js +0 -1201
  440. package/frontend/dist/frontend/src/ui/link.d.ts +0 -25
  441. package/frontend/dist/frontend/src/ui/link.js +0 -140
  442. package/frontend/dist/frontend/src/ui/list.d.ts +0 -37
  443. package/frontend/dist/frontend/src/ui/list.js +0 -170
  444. package/frontend/dist/frontend/src/ui/loader_button.d.ts +0 -80
  445. package/frontend/dist/frontend/src/ui/loader_button.js +0 -193
  446. package/frontend/dist/frontend/src/ui/loaders.d.ts +0 -57
  447. package/frontend/dist/frontend/src/ui/loaders.js +0 -157
  448. package/frontend/dist/frontend/src/ui/popup.d.ts +0 -94
  449. package/frontend/dist/frontend/src/ui/popup.js +0 -510
  450. package/frontend/dist/frontend/src/ui/pseudo.d.ts +0 -44
  451. package/frontend/dist/frontend/src/ui/pseudo.js +0 -154
  452. package/frontend/dist/frontend/src/ui/scroller.d.ts +0 -105
  453. package/frontend/dist/frontend/src/ui/scroller.js +0 -1253
  454. package/frontend/dist/frontend/src/ui/slider.d.ts +0 -45
  455. package/frontend/dist/frontend/src/ui/slider.js +0 -217
  456. package/frontend/dist/frontend/src/ui/spacer.d.ts +0 -15
  457. package/frontend/dist/frontend/src/ui/spacer.js +0 -78
  458. package/frontend/dist/frontend/src/ui/span.d.ts +0 -15
  459. package/frontend/dist/frontend/src/ui/span.js +0 -73
  460. package/frontend/dist/frontend/src/ui/stack.d.ts +0 -66
  461. package/frontend/dist/frontend/src/ui/stack.js +0 -335
  462. package/frontend/dist/frontend/src/ui/steps.d.ts +0 -131
  463. package/frontend/dist/frontend/src/ui/steps.js +0 -308
  464. package/frontend/dist/frontend/src/ui/style.d.ts +0 -17
  465. package/frontend/dist/frontend/src/ui/style.js +0 -73
  466. package/frontend/dist/frontend/src/ui/switch.d.ts +0 -69
  467. package/frontend/dist/frontend/src/ui/switch.js +0 -357
  468. package/frontend/dist/frontend/src/ui/table.d.ts +0 -100
  469. package/frontend/dist/frontend/src/ui/table.js +0 -405
  470. package/frontend/dist/frontend/src/ui/tabs.d.ts +0 -111
  471. package/frontend/dist/frontend/src/ui/tabs.js +0 -424
  472. package/frontend/dist/frontend/src/ui/text.d.ts +0 -15
  473. package/frontend/dist/frontend/src/ui/text.js +0 -83
  474. package/frontend/dist/frontend/src/ui/title.d.ts +0 -91
  475. package/frontend/dist/frontend/src/ui/title.js +0 -272
  476. package/frontend/dist/frontend/src/ui/ui.d.ts +0 -35
  477. package/frontend/dist/frontend/src/ui/ui.js +0 -38
  478. package/frontend/dist/frontend/src/ui/view.d.ts +0 -15
  479. package/frontend/dist/frontend/src/ui/view.js +0 -88
  480. package/frontend/dist/frontend/src/volt.d.ts +0 -20
  481. package/frontend/dist/frontend/src/volt.js +0 -27
  482. package/frontend/examples/theme/theme.ts +0 -58
  483. package/frontend/tools/bundle_d_ts.js +0 -71
  484. package/frontend/tools/convert_to_jsdoc_input.txt +0 -9452
  485. package/frontend/tools/convert_to_jsdoc_output.txt +0 -7626
  486. package/frontend/tools/convert_to_jsdoc_tmp.js +0 -345
  487. package/frontend/tools/scan_mixed_imports.js +0 -69
  488. /package/frontend/{dist/frontend/src/css → css}/adyen.css +0 -0
  489. /package/frontend/{dist/frontend/src/css → css}/volt.css +0 -0
@@ -1,1219 +0,0 @@
1
- /**
2
- * @author Daan van den Bergh
3
- * @copyright © 2025 - 2025 Daan van den Bergh. All rights reserved.
4
- */
5
- // Imports.
6
- import { SystemError } from "../../errors/system_error.js";
7
- import { Collection, TransactionCollection } from "../collection.js";
8
- import { InvalidUsageError } from "../../errors/index.js";
9
- import { SafeInt } from "./safe_int.js";
10
- /**
11
- * The quota manager.
12
- * This manager can be used to manage and enforce usage quotas for different resources,
13
- * for instance limiting money spent on tokens, or for rate limiting purposes.
14
- *
15
- * For managing monetary quotas, it is advised to use a nano-scale amount system
16
- * (smallest unit integer accounting). See {@link Quota.to_nano} for converting to nano scale.
17
- *
18
- * The quota manager ties each operation to a specific user id. For creating system quota's you can
19
- * use a non numeric `uid` to simulate a `uid` if needed.
20
- *
21
- * @template Type The allowed type for {@link Query.type}, provided at class level.
22
- *
23
- * @nav Database
24
- * @docs
25
- */
26
- export class QuotaManager {
27
- // ----------------------------------------------------------------
28
- // Attributes
29
- // ----------------------------------------------------------------
30
- /** The collection for database operations. */
31
- collection;
32
- /** The system error options. */
33
- system_error;
34
- /**
35
- * Construct a new quota manager with a specific quota type.
36
- *
37
- * @throws {InvalidUsageError} If {@link QuotaManager.Opts.collection} is already initialized and does not have the correct index.
38
- * If the passed collection has manually assigned fields for {@link Collection.record_version} or {@link Collection.on_transform_version}.
39
- * If the passed collection is transaction based.
40
- *
41
- * @docs
42
- */
43
- constructor(opts) {
44
- // Attributes.
45
- this.collection = opts.server.db.collection({
46
- name: opts.collection.name,
47
- ttl: opts.collection.ttl,
48
- indexes: [
49
- { key: "id", unique: true, forced: true },
50
- { key: "uid", forced: true },
51
- ],
52
- unique: true,
53
- persist_transformed_on_load: "replace",
54
- record_version: 1,
55
- // on_transform_version() {}
56
- });
57
- this.system_error = opts.system_error;
58
- }
59
- // ----------------------------------------------------------------
60
- // Private utility methods.
61
- // ----------------------------------------------------------------
62
- /**
63
- * Helper to create consistent query objects for MongoDB.
64
- */
65
- create_db_query(query) {
66
- return { uid: query.uid, id: query.id };
67
- }
68
- /**
69
- * Format a compact, human-readable quota identifier for logs and error messages.
70
- *
71
- * @param query An object carrying the `uid` and `id` fields.
72
- * @returns A stable identifier of the form `<uid>:<id>`.
73
- */
74
- format_quota_id(query) {
75
- return `${query.uid}:${query.id}`;
76
- }
77
- /**
78
- * Clamp a bigint to a minimum of 0n.
79
- */
80
- static clamp_min0(v) {
81
- return v > 0n ? v : 0n;
82
- }
83
- /**
84
- * Compute remaining capacity, clamped to [0, +inf).
85
- */
86
- static remaining(max, usage) {
87
- return max > usage ? (max - usage) : 0n;
88
- }
89
- /**
90
- * Compute percentage used in [0, 100] (2 decimals), based on bigint usage/max.
91
- */
92
- static percentage_used(usage, max) {
93
- if (max > 0n) {
94
- // 2-decimal fixed point percent: (usage/max)*100 scaled by 100 => *10000
95
- const scaled = (usage * 10000n) / max;
96
- const capped = scaled < 0n ? 0n : (scaled > 10000n ? 10000n : scaled);
97
- return Number(capped) / 100;
98
- }
99
- return usage > 0n ? 100 : 0;
100
- }
101
- /**
102
- * Multiply a bigint by a floating ratio, returning a bigint.
103
- * - For positive values, rounds up (ceil) to avoid underestimation.
104
- * - For negative values, rounds down (floor).
105
- */
106
- static mul_bigint_ratio(value, ratio) {
107
- if (ratio === 1)
108
- return value;
109
- if (!Number.isFinite(ratio)) {
110
- // Defensive: callers validate, but keep this stable.
111
- throw new Error(`Invalid ratio: ${ratio}`);
112
- }
113
- if (ratio <= 0)
114
- throw new Error(`Invalid ratio: ${ratio}. Must be > 0.`);
115
- if (Number.isInteger(ratio)) {
116
- if (!Number.isSafeInteger(ratio)) {
117
- throw new Error(`Invalid ratio: ${ratio}. Integer ratio must be a safe integer.`);
118
- }
119
- return value * BigInt(ratio);
120
- }
121
- // Fixed-point multiply using 1e9 scaling to keep decent precision.
122
- const SCALE_NUM = 1_000_000_000;
123
- const SCALE = 1000000000n;
124
- const scaled_float = ratio * SCALE_NUM;
125
- if (!Number.isFinite(scaled_float) || Math.abs(scaled_float) > Number.MAX_SAFE_INTEGER) {
126
- // If this overflows safe integer range, BigInt(Math.round(...)) becomes unreliable.
127
- throw new Error(`Invalid ratio: ${ratio}. Too large for fixed-point scaling at 1e9 precision.`);
128
- }
129
- const scaled_int = Math.round(scaled_float);
130
- if (!Number.isSafeInteger(scaled_int)) {
131
- throw new Error(`Invalid ratio scaling: ratio=${ratio}, scaled=${scaled_int}.`);
132
- }
133
- const scaled_ratio = BigInt(scaled_int);
134
- const product = value * scaled_ratio;
135
- let q = product / SCALE;
136
- const r = product % SCALE;
137
- if (r !== 0n) {
138
- if (value > 0n) {
139
- // ceil for positive
140
- q += 1n;
141
- }
142
- else if (value < 0n) {
143
- // floor for negative (BigInt division truncates toward 0)
144
- q -= 1n;
145
- }
146
- }
147
- return q;
148
- }
149
- // ----------------------------------------------------------------
150
- // Database operation methods.
151
- // ----------------------------------------------------------------
152
- /**
153
- * Get current quota status without modifying it.
154
- *
155
- * @note The `opts.retry` field defaults to `25`.
156
- * @note System load errors are not saved inside this function.
157
- *
158
- * @param query The quota identifier arguments.
159
- * @param opts Additional load options.
160
- *
161
- * @returns A load result depending on `opts`, see {@link Collection.LoadResult}
162
- *
163
- * @throws {Collection.NotFoundError} When `opts.throw !== false` and the quota does not exist.
164
- * @throws {Collection.LoadError} When `opts.throw !== false` and a database error was encountered during the load operation.
165
- * @throws {Collection.InvalidUsageError} When `opts.throw !== false` and the query is invalid.
166
- *
167
- * @docs
168
- */
169
- async get(query, opts) {
170
- // Validate quota identity + config
171
- const val_err = QuotaManager.Query.validate(query);
172
- if (val_err) {
173
- const err = new InvalidUsageError({
174
- message: `Invalid quota: ${val_err}`,
175
- reason: "invalid_quota",
176
- field: "quota",
177
- });
178
- if (opts?.throw ?? true)
179
- throw err;
180
- return err;
181
- }
182
- // Load.
183
- if (opts) {
184
- opts = { retry: 25, ...opts };
185
- return this.collection.load(this.create_db_query(query), opts);
186
- }
187
- else {
188
- return this.collection.load(this.create_db_query(query), { retry: 25 });
189
- }
190
- }
191
- /**
192
- * Get current quota status without modifying it.
193
- *
194
- * @param query The quota identifier arguments.
195
- * @param opts Additional load options, see {@link Collection.LoadOpts}.
196
- *
197
- * @returns An object containing error or status information,
198
- * see {@link QuotaManager.GetStatusResult}
199
- *
200
- * @docs
201
- */
202
- async get_status(query, opts) {
203
- const now_sec = Math.floor(Date.now() / 1000);
204
- const db_query = this.create_db_query(query);
205
- // Validate query.
206
- const val_err = QuotaManager.Query.validate(query);
207
- if (val_err) {
208
- return {
209
- found: false,
210
- reason: "invalid_query",
211
- error: `Invalid query: ${val_err}`,
212
- };
213
- }
214
- const loaded_quota = await this.collection.load(db_query, { retry: 25, throw: false, timeout: opts?.timeout });
215
- if (loaded_quota instanceof Collection.NotFoundError) {
216
- return {
217
- found: false,
218
- reason: "not_found",
219
- error: `Quota not found: ${this.format_quota_id(query)} for user ${query.uid}`,
220
- };
221
- }
222
- if (loaded_quota instanceof Error) {
223
- SystemError.create_detach({
224
- owner: "volt.QuotaManager",
225
- collection: this.system_error?.collection,
226
- logger: this.system_error?.logger,
227
- message: "Failed to load quota in get_status().",
228
- details: {
229
- quota_id: this.format_quota_id(query),
230
- uid: query.uid,
231
- original_error: loaded_quota?.message ?? String(loaded_quota),
232
- },
233
- });
234
- return {
235
- found: false,
236
- reason: "system_error",
237
- error: `Encountered an unknown error while loading quota: ${this.format_quota_id(query)} for user ${query.uid}.`,
238
- };
239
- }
240
- const needs_reset = now_sec >= (loaded_quota.start + loaded_quota.interval);
241
- const effective_usage = needs_reset ? 0n : loaded_quota.usage;
242
- const time_until_reset = needs_reset ? 0 : Math.max(0, (loaded_quota.start + loaded_quota.interval) - now_sec);
243
- const remaining = QuotaManager.remaining(loaded_quota.max, effective_usage);
244
- const percentage_used = QuotaManager.percentage_used(effective_usage, loaded_quota.max);
245
- return {
246
- found: true,
247
- quota: needs_reset ? { ...loaded_quota, usage: 0n, start: now_sec } : loaded_quota,
248
- remaining,
249
- percentage_used,
250
- needs_reset,
251
- time_until_reset,
252
- };
253
- }
254
- /**
255
- * List all quotas for a user, optionally filtered by type.
256
- *
257
- * @param uid The user identifier.
258
- * @param type Optional quota type filter.
259
- * @returns List of quotas with their current status.
260
- *
261
- * @docs
262
- */
263
- async list({ uid, timeout, }) {
264
- const now_sec = Math.floor(Date.now() / 1000);
265
- const listed = [];
266
- await this.collection.list({ uid }, {
267
- timeout,
268
- retry: 5,
269
- callback: (q) => {
270
- const needs_reset = now_sec >= (q.start + q.interval);
271
- const effective_usage = needs_reset ? 0n : q.usage;
272
- const percentage_used = QuotaManager.percentage_used(effective_usage, q.max);
273
- listed.push({
274
- quota: needs_reset ? { ...q, usage: 0n, start: now_sec } : q,
275
- remaining: QuotaManager.remaining(q.max, effective_usage),
276
- percentage_used,
277
- needs_reset,
278
- });
279
- }
280
- });
281
- return listed;
282
- }
283
- /**
284
- * Update or save quota configuration (`max`, `interval`) for an existing quota document,
285
- * automatically creating the document if it does not exist.
286
- *
287
- * This method never resets runtime fields on existing documents:
288
- * - On update: only `max` and `interval` are changed.
289
- * - On insert: `usage` is initialized to `0` and `start` to the current UNIX timestamp (seconds).
290
- *
291
- * @param quota The configuration document (identity + {@link Quota.Opts}). Fields `usage` and `start`
292
- * are forbidden at the type level and ignored defensively at runtime.
293
- * @param opts Additional save options; see {@link Collection.SaveOpts}.
294
- *
295
- * @note The `opts.throw` field defaults to `true`.
296
- *
297
- * @returns The updated (or newly created) quota document, or an error-like result depending on `opts.throw`.
298
- * See {@link Collection.SaveResult}.
299
- *
300
- * @throws {InvalidUsageError} When `opts.throw !== false` and validation fails.
301
- * @throws {Collection.SaveError} When `opts.throw !== false` and a database error occurs during the save operation.
302
- *
303
- * @docs
304
- */
305
- async set(quota, opts) {
306
- // Validate quota identity + config
307
- const val_err = QuotaManager.Document.Opts.validate(quota, this.collection);
308
- if (val_err) {
309
- const err = new InvalidUsageError({
310
- message: `Invalid quota: ${val_err}`,
311
- reason: "invalid_quota",
312
- field: "quota",
313
- });
314
- if (opts?.throw ?? true)
315
- throw err;
316
- return err;
317
- }
318
- // Atomic upsert that never resets runtime fields on existing documents:
319
- // - $set updates config only.
320
- // - $setOnInsert initializes runtime counters on first creation.
321
- const now_sec = Math.floor(Date.now() / 1000);
322
- const save_opts = {
323
- return: true,
324
- upsert: true,
325
- retry: 25,
326
- throw: opts?.throw ?? true,
327
- timeout: opts?.timeout,
328
- };
329
- return await this.collection.save(this.create_db_query(quota), {
330
- $set: {
331
- max: quota.max,
332
- interval: quota.interval,
333
- },
334
- $setOnInsert: {
335
- usage: 0n,
336
- start: now_sec,
337
- },
338
- }, save_opts);
339
- }
340
- /**
341
- * Reset quota usage to zero & timestamp to the current unix timestamp for the specified quota.
342
- *
343
- * @param query The quota identifier, see {@link QuotaManager.Query}.
344
- * @param opts Additional save options, see {@link Collection.SaveOpts}
345
- *
346
- * @note The `opts.throw` field defaults to the default value of {@link Collection.SaveOpts.throw}.
347
- *
348
- * @returns The updated quota document after resetting quota or an error depending on `throw`.
349
- * See {@link Collection.SaveResult}.
350
- *
351
- * @throws {Collection.NotFoundError} When `opts.throw !== false` and the quota does not exist.
352
- * @throws {Collection.SaveError} When `opts.throw !== false` and a database error was encountered during the save operation.
353
- * @throws {Collection.InvalidUsageError} When `opts.throw !== false` and the query is invalid.
354
- *
355
- * @docs
356
- */
357
- async reset_usage(query, opts) {
358
- // Check.
359
- const query_err = QuotaManager.Query.validate(query);
360
- if (query_err) {
361
- const err = new InvalidUsageError({
362
- message: `Invalid quota query: ${query_err}`,
363
- reason: "invalid_query",
364
- field: "query",
365
- });
366
- if (opts?.throw ?? true)
367
- throw err;
368
- return err;
369
- }
370
- // Save.
371
- const save_opts = {
372
- return: true,
373
- upsert: false,
374
- retry: 25,
375
- throw: opts?.throw,
376
- timeout: opts?.timeout,
377
- };
378
- return await this.collection.save(this.create_db_query(query), {
379
- $set: {
380
- usage: 0n,
381
- start: Math.floor(Date.now() / 1000)
382
- },
383
- }, save_opts);
384
- }
385
- /**
386
- * Delete all quotas for a user.
387
- *
388
- * @docs
389
- */
390
- async delete_by_user({ uid }) {
391
- await this.collection.delete_many({ uid }, { retry: 25 });
392
- }
393
- // ----------------------------------------------------------------
394
- // Quota limiting.
395
- // ----------------------------------------------------------------
396
- /**
397
- * Validate the required {@link limit_helper} parameters.
398
- * @note Parameter `requested_usage` may be a negative number.
399
- *
400
- * @docs
401
- */
402
- validate_limit_helper_params({ requested_usage, safety_ratio, query, upsert, }) {
403
- // Param `requested_usage` may be a negative number in case the
404
- // estimated quota usage was higher then the actual usage.
405
- // This could for instance happen in class OpenAI.
406
- // Validate safety ratio.
407
- if (safety_ratio !== undefined) {
408
- if (!Number.isFinite(safety_ratio) || safety_ratio < 1) {
409
- // Safety ratio must be finite and >= 1 to avoid underestimation.
410
- return {
411
- success: false,
412
- status: "invalid_usage",
413
- error: `Invalid 'safety_ratio' value: ${safety_ratio}. Must be finite and >= 1.`,
414
- };
415
- }
416
- // Ensure the fixed-point path in mul_bigint_ratio() remains exact in JS number space.
417
- const SCALE_NUM = 1_000_000_000;
418
- const scaled = safety_ratio * SCALE_NUM;
419
- if (!Number.isFinite(scaled) || Math.abs(scaled) > Number.MAX_SAFE_INTEGER) {
420
- return {
421
- success: false,
422
- status: "invalid_usage",
423
- error: `Invalid 'safety_ratio' value: ${safety_ratio}. Too large for safe fixed-point scaling.`,
424
- };
425
- }
426
- }
427
- // Validate query.
428
- const val_err = QuotaManager.Query.validate(query);
429
- if (val_err) {
430
- return {
431
- success: false,
432
- status: "invalid_usage",
433
- error: `Invalid quota query: ${val_err}`,
434
- };
435
- }
436
- // Validate upsertion.
437
- if (upsert) {
438
- const record = {
439
- ...upsert,
440
- ...query,
441
- };
442
- const val_err = QuotaManager.Document.Opts.validate(record, this.collection);
443
- if (val_err) {
444
- return {
445
- success: false,
446
- status: "invalid_usage",
447
- error: `Invalid quota upsert: ${val_err}`,
448
- };
449
- }
450
- }
451
- }
452
- /**
453
- * Validates quota limits and atomically increments usage if within bounds.
454
- * Handles interval resets automatically in a single database operation.
455
- *
456
- * @warning Ensure the quota exists in the database, or define `upsert` to create it when needed.
457
- *
458
- * @note This automatically increments the quota usage with the requested usage when `perform_increment` is true.
459
- * When `perform_increment` is false, it only validates availability without modifying the database.
460
- *
461
- * @returns Success with updated quota info or validation/error details.
462
- *
463
- * @docs
464
- *
465
- */
466
- async limit_helper({ query, requested_usage, upsert, safety_ratio = 1, check_limit = true, perform_increment = true, collection, }) {
467
- const val_input_res = this.validate_limit_helper_params({
468
- requested_usage,
469
- safety_ratio,
470
- upsert,
471
- query,
472
- });
473
- if (val_input_res)
474
- return val_input_res;
475
- const safety_usage = QuotaManager.mul_bigint_ratio(requested_usage, safety_ratio);
476
- const now_sec = Math.floor(Date.now() / 1000);
477
- const db_query = this.create_db_query(query);
478
- // ---------------------------
479
- // fast path (no reset needed)
480
- // ---------------------------
481
- if (check_limit) {
482
- if (perform_increment) {
483
- // enforce BOTH actual and safety-ratio checks, and prevent negative usage
484
- const result = await collection.save({
485
- ...db_query,
486
- $expr: {
487
- $and: [
488
- { $lt: [now_sec, { $add: ["$start", "$interval"] }] },
489
- { $lte: [{ $add: ["$usage", requested_usage] }, "$max"] },
490
- { $lte: [{ $add: ["$usage", safety_usage] }, "$max"] },
491
- { $gte: [{ $add: ["$usage", requested_usage] }, 0n] },
492
- ]
493
- }
494
- }, { $inc: { usage: requested_usage } }, { return: true, upsert: false, retry: 25, throw: false });
495
- if (!(result instanceof Error)) {
496
- return {
497
- success: true,
498
- status: "success",
499
- quota: result,
500
- remaining: QuotaManager.remaining(result.max, result.usage),
501
- was_reset: false,
502
- };
503
- }
504
- }
505
- else {
506
- // check-only fast path: verify constraints without incrementing
507
- const result = await collection.load({
508
- ...db_query,
509
- $expr: {
510
- $and: [
511
- { $lt: [now_sec, { $add: ["$start", "$interval"] }] },
512
- { $lte: [{ $add: ["$usage", requested_usage] }, "$max"] },
513
- { $lte: [{ $add: ["$usage", safety_usage] }, "$max"] },
514
- { $gte: [{ $add: ["$usage", requested_usage] }, 0n] },
515
- ]
516
- }
517
- }, { retry: 25, throw: false });
518
- if (!(result instanceof Error)) {
519
- return {
520
- success: true,
521
- status: "success",
522
- quota: result,
523
- remaining: QuotaManager.remaining(result.max, result.usage),
524
- was_reset: false,
525
- };
526
- }
527
- }
528
- }
529
- else {
530
- if (perform_increment) {
531
- // increment without limit checks; still prevent negative usage via atomic guard
532
- const result = await collection.save({
533
- ...db_query,
534
- $expr: {
535
- $and: [
536
- { $lt: [now_sec, { $add: ["$start", "$interval"] }] },
537
- { $gte: [{ $add: ["$usage", requested_usage] }, 0n] },
538
- ]
539
- }
540
- }, { $inc: { usage: requested_usage } }, { return: true, upsert: false, throw: false, retry: 25 });
541
- if (!(result instanceof Error)) {
542
- return {
543
- success: true,
544
- status: "success",
545
- quota: result,
546
- remaining: QuotaManager.remaining(result.max, result.usage),
547
- was_reset: false,
548
- };
549
- }
550
- }
551
- // perform_increment === false and check_limit === false:
552
- // fall through to slow path; we'll just load and return current state.
553
- }
554
- // ---------------------------
555
- // slow path (load current)
556
- // ---------------------------
557
- let current = await collection.load(db_query, { retry: 25, throw: false });
558
- if (current instanceof Error) {
559
- // only treat NotFoundError as "document missing"; everything else is a system error
560
- if (!(current instanceof Collection.NotFoundError)) {
561
- SystemError.create_detach({
562
- owner: "volt.QuotaManager",
563
- collection: this.system_error?.collection,
564
- logger: this.system_error?.logger,
565
- message: `Encountered an unknown error while loading quota '${this.format_quota_id(query)}' for user '${query.uid}'`,
566
- details: {
567
- query,
568
- requested_usage,
569
- upsert,
570
- safety_ratio,
571
- check_limit,
572
- perform_increment,
573
- is_transaction: collection instanceof TransactionCollection,
574
- original_error: current?.message ?? String(current),
575
- },
576
- });
577
- return {
578
- success: false,
579
- status: "system_error",
580
- error: `Encountered an unknown error while loading quota '${this.format_quota_id(query)}' for user '${query.uid}'`,
581
- };
582
- }
583
- // not found
584
- if (!upsert) {
585
- return {
586
- success: false,
587
- status: "not_found",
588
- error: `Quota not found '${this.format_quota_id(query)}' for user '${query.uid}'`,
589
- };
590
- }
591
- // upsert provided but perform a check-only (no write)
592
- if (!perform_increment) {
593
- const would_exceed_actual = requested_usage > upsert.max;
594
- const would_exceed_ratio = safety_usage > upsert.max;
595
- if (check_limit && (would_exceed_actual || would_exceed_ratio)) {
596
- return {
597
- success: false,
598
- status: "would_exceed",
599
- error: `Requested usage (${requested_usage}, safety=${safety_usage}) exceeds fresh-window maximum (${upsert.max}).`,
600
- remaining: upsert.max,
601
- };
602
- }
603
- const virtual_doc = {
604
- uid: query.uid,
605
- id: query.id,
606
- max: upsert.max,
607
- interval: upsert.interval,
608
- start: now_sec,
609
- usage: QuotaManager.clamp_min0(requested_usage),
610
- };
611
- return {
612
- success: true,
613
- status: "success",
614
- quota: virtual_doc,
615
- remaining: QuotaManager.remaining(virtual_doc.max, virtual_doc.usage),
616
- was_reset: false,
617
- };
618
- }
619
- // perform_increment === true: create within the provided collection/transaction
620
- const doc_record = {
621
- uid: query.uid,
622
- id: query.id,
623
- max: upsert.max,
624
- interval: upsert.interval,
625
- start: now_sec,
626
- usage: QuotaManager.clamp_min0(requested_usage),
627
- };
628
- const created = await collection.set(db_query, doc_record, { return: true, upsert: true, throw: false, retry: 25 });
629
- if (created instanceof Error) {
630
- SystemError.create_detach({
631
- owner: "volt.QuotaManager",
632
- collection: this.system_error?.collection,
633
- logger: this.system_error?.logger,
634
- message: `Failed to create quota '${this.format_quota_id(query)}' for user '${query.uid}'`,
635
- details: {
636
- query,
637
- requested_usage,
638
- upsert,
639
- safety_ratio,
640
- check_limit,
641
- perform_increment,
642
- is_transaction: collection instanceof TransactionCollection,
643
- original_error: created?.message ?? String(created),
644
- },
645
- });
646
- return {
647
- success: false,
648
- status: "system_error",
649
- error: `Failed to create quota '${this.format_quota_id(query)}' for user '${query.uid}'`,
650
- };
651
- }
652
- current = created;
653
- }
654
- // interval expired -> reset window then apply/check
655
- const interval_expired = now_sec >= (current.start + current.interval);
656
- if (interval_expired) {
657
- if (check_limit) {
658
- const would_exceed_actual = requested_usage > current.max;
659
- const would_exceed_ratio = safety_usage > current.max;
660
- if (would_exceed_actual || would_exceed_ratio) {
661
- return {
662
- success: false,
663
- status: "would_exceed",
664
- error: `Requested usage (${requested_usage}, safety=${safety_usage}) exceeds fresh-window maximum (${current.max}).`,
665
- quota: current,
666
- remaining: current.max,
667
- };
668
- }
669
- }
670
- if (!perform_increment) {
671
- // check-only: return a "would-be" reset view without writing
672
- const view_after_reset = { ...current, usage: 0n, start: now_sec };
673
- return {
674
- success: true,
675
- status: "success",
676
- quota: view_after_reset,
677
- remaining: QuotaManager.remaining(view_after_reset.max, view_after_reset.usage),
678
- was_reset: false,
679
- };
680
- }
681
- // perform_increment === true -> actually reset (and apply increment if any)
682
- const new_usage = QuotaManager.clamp_min0(requested_usage);
683
- const reset_result = await collection.save({
684
- ...db_query,
685
- // optimistic lock against concurrent reset
686
- start: current.start,
687
- }, {
688
- $set: {
689
- usage: new_usage,
690
- start: now_sec,
691
- }
692
- }, { return: true, upsert: false, throw: false, retry: 25 });
693
- if (!(reset_result instanceof Error)) {
694
- return {
695
- success: true,
696
- status: "success",
697
- quota: reset_result,
698
- remaining: QuotaManager.remaining(reset_result.max, reset_result.usage),
699
- was_reset: true,
700
- };
701
- }
702
- SystemError.create_detach({
703
- owner: "volt.QuotaManager",
704
- collection: this.system_error?.collection,
705
- logger: this.system_error?.logger,
706
- message: `Race condition detected after maximum retries.`,
707
- details: {
708
- query, requested_usage, upsert,
709
- safety_ratio, check_limit, perform_increment,
710
- is_transaction: collection instanceof TransactionCollection,
711
- },
712
- });
713
- return {
714
- success: false,
715
- status: "system_error",
716
- error: `Race condition detected after maximum retries.`,
717
- quota: current,
718
- remaining: QuotaManager.remaining(current.max, current.usage),
719
- };
720
- }
721
- // interval active (no reset)
722
- if (check_limit) {
723
- /**
724
- * Check-only path in the active window: `check_limit === true && perform_increment === false`.
725
- * Performs validation without mutating the database.
726
- * - Fails if the quota is already exceeded.
727
- * - Fails if `(usage + requested_usage)` would exceed `max` (including safety ratio).
728
- * - Returns the current quota snapshot on success with `was_reset: false`.
729
- */
730
- if (!perform_increment) {
731
- /** Quota already exceeded; no capacity remains. */
732
- if (current.usage > current.max) {
733
- return {
734
- success: false,
735
- status: "exceeded",
736
- error: `Quota usage '${current.usage}' has already exceeded maximum quota '${current.max}'`,
737
- quota: current,
738
- remaining: QuotaManager.remaining(current.max, current.usage),
739
- };
740
- }
741
- const would_exceed_actual = (current.usage + requested_usage) > current.max;
742
- const would_exceed_ratio = (current.usage + safety_usage) > current.max;
743
- if (would_exceed_actual || would_exceed_ratio) {
744
- return {
745
- success: false,
746
- status: "would_exceed",
747
- error: `Requested usage (${requested_usage}, safety=${safety_usage}) would exceed remaining quota.`,
748
- quota: current,
749
- remaining: QuotaManager.remaining(current.max, current.usage),
750
- };
751
- }
752
- /** Success: constraints satisfied, no DB mutations performed. */
753
- return {
754
- success: true,
755
- status: "success",
756
- quota: current,
757
- remaining: QuotaManager.remaining(current.max, current.usage),
758
- was_reset: false,
759
- };
760
- }
761
- /**
762
- * `check_limit && perform_increment === true` -> original guarded increment
763
- */
764
- if ((current.usage + requested_usage) < 0n) {
765
- const clamp_result = await collection.save({
766
- ...db_query,
767
- start: current.start, // optimistic lock in the same window
768
- }, { $set: { usage: 0n } }, { return: true, upsert: false, throw: false, retry: 25 });
769
- if (!(clamp_result instanceof Error)) {
770
- return {
771
- success: true,
772
- status: "success",
773
- quota: clamp_result,
774
- remaining: QuotaManager.remaining(clamp_result.max, clamp_result.usage),
775
- was_reset: false,
776
- };
777
- }
778
- SystemError.create_detach({
779
- owner: "volt.QuotaManager",
780
- collection: this.system_error?.collection,
781
- logger: this.system_error?.logger,
782
- message: `Failed to clamp usage to zero for query '${this.format_quota_id(query)}'.`,
783
- details: {
784
- query, requested_usage, upsert,
785
- safety_ratio, check_limit, perform_increment,
786
- is_transaction: collection instanceof TransactionCollection,
787
- },
788
- });
789
- return {
790
- success: false,
791
- status: "system_error",
792
- error: `Failed to clamp usage to zero for query '${this.format_quota_id(query)}'.`,
793
- quota: current,
794
- remaining: QuotaManager.remaining(current.max, current.usage),
795
- };
796
- }
797
- /** Quota already exceeded; do not allow further increments. */
798
- if (current.usage > current.max) {
799
- return {
800
- success: false,
801
- status: "exceeded",
802
- error: `Quota usage '${current.usage}' has already exceeded maximum quota '${current.max}'`,
803
- quota: current,
804
- remaining: QuotaManager.remaining(current.max, current.usage),
805
- };
806
- }
807
- const would_exceed_actual = (current.usage + requested_usage) > current.max;
808
- const would_exceed_ratio = (current.usage + safety_usage) > current.max;
809
- if (would_exceed_actual || would_exceed_ratio) {
810
- return {
811
- success: false,
812
- status: "would_exceed",
813
- error: `Requested usage (${requested_usage}, safety=${safety_usage}) would exceed remaining quota.`,
814
- quota: current,
815
- remaining: QuotaManager.remaining(current.max, current.usage),
816
- };
817
- }
818
- // race-safe increment guarded by start equality and non-negative
819
- const inc_result = await collection.save({
820
- ...db_query,
821
- start: current.start,
822
- $expr: { $gte: [{ $add: ["$usage", requested_usage] }, 0n] },
823
- }, { $inc: { usage: requested_usage } }, { return: true, upsert: false, throw: false, retry: 25 });
824
- if (!(inc_result instanceof Error)) {
825
- return {
826
- success: true,
827
- status: "success",
828
- quota: inc_result,
829
- remaining: QuotaManager.remaining(inc_result.max, inc_result.usage),
830
- was_reset: false,
831
- };
832
- }
833
- SystemError.create_detach({
834
- owner: "volt.QuotaManager",
835
- collection: this.system_error?.collection,
836
- logger: this.system_error?.logger,
837
- message: `Failed to update quota for query '${this.format_quota_id(query)}'.`,
838
- details: {
839
- query, requested_usage, upsert,
840
- safety_ratio, check_limit, perform_increment,
841
- is_transaction: collection instanceof TransactionCollection,
842
- },
843
- });
844
- return {
845
- success: false,
846
- status: "system_error",
847
- error: `Failed to update quota for query '${this.format_quota_id(query)}'.`,
848
- quota: current,
849
- remaining: QuotaManager.remaining(current.max, current.usage),
850
- };
851
- }
852
- // check_limit === false
853
- else {
854
- if (!perform_increment) {
855
- // check-only without limit checks: return current state (window-view) without writing
856
- const needs_reset = now_sec >= current.start + current.interval;
857
- const effective_usage = needs_reset ? 0n : current.usage;
858
- const view_quota = needs_reset ? { ...current, usage: 0n, start: now_sec } : current;
859
- return {
860
- success: true,
861
- status: "success",
862
- quota: view_quota,
863
- remaining: QuotaManager.remaining(view_quota.max, effective_usage),
864
- was_reset: false,
865
- };
866
- }
867
- /**
868
- * Increment-only slow path with race safety and non-negative invariant.
869
- *
870
- * Invariant:
871
- * - Never persist a negative `usage`.
872
- *
873
- * Mechanism:
874
- * - First attempt an atomic guarded increment (`$expr: usage + requested_usage >= 0`)
875
- * under optimistic lock `start: current.start`.
876
- * - If the guard fails (e.g., the increment would underflow), clamp `usage` to `0`
877
- * with the same optimistic lock to avoid TOCTOU races.
878
- */
879
- const inc_result = await collection.save({
880
- ...db_query,
881
- start: current.start,
882
- $expr: { $gte: [{ $add: ["$usage", requested_usage] }, 0n] },
883
- }, { $inc: { usage: requested_usage } }, { return: true, upsert: false, throw: false, retry: 25 });
884
- if (!(inc_result instanceof Error)) {
885
- return {
886
- success: true,
887
- status: "success",
888
- quota: inc_result,
889
- remaining: QuotaManager.remaining(inc_result.max, inc_result.usage),
890
- was_reset: false,
891
- };
892
- }
893
- /**
894
- * Guard failed — clamp to zero atomically (same optimistic lock).
895
- */
896
- const clamp_result = await collection.save({ ...db_query, start: current.start }, { $set: { usage: 0n } }, { return: true, upsert: false, throw: false, retry: 25 });
897
- if (!(clamp_result instanceof Error)) {
898
- return {
899
- success: true,
900
- status: "success",
901
- quota: clamp_result,
902
- remaining: QuotaManager.remaining(clamp_result.max, clamp_result.usage),
903
- was_reset: false,
904
- };
905
- }
906
- SystemError.create_detach({
907
- owner: "volt.QuotaManager",
908
- collection: this.system_error?.collection,
909
- logger: this.system_error?.logger,
910
- message: `Failed to update quota for query '${this.format_quota_id(query)}'.`,
911
- details: {
912
- query, requested_usage, upsert,
913
- safety_ratio, check_limit, perform_increment,
914
- is_transaction: collection instanceof TransactionCollection,
915
- },
916
- });
917
- return {
918
- success: false,
919
- status: "system_error",
920
- error: `Failed to update quota for query '${this.format_quota_id(query)}'.`,
921
- quota: current,
922
- remaining: QuotaManager.remaining(current.max, current.usage),
923
- };
924
- }
925
- }
926
- /**
927
- * Validate quota limits and, optionally, atomically increment usage if within bounds.
928
- * Handles interval resets automatically in a single database operation.
929
- *
930
- * @warning Ensure the quota exists in the database, or provide `upsert` to create it when needed.
931
- *
932
- * @param perform_increment When `true` (default), performs the atomic increment. When `false`,
933
- * executes a dry-run validation without modifying the database.
934
- *
935
- * @note Negative `requested_usage` is not allowed. Use {@link increment} for decrements.
936
- *
937
- * @returns On success, returns the (possibly updated) quota and remaining capacity; on failure,
938
- * returns a diagnostic indicating why the request was rejected.
939
- *
940
- * @docs
941
- */
942
- async limit({ query, requested_usage, upsert, safety_ratio, perform_increment = true }) {
943
- if (requested_usage < 0n) {
944
- return {
945
- success: false,
946
- status: "invalid_usage",
947
- error: `Negative requested_usage (${requested_usage}) is not allowed in 'limit'. Use 'increment' for decrements.`,
948
- };
949
- }
950
- return this.limit_helper({
951
- query,
952
- requested_usage,
953
- upsert,
954
- safety_ratio,
955
- collection: this.collection,
956
- check_limit: true,
957
- perform_increment,
958
- });
959
- }
960
- /**
961
- * Increment the usage on a quota.
962
- *
963
- * @warning This does not check for quota limits.
964
- * @warning Ensure the quota exists in the database, or define `upsert` to create it when needed.
965
- *
966
- * @note This function allows for negative `requested_usage` values.
967
- *
968
- * @returns The updated quota record or a diagnostic if the quota was not found in the database or if the max retries have been exceeded.
969
- *
970
- * @docs
971
- */
972
- async increment({ query, requested_usage, upsert, }) {
973
- return this.limit_helper({
974
- query,
975
- requested_usage,
976
- upsert,
977
- collection: this.collection,
978
- check_limit: false,
979
- perform_increment: true,
980
- });
981
- }
982
- /**
983
- * Validates multiple quota limits and atomically increments usage if within bounds.
984
- * Handles interval resets automatically in a single database operation.
985
- *
986
- * This transaction based operation only commits changes if all quotas pass validation.
987
- *
988
- * @warning Ensure the quota exists in the database.
989
- *
990
- * @note This function does not allow for negative usage values, use {@link increment} for decrements.
991
- * @note This automatically increments the quota usage with the requested usage.
992
- *
993
- * @param limits The quota limits to validate and increment upon success, or roll back upon failure.
994
- *
995
- * @returns Success with updated quota info or validation/error details.
996
- *
997
- * @docs
998
- */
999
- async batch_limit({ limits }) {
1000
- // Throw invalid usage error when no limits are provided, dont return response.
1001
- if (limits.length === 0) {
1002
- throw new Error("No limits provided for batch_limit");
1003
- }
1004
- // Early validation
1005
- for (const item of limits) {
1006
- if (item.requested_usage < 0n) {
1007
- return {
1008
- success: false,
1009
- status: "invalid_usage",
1010
- failed_query: item.query,
1011
- error: `Negative 'requested_usage' (${item.requested_usage}) is not allowed in 'batch_limit'. Use 'increment' for decrements.`,
1012
- };
1013
- }
1014
- // Validate input
1015
- const val_input_res = this.validate_limit_helper_params({
1016
- requested_usage: item.requested_usage,
1017
- safety_ratio: item.safety_ratio,
1018
- upsert: item.upsert,
1019
- query: item.query,
1020
- });
1021
- if (val_input_res) {
1022
- return {
1023
- success: false,
1024
- status: val_input_res.status,
1025
- failed_query: item.query,
1026
- error: val_input_res.error
1027
- };
1028
- }
1029
- }
1030
- // Start transaction.
1031
- const transaction = await this.collection.start_transaction();
1032
- const results = [];
1033
- let active_limit = limits[0];
1034
- try {
1035
- for (const limit of limits) {
1036
- active_limit = limit;
1037
- const result = await this.limit_helper({
1038
- query: limit.query,
1039
- requested_usage: limit.requested_usage,
1040
- upsert: limit.upsert,
1041
- collection: transaction,
1042
- safety_ratio: limit.safety_ratio,
1043
- check_limit: limit.check_limit ?? true,
1044
- perform_increment: limit.perform_increment ?? true,
1045
- });
1046
- if (!result.success) {
1047
- await transaction.abort();
1048
- return {
1049
- ...result,
1050
- failed_query: limit.query,
1051
- };
1052
- }
1053
- results.push(result);
1054
- }
1055
- }
1056
- catch (error) {
1057
- await transaction.abort();
1058
- SystemError.create_detach({
1059
- owner: "volt.QuotaManager",
1060
- collection: this.system_error?.collection,
1061
- logger: this.system_error?.logger,
1062
- message: `Transaction failed: ${error instanceof Error ? error.message : String(error)}`,
1063
- details: {
1064
- failed_query: active_limit.query,
1065
- is_transaction: true,
1066
- },
1067
- });
1068
- return {
1069
- success: false,
1070
- status: "system_error",
1071
- failed_query: active_limit.query,
1072
- error: `Transaction failed: ${error instanceof Error ? error.message : String(error)}`,
1073
- };
1074
- }
1075
- // Commit with error handling; abort on failure to preserve atomicity.
1076
- try {
1077
- await transaction.commit();
1078
- }
1079
- catch (error) {
1080
- await transaction.abort();
1081
- SystemError.create_detach({
1082
- owner: "volt.QuotaManager",
1083
- collection: this.system_error?.collection,
1084
- logger: this.system_error?.logger,
1085
- message: `Transaction commit failed: ${error instanceof Error ? error.message : String(error)}`,
1086
- details: {
1087
- failed_query: active_limit.query,
1088
- is_transaction: true,
1089
- },
1090
- });
1091
- return {
1092
- success: false,
1093
- status: "system_error",
1094
- failed_query: active_limit.query,
1095
- error: `Transaction commit failed: ${error instanceof Error ? error.message : String(error)}`,
1096
- };
1097
- }
1098
- // Result.
1099
- return {
1100
- success: true,
1101
- status: "success",
1102
- results,
1103
- };
1104
- }
1105
- }
1106
- /**
1107
- * The quota manager module.
1108
- * This manager can be used to manage and enforce usage quotas for different resources,
1109
- * for instance limiting money spent on tokens, or for rate limiting purposes.
1110
- *
1111
- * For managing monetary quotas, it is advised to use a nano-scale amount system
1112
- * (smallest unit integer accounting). See {@link Quota.to_nano} for converting to nano scale.
1113
- */
1114
- (function (QuotaManager) {
1115
- // ----------------------------------------------------------------
1116
- // Constructor options.
1117
- // ----------------------------------------------------------------
1118
- /** Nested types for the {@link QuotaManager.Query} type. */
1119
- let Query;
1120
- (function (Query) {
1121
- /**
1122
- * Validate {@link QuotaManager.Query} at runtime.
1123
- * @returns An error message if the query is invalid, or undefined if it is valid.
1124
- */
1125
- function validate(query) {
1126
- // Validate query.
1127
- if (!query.uid?.trim()) {
1128
- return "Invalid uid: must be a non-empty string.";
1129
- }
1130
- if (!query.id?.trim()) {
1131
- return "Invalid id: must be a non-empty string.";
1132
- }
1133
- }
1134
- Query.validate = validate;
1135
- })(Query = QuotaManager.Query || (QuotaManager.Query = {}));
1136
- ;
1137
- /** Nested types for the {@link Quota} interface. */
1138
- let Quota;
1139
- (function (Quota) {
1140
- /** Nested types for the {@link Opts} interface. */
1141
- let Opts;
1142
- (function (Opts) {
1143
- /** The schema to validate quota {@link Opts} */
1144
- Opts.Schema = {
1145
- max: {
1146
- type: "bigint",
1147
- required: true,
1148
- },
1149
- interval: {
1150
- type: "number",
1151
- required: true,
1152
- },
1153
- };
1154
- /**
1155
- * Validate {@link Quota.Opts} at runtime.
1156
- * @returns An error message if the quota is invalid, or undefined if it is valid.
1157
- */
1158
- function validate(quota, collection) {
1159
- // Validate quota fields
1160
- if (typeof quota.max !== "bigint" || quota.max <= 0n) {
1161
- return `Invalid quota 'max': ${String(quota.max)}. Must be a positive bigint.`;
1162
- }
1163
- if (typeof quota.interval !== "number" || quota.interval <= 0 || !Number.isFinite(quota.interval)) {
1164
- return `Invalid quota 'interval': ${quota.interval}. Must be positive and finite.`;
1165
- }
1166
- else if (collection.ttl != null && quota.interval * 1000 >= collection.ttl) {
1167
- return `Invalid quota 'interval': ${quota.interval}. Must be less than the collection TTL of ${Math.ceil(collection.ttl / 1000)} seconds.`;
1168
- }
1169
- }
1170
- Opts.validate = validate;
1171
- })(Opts = Quota.Opts || (Quota.Opts = {}));
1172
- /**
1173
- * Helper function to convert a number or quota scales to a nano integer.
1174
- * For `Quota` options it only updates the `max` attribute.
1175
- * Internally using {@link QuotaManager.to_scaled_amount}.
1176
- *
1177
- * @note This should be used before saving / upserting the quota,
1178
- * this should not be applied on saved quota's
1179
- *
1180
- * @param q The number, quota or undefined to convert, undefined will simply return undefined again.
1181
- *
1182
- * @returns The scaled input type.
1183
- *
1184
- * @docs
1185
- */
1186
- function to_nano(q, opts) {
1187
- if (q == null)
1188
- return undefined;
1189
- else if (typeof q === "number" || typeof q === "bigint") {
1190
- return new SafeInt(q, { from_scale: SafeInt.Scale.Base, to_scale: SafeInt.Scale.Nano, round: opts?.round }).value();
1191
- }
1192
- return {
1193
- max: new SafeInt(q.max, { from_scale: SafeInt.Scale.Base, to_scale: SafeInt.Scale.Nano, round: opts?.round }).value(),
1194
- interval: q.interval,
1195
- };
1196
- }
1197
- Quota.to_nano = to_nano;
1198
- })(Quota = QuotaManager.Quota || (QuotaManager.Quota = {}));
1199
- /** Nested types for the {@link Document} type. */
1200
- let Document;
1201
- (function (Document) {
1202
- /** Nested types for the {@link Document.Opts} type. */
1203
- let Opts;
1204
- (function (Opts) {
1205
- /**
1206
- * Validate {@link Document.Opts} at runtime.
1207
- * @returns An error message if the quota is invalid, or undefined if it is valid.
1208
- */
1209
- function validate(quota, collection) {
1210
- let e;
1211
- if ((e = Query.validate(quota)))
1212
- return e;
1213
- if ((e = Quota.Opts.validate(quota, collection)))
1214
- return e;
1215
- }
1216
- Opts.validate = validate;
1217
- })(Opts = Document.Opts || (Document.Opts = {}));
1218
- })(Document = QuotaManager.Document || (QuotaManager.Document = {}));
1219
- })(QuotaManager || (QuotaManager = {}));