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