@blamejs/core 0.13.43 → 0.13.45

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/lib/mcp.js CHANGED
@@ -81,30 +81,30 @@ function parseRequest(body, opts) {
81
81
  try {
82
82
  parsed = typeof body === "string" ? safeJson.parse(body, { maxBytes: C.BYTES.mib(1) }) : body; // allow:JSON.parse — routed via safeJson.parse
83
83
  } catch (_e) {
84
- throw errorClass.factory("BAD_JSON",
84
+ throw errorClass.factory("mcp/bad-json",
85
85
  "mcp.parseRequest: body is not valid JSON");
86
86
  }
87
87
  if (!parsed || typeof parsed !== "object" || Array.isArray(parsed)) {
88
- throw errorClass.factory("BAD_ENVELOPE",
88
+ throw errorClass.factory("mcp/bad-envelope",
89
89
  "mcp.parseRequest: request must be a JSON-RPC object");
90
90
  }
91
91
  if (parsed.jsonrpc !== "2.0") {
92
- throw errorClass.factory("BAD_VERSION",
92
+ throw errorClass.factory("mcp/bad-version",
93
93
  "mcp.parseRequest: jsonrpc must be \"2.0\"");
94
94
  }
95
95
  if (typeof parsed.method !== "string" || parsed.method.length === 0 ||
96
96
  parsed.method.length > METHOD_NAME_MAX) {
97
- throw errorClass.factory("BAD_METHOD",
97
+ throw errorClass.factory("mcp/bad-method",
98
98
  "mcp.parseRequest: method must be a non-empty string under 256 bytes");
99
99
  }
100
100
  if (parsed.id !== undefined && parsed.id !== null &&
101
101
  typeof parsed.id !== "string" && typeof parsed.id !== "number") {
102
- throw errorClass.factory("BAD_ID",
102
+ throw errorClass.factory("mcp/bad-id",
103
103
  "mcp.parseRequest: id must be string, number, or null");
104
104
  }
105
105
  if (parsed.params !== undefined && parsed.params !== null &&
106
106
  typeof parsed.params !== "object") {
107
- throw errorClass.factory("BAD_PARAMS",
107
+ throw errorClass.factory("mcp/bad-params",
108
108
  "mcp.parseRequest: params must be object or array");
109
109
  }
110
110
  return parsed;
@@ -167,7 +167,7 @@ function _readBodyBuffered(req, maxBytes, errorClass) {
167
167
  try { collector.push(chunk); }
168
168
  catch (_e) {
169
169
  req.destroy();
170
- reject(errorClass.factory("BODY_TOO_LARGE",
170
+ reject(errorClass.factory("mcp/body-too-large",
171
171
  "mcp: request body exceeds " + maxBytes + " bytes"));
172
172
  }
173
173
  });
@@ -178,17 +178,17 @@ function _readBodyBuffered(req, maxBytes, errorClass) {
178
178
 
179
179
  function _checkRedirectUri(uri, allowlist, errorClass) {
180
180
  if (typeof uri !== "string") {
181
- throw errorClass.factory("BAD_REDIRECT_URI",
181
+ throw errorClass.factory("mcp/bad-redirect-uri",
182
182
  "mcp: redirect_uri must be a string");
183
183
  }
184
184
  if (!Array.isArray(allowlist) || allowlist.indexOf(uri) === -1) {
185
- throw errorClass.factory("REDIRECT_URI_REFUSED",
185
+ throw errorClass.factory("mcp/redirect-uri-refused",
186
186
  "mcp: redirect_uri not in allowlist (OAuth 2.1 / RFC 9700 sec 4.1.1)");
187
187
  }
188
188
  var parsed;
189
189
  try { parsed = safeUrl.parse(uri); }
190
190
  catch (_e) {
191
- throw errorClass.factory("BAD_REDIRECT_URI",
191
+ throw errorClass.factory("mcp/bad-redirect-uri",
192
192
  "mcp: redirect_uri did not parse");
193
193
  }
194
194
  var isHttps = parsed.protocol === "https:";
@@ -205,7 +205,7 @@ function _checkRedirectUri(uri, allowlist, errorClass) {
205
205
  }
206
206
  var isLocal = rawHost === "localhost" || rawHost === "127.0.0.1" || rawHost === "::1";
207
207
  if (!isHttps && !isLocal) {
208
- throw errorClass.factory("INSECURE_REDIRECT_URI",
208
+ throw errorClass.factory("mcp/insecure-redirect-uri",
209
209
  "mcp: redirect_uri must be HTTPS (or localhost; RFC 9700 sec 4.1.1)");
210
210
  }
211
211
  }
@@ -257,7 +257,7 @@ function serverGuard(opts) {
257
257
  var requireBearer = opts.requireBearer !== false;
258
258
  var verifyBearer = opts.verifyBearer || null;
259
259
  if (requireBearer && typeof verifyBearer !== "function") {
260
- throw errorClass.factory("BAD_OPTS",
260
+ throw errorClass.factory("mcp/bad-opts",
261
261
  "mcp.serverGuard: verifyBearer required when requireBearer=true");
262
262
  }
263
263
  var redirectUriAllowlist = Array.isArray(opts.redirectUriAllowlist)
@@ -266,7 +266,7 @@ function serverGuard(opts) {
266
266
  var registerClientAllowlist = typeof opts.registerClientAllowlist === "function"
267
267
  ? opts.registerClientAllowlist : null;
268
268
  if (allowDynamicRegister && !registerClientAllowlist) {
269
- throw errorClass.factory("BAD_OPTS",
269
+ throw errorClass.factory("mcp/bad-opts",
270
270
  "mcp.serverGuard: allowDynamicRegister=true requires registerClientAllowlist function");
271
271
  }
272
272
  var toolAllowlist = Array.isArray(opts.toolAllowlist) ? opts.toolAllowlist : null;
@@ -123,11 +123,11 @@ function create(opts) {
123
123
  ], "middleware.requireStepUp");
124
124
 
125
125
  if (!opts.requirement || typeof opts.requirement !== "object") {
126
- throw new AuthError("auth-stepUp/bad-requirement",
126
+ throw new AuthError("auth-step-up/bad-requirement",
127
127
  "middleware.requireStepUp: opts.requirement must be an object");
128
128
  }
129
129
  validateOpts.optionalFunction(opts.getClaims,
130
- "middleware.requireStepUp: getClaims", AuthError, "auth-stepUp/bad-opt");
130
+ "middleware.requireStepUp: getClaims", AuthError, "auth-step-up/bad-opt");
131
131
 
132
132
  var realm = (typeof opts.realm === "string" && opts.realm.length > 0)
133
133
  ? opts.realm : "api";
@@ -146,7 +146,7 @@ function create(opts) {
146
146
  // on the first hot-path request.
147
147
  var probe = stepUp().evaluate({ claims: { acr: "0" }, requirement: opts.requirement });
148
148
  if (probe.error === "bad_requirement" || probe.error === "unknown_acr") {
149
- throw new AuthError("auth-stepUp/bad-requirement",
149
+ throw new AuthError("auth-step-up/bad-requirement",
150
150
  "middleware.requireStepUp: " + (probe.reason || probe.error));
151
151
  }
152
152
 
@@ -20,6 +20,7 @@ var NetworkTlsError = defineClass("NetworkTlsError", { alwaysPermanent: true });
20
20
  var observability = lazyRequire(function () { return require("./observability"); });
21
21
  var audit = lazyRequire(function () { return require("./audit"); });
22
22
  var networkDns = lazyRequire(function () { return require("./network-dns"); });
23
+ var httpClient = lazyRequire(function () { return require("./http-client"); });
23
24
  var asn1 = require("./asn1-der");
24
25
 
25
26
  // STATE.tlsKeyShares is initialized to the default PQC group list at
@@ -1194,9 +1195,10 @@ function evaluateOcspResponse(ocspDer, opts) {
1194
1195
  //
1195
1196
  // Constructs a DER-encoded OCSPRequest for a single (leafCertDer,
1196
1197
  // issuerCertDer) pair, optionally with an RFC 8954 nonce extension.
1197
- // Operators send the returned `requestDer` to the OCSP responder URL
1198
- // (e.g. via b.httpClient with `Content-Type: application/ocsp-request`)
1199
- // and pass `nonce` to `ocsp.evaluate(responseDer, { expectedNonce })`
1198
+ // `ocsp.fetch` composes this with `b.httpClient` to POST the request to
1199
+ // the cert's responder and return a validated response; operators who
1200
+ // need the raw request (custom transport, batched requests) call this
1201
+ // directly and pass `nonce` to `ocsp.evaluate(responseDer, { expectedNonce })`
1200
1202
  // to defend against replay attacks.
1201
1203
  //
1202
1204
  // Nonce DEFAULT ON — defense in depth. RFC 6960 §4.4.1 marks nonce
@@ -1291,8 +1293,10 @@ function buildOcspRequest(opts) {
1291
1293
  // framework that touches SHA-1" need a signal. Emit an audit row
1292
1294
  // on every OCSP request build so the algorithm choice is visible
1293
1295
  // in the chain.
1294
- var nameHash = nodeCrypto.createHash("sha1").update(iss.issuerNameDer).digest();
1295
- var keyHash = nodeCrypto.createHash("sha1").update(iss.issuerKey).digest();
1296
+ // lgtm[js/weak-cryptographic-algorithm] RFC 6960 §4.1.1 CertID lookup hash over the PUBLIC issuer name; a name/key lookup, not an integrity or secrecy operation. SHA-256 CertIDs are §4.3-optional and rejected by most responders.
1297
+ var nameHash = nodeCrypto.createHash("sha1").update(iss.issuerNameDer).digest(); // lgtm[js/weak-cryptographic-algorithm]
1298
+ // lgtm[js/weak-cryptographic-algorithm] — RFC 6960 §4.1.1 CertID lookup hash over the PUBLIC issuer key; a name/key lookup, not an integrity or secrecy operation.
1299
+ var keyHash = nodeCrypto.createHash("sha1").update(iss.issuerKey).digest(); // lgtm[js/weak-cryptographic-algorithm]
1296
1300
  setImmediate(function () {
1297
1301
  try {
1298
1302
  var auditMod = require("./audit"); // allow:inline-require — circular-load defense (audit imports network-tls)
@@ -1339,6 +1343,77 @@ function buildOcspRequest(opts) {
1339
1343
  return { requestDer: requestDer, nonce: nonceBytes };
1340
1344
  }
1341
1345
 
1346
+ // _ocspResponderUrl — pull the OCSP responder URL out of a cert's
1347
+ // Authority Information Access extension. node:crypto exposes it as a
1348
+ // multi-line string ("OCSP - URI:http://...\nCA Issuers - URI:...\n").
1349
+ function _ocspResponderUrl(x509) {
1350
+ var ia = x509 && x509.infoAccess;
1351
+ if (typeof ia !== "string") return null;
1352
+ var m = ia.match(/OCSP\s*-\s*URI:(\S+)/i);
1353
+ return m ? m[1].trim() : null;
1354
+ }
1355
+
1356
+ // fetch — POST a freshly-built OCSPRequest to the cert's responder and
1357
+ // return the validated, known-good response bytes. Composes buildRequest +
1358
+ // b.httpClient + evaluate, completing the server-side-stapling fetch path
1359
+ // (the response is what a TLS server staples via its 'OCSPRequest' handler).
1360
+ // The responder URL is taken from the leaf cert's AIA extension unless
1361
+ // opts.responderUrl overrides it. Throws TlsTrustError on any failure
1362
+ // (no responder, transport error, non-good certStatus, signature mismatch);
1363
+ // callers that staple should treat a throw as "no staple this cycle".
1364
+ async function fetchOcspResponse(opts) {
1365
+ opts = opts || {};
1366
+ if (typeof opts.leafPem !== "string" || typeof opts.issuerPem !== "string") {
1367
+ throw new TlsTrustError("tls/ocsp-bad-input",
1368
+ "ocsp.fetch: opts.leafPem and opts.issuerPem (PEM strings) are required");
1369
+ }
1370
+ var leafX, issuerX;
1371
+ try {
1372
+ leafX = new nodeCrypto.X509Certificate(opts.leafPem);
1373
+ issuerX = new nodeCrypto.X509Certificate(opts.issuerPem);
1374
+ } catch (e) {
1375
+ throw new TlsTrustError("tls/ocsp-bad-cert",
1376
+ "ocsp.fetch: could not parse leaf/issuer PEM: " + ((e && e.message) || String(e)));
1377
+ }
1378
+ var responderUrl = opts.responderUrl || _ocspResponderUrl(leafX);
1379
+ if (!responderUrl) {
1380
+ throw new TlsTrustError("tls/ocsp-no-responder",
1381
+ "ocsp.fetch: cert has no AIA OCSP responder URL; pass opts.responderUrl");
1382
+ }
1383
+ var built = buildOcspRequest({
1384
+ leafCertDer: leafX.raw, issuerCertDer: issuerX.raw,
1385
+ nonce: opts.nonce, nonceLen: opts.nonceLen,
1386
+ });
1387
+ var res;
1388
+ try {
1389
+ res = await httpClient().request({
1390
+ url: responderUrl,
1391
+ method: "POST",
1392
+ headers: { "content-type": "application/ocsp-request", "accept": "application/ocsp-response" },
1393
+ body: built.requestDer,
1394
+ responseMode: "buffer",
1395
+ timeoutMs: opts.timeoutMs || C.TIME.seconds(10),
1396
+ });
1397
+ } catch (e) {
1398
+ throw new TlsTrustError("tls/ocsp-fetch-failed",
1399
+ "ocsp.fetch: responder request to " + responderUrl + " failed: " + ((e && e.message) || String(e)));
1400
+ }
1401
+ if (res.status !== 200 || !Buffer.isBuffer(res.body) || res.body.length === 0) {
1402
+ throw new TlsTrustError("tls/ocsp-fetch-bad-status",
1403
+ "ocsp.fetch: responder returned status " + res.status + " with an empty/non-buffer body");
1404
+ }
1405
+ var evald = evaluateOcspResponse(res.body, {
1406
+ issuerPem: opts.issuerPem,
1407
+ serialHex: opts.serialHex || null,
1408
+ expectedNonce: opts.nonce === false ? null : built.nonce,
1409
+ });
1410
+ if (!evald.ok) {
1411
+ throw new TlsTrustError("tls/ocsp-not-good",
1412
+ "ocsp.fetch: response is not good: " + (evald.errors || []).join("; "));
1413
+ }
1414
+ return { ocspDer: res.body, evaluation: evald, responderUrl: responderUrl };
1415
+ }
1416
+
1342
1417
  var ocsp = Object.freeze({
1343
1418
  // Connect with OCSP requested. Returns { authorized, ocspBytes,
1344
1419
  // peerCert }. requireStapled: true makes empty / not-stapled responses
@@ -1379,6 +1454,7 @@ var ocsp = Object.freeze({
1379
1454
  },
1380
1455
  parseResponse: parseOcspResponse,
1381
1456
  evaluate: evaluateOcspResponse,
1457
+ fetch: fetchOcspResponse,
1382
1458
  // buildRequest — construct a DER-encoded OCSPRequest for a single
1383
1459
  // (leafCertDer, issuerCertDer) pair. RFC 8954 nonce extension is ON
1384
1460
  // by default (16 random bytes; opts.nonceLen overrides within RFC
package/lib/sec-cyber.js CHANGED
@@ -91,13 +91,13 @@ function _addBusinessDays(startMs, days) {
91
91
 
92
92
  function eightKArtifact(opts) {
93
93
  if (!opts || typeof opts !== "object") {
94
- throw SecCyberError.factory("BAD_OPTS",
94
+ throw SecCyberError.factory("sec-cyber/bad-opts",
95
95
  "secCyber.eightKArtifact: opts required");
96
96
  }
97
97
  validateOpts.requireNonEmptyString(opts.incidentId,
98
98
  "secCyber.eightKArtifact: incidentId", SecCyberError, "BAD_INCIDENT_ID");
99
99
  if (!opts.registrant || typeof opts.registrant !== "object") {
100
- throw SecCyberError.factory("BAD_REGISTRANT",
100
+ throw SecCyberError.factory("sec-cyber/bad-registrant",
101
101
  "secCyber.eightKArtifact: registrant object required");
102
102
  }
103
103
  validateOpts.requireNonEmptyString(opts.registrant.name,
@@ -110,7 +110,7 @@ function eightKArtifact(opts) {
110
110
  "secCyber.eightKArtifact: materialityDeterminedAt", SecCyberError, "BAD_MAT_AT");
111
111
 
112
112
  if (FINDINGS.indexOf(opts.materialityFinding) === -1) {
113
- throw SecCyberError.factory("BAD_FINDING",
113
+ throw SecCyberError.factory("sec-cyber/bad-finding",
114
114
  "secCyber.eightKArtifact: materialityFinding must be one of " + FINDINGS.join(", "));
115
115
  }
116
116
  validateOpts.requireNonEmptyString(opts.materialityReasoning,
package/lib/sse.js CHANGED
@@ -89,7 +89,7 @@ function _validateRetry(retry, errorClass) {
89
89
  if (retry === undefined || retry === null) return null;
90
90
  if (typeof retry !== "number" || !isFinite(retry) || retry < 0 ||
91
91
  Math.floor(retry) !== retry) {
92
- throw errorClass.factory("BAD_RETRY",
92
+ throw errorClass.factory("sse/bad-retry",
93
93
  "sse.send: retry must be a non-negative finite integer (got " +
94
94
  JSON.stringify(retry) + ")");
95
95
  }
@@ -98,14 +98,14 @@ function _validateRetry(retry, errorClass) {
98
98
 
99
99
  function _refuseInjection(field, value, errorClass) {
100
100
  if (typeof value !== "string") {
101
- throw errorClass.factory("BAD_FIELD",
101
+ throw errorClass.factory("sse/bad-field",
102
102
  "sse.send: " + field + " must be a string");
103
103
  }
104
104
  // Length-bound BEFORE the regex test — _capField applies a tighter
105
105
  // cap further along, but the regex itself runs against the full
106
106
  // value so we bound here too.
107
107
  if (value.length > MAX_DATA_BYTES) {
108
- throw errorClass.factory("FIELD_TOO_LARGE",
108
+ throw errorClass.factory("sse/field-too-large",
109
109
  "sse.send: " + field + " too large for injection scan");
110
110
  }
111
111
  if (INJECTION_RE.test(value)) { // allow:regex-no-length-cap — value length capped above
@@ -114,7 +114,7 @@ function _refuseInjection(field, value, errorClass) {
114
114
  outcome: "denied",
115
115
  metadata: { field: field, length: value.length },
116
116
  });
117
- throw errorClass.factory("INJECTION",
117
+ throw errorClass.factory("sse/injection",
118
118
  "sse.send: " + field + " contains LF/CR/NUL — refused " +
119
119
  "(CVE-2026-33128 / 29085 / 44217 class)");
120
120
  }
@@ -130,7 +130,7 @@ var MAX_DATA_BYTES = C.BYTES.mib(1);
130
130
  function _capField(field, value, capBytes, errorClass) {
131
131
  var len = Buffer.byteLength(value, "utf8");
132
132
  if (len > capBytes) {
133
- throw errorClass.factory("FIELD_TOO_LARGE",
133
+ throw errorClass.factory("sse/field-too-large",
134
134
  "sse.send: " + field + " exceeds cap (" + len + " > " +
135
135
  capBytes + " bytes)");
136
136
  }
@@ -139,7 +139,7 @@ function _capField(field, value, capBytes, errorClass) {
139
139
  function serializeEvent(opts, errorClass) {
140
140
  errorClass = errorClass || SseError;
141
141
  if (!opts || typeof opts !== "object") {
142
- throw errorClass.factory("BAD_OPTS", "sse.serializeEvent: opts required");
142
+ throw errorClass.factory("sse/bad-opts", "sse.serializeEvent: opts required");
143
143
  }
144
144
  var out = "";
145
145
  // Field order: id, event, retry, data — matches the framework's
@@ -162,7 +162,7 @@ function serializeEvent(opts, errorClass) {
162
162
  }
163
163
  if (opts.data !== undefined && opts.data !== null) {
164
164
  if (typeof opts.data !== "string") {
165
- throw errorClass.factory("BAD_FIELD",
165
+ throw errorClass.factory("sse/bad-field",
166
166
  "sse.send: data must be a string");
167
167
  }
168
168
  _capField("data", opts.data, MAX_DATA_BYTES, errorClass);
@@ -174,7 +174,7 @@ function serializeEvent(opts, errorClass) {
174
174
  outcome: "denied",
175
175
  metadata: { field: "data", length: opts.data.length, char: "cr-or-nul" },
176
176
  });
177
- throw errorClass.factory("INJECTION",
177
+ throw errorClass.factory("sse/injection",
178
178
  "sse.send: data contains CR or NUL — refused");
179
179
  }
180
180
  var lines = opts.data.split("\n");
@@ -189,11 +189,11 @@ function serializeEvent(opts, errorClass) {
189
189
 
190
190
  function _validateComment(text, errorClass) {
191
191
  if (typeof text !== "string") {
192
- throw errorClass.factory("BAD_FIELD",
192
+ throw errorClass.factory("sse/bad-field",
193
193
  "sse.comment: text must be a string");
194
194
  }
195
195
  if (text.length > MAX_DATA_BYTES) {
196
- throw errorClass.factory("FIELD_TOO_LARGE",
196
+ throw errorClass.factory("sse/field-too-large",
197
197
  "sse.comment: text too large for injection scan");
198
198
  }
199
199
  if (INJECTION_RE.test(text)) { // allow:regex-no-length-cap — text length capped above
@@ -202,7 +202,7 @@ function _validateComment(text, errorClass) {
202
202
  outcome: "denied",
203
203
  metadata: { field: "comment", length: text.length },
204
204
  });
205
- throw errorClass.factory("INJECTION",
205
+ throw errorClass.factory("sse/injection",
206
206
  "sse.comment: text contains LF/CR/NUL — refused");
207
207
  }
208
208
  }
@@ -224,14 +224,14 @@ function create(req, res, opts) {
224
224
  opts = opts || {};
225
225
  var errorClass = opts.errorClass || SseError;
226
226
  if (!res || typeof res.write !== "function" || typeof res.end !== "function") {
227
- throw errorClass.factory("BAD_RES",
227
+ throw errorClass.factory("sse/bad-res",
228
228
  "sse.create: res must be a writable response stream");
229
229
  }
230
230
  var heartbeatMs = opts.heartbeatMs;
231
231
  if (heartbeatMs === undefined) heartbeatMs = C.TIME.seconds(15);
232
232
  if (typeof heartbeatMs !== "number" || !isFinite(heartbeatMs) ||
233
233
  heartbeatMs < 0 || Math.floor(heartbeatMs) !== heartbeatMs) {
234
- throw errorClass.factory("BAD_OPTS",
234
+ throw errorClass.factory("sse/bad-opts",
235
235
  "sse.create: heartbeatMs must be a non-negative integer ms (got " +
236
236
  JSON.stringify(heartbeatMs) + ")");
237
237
  }
@@ -264,7 +264,7 @@ function create(req, res, opts) {
264
264
 
265
265
  function _writeRaw(s) {
266
266
  if (closed) {
267
- throw errorClass.factory("CLOSED",
267
+ throw errorClass.factory("sse/closed",
268
268
  "sse.send: channel closed");
269
269
  }
270
270
  res.write(s);
package/lib/tcpa-10dlc.js CHANGED
@@ -72,11 +72,11 @@ var records = new Map(); // phoneE164 → record
72
72
 
73
73
  function recordConsent(opts) {
74
74
  if (!opts || typeof opts !== "object") {
75
- throw Tcpa10dlcError.factory("BAD_OPTS",
75
+ throw Tcpa10dlcError.factory("tcpa-10dlc/bad-opts",
76
76
  "tcpa10dlc.recordConsent: opts required");
77
77
  }
78
78
  if (typeof opts.phoneE164 !== "string" || !E164_RE.test(opts.phoneE164)) {
79
- throw Tcpa10dlcError.factory("BAD_PHONE",
79
+ throw Tcpa10dlcError.factory("tcpa-10dlc/bad-phone",
80
80
  "tcpa10dlc.recordConsent: phoneE164 must match " + E164_RE);
81
81
  }
82
82
  validateOpts.requireNonEmptyString(opts.brand,
@@ -86,7 +86,7 @@ function recordConsent(opts) {
86
86
  validateOpts.requireNonEmptyString(opts.formUrl,
87
87
  "tcpa10dlc.recordConsent: formUrl", Tcpa10dlcError, "BAD_FORM_URL");
88
88
  if (DISCLOSURE_PARTIES.indexOf(opts.disclosurePartyKind) === -1) {
89
- throw Tcpa10dlcError.factory("BAD_DISCLOSURE_PARTY",
89
+ throw Tcpa10dlcError.factory("tcpa-10dlc/bad-disclosure-party",
90
90
  "tcpa10dlc.recordConsent: disclosurePartyKind must be one of " +
91
91
  DISCLOSURE_PARTIES.join(", "));
92
92
  }
@@ -133,12 +133,12 @@ function lookup(phoneE164) {
133
133
 
134
134
  function revoke(phoneE164, reason) {
135
135
  if (typeof phoneE164 !== "string" || !E164_RE.test(phoneE164)) {
136
- throw Tcpa10dlcError.factory("BAD_PHONE",
136
+ throw Tcpa10dlcError.factory("tcpa-10dlc/bad-phone",
137
137
  "tcpa10dlc.revoke: phoneE164 must match " + E164_RE);
138
138
  }
139
139
  var existing = records.get(phoneE164);
140
140
  if (!existing) {
141
- throw Tcpa10dlcError.factory("NO_RECORD",
141
+ throw Tcpa10dlcError.factory("tcpa-10dlc/no-record",
142
142
  "tcpa10dlc.revoke: no consent record for " + phoneE164);
143
143
  }
144
144
  if (existing.revoked) {
@@ -113,23 +113,23 @@ function create(opts) {
113
113
 
114
114
  if (!opts.db || typeof opts.db.from !== "function" ||
115
115
  typeof opts.db.getTableMetadata !== "function") {
116
- throw new TenantQuotaError("tenantQuota/bad-db",
116
+ throw new TenantQuotaError("tenant-quota/bad-db",
117
117
  "tenantQuota.create: opts.db must be the framework's b.db namespace");
118
118
  }
119
119
  validateOpts.requireNonEmptyString(opts.tenantField,
120
- "tenantQuota.create: tenantField", TenantQuotaError, "tenantQuota/bad-field");
120
+ "tenantQuota.create: tenantField", TenantQuotaError, "tenant-quota/bad-field");
121
121
 
122
122
  var defaultBytesCap = (opts.defaultBytesCap == null)
123
123
  ? DEFAULT_BYTES_CAP
124
124
  : opts.defaultBytesCap;
125
125
  if (typeof defaultBytesCap !== "number" || !isFinite(defaultBytesCap) || defaultBytesCap <= 0) {
126
- throw new TenantQuotaError("tenantQuota/bad-cap",
126
+ throw new TenantQuotaError("tenant-quota/bad-cap",
127
127
  "tenantQuota.create: defaultBytesCap must be a positive finite number");
128
128
  }
129
129
 
130
130
  var perTenantBytesCap = opts.perTenantBytesCap || {};
131
131
  if (typeof perTenantBytesCap !== "object" || Array.isArray(perTenantBytesCap)) {
132
- throw new TenantQuotaError("tenantQuota/bad-per-tenant",
132
+ throw new TenantQuotaError("tenant-quota/bad-per-tenant",
133
133
  "tenantQuota.create: perTenantBytesCap must be a plain object {tenantId: bytes}");
134
134
  }
135
135
  // Validate every per-tenant override at config time so a typo
@@ -138,7 +138,7 @@ function create(opts) {
138
138
  for (var pi = 0; pi < ptKeys.length; pi++) {
139
139
  var v = perTenantBytesCap[ptKeys[pi]];
140
140
  if (typeof v !== "number" || !isFinite(v) || v <= 0) {
141
- throw new TenantQuotaError("tenantQuota/bad-per-tenant",
141
+ throw new TenantQuotaError("tenant-quota/bad-per-tenant",
142
142
  "tenantQuota.create: perTenantBytesCap['" + ptKeys[pi] +
143
143
  "'] must be a positive finite number");
144
144
  }
@@ -147,7 +147,7 @@ function create(opts) {
147
147
  var auditOn = opts.audit !== false;
148
148
  var cacheTtlMs = (opts.cacheTtlMs == null) ? DEFAULT_CACHE_TTL_MS : opts.cacheTtlMs;
149
149
  if (typeof cacheTtlMs !== "number" || !isFinite(cacheTtlMs) || cacheTtlMs < 0) {
150
- throw new TenantQuotaError("tenantQuota/bad-ttl",
150
+ throw new TenantQuotaError("tenant-quota/bad-ttl",
151
151
  "tenantQuota.create: cacheTtlMs must be a non-negative finite number");
152
152
  }
153
153
 
@@ -230,7 +230,7 @@ function create(opts) {
230
230
 
231
231
  async function snapshot(tenantId) {
232
232
  validateOpts.requireNonEmptyString(tenantId,
233
- "tenantQuota.snapshot: tenantId", TenantQuotaError, "tenantQuota/bad-tenant");
233
+ "tenantQuota.snapshot: tenantId", TenantQuotaError, "tenant-quota/bad-tenant");
234
234
  var now = Date.now();
235
235
  var cached = cache.get(tenantId);
236
236
  var bytesUsed;
@@ -258,7 +258,7 @@ function create(opts) {
258
258
  bytesCap: snap.bytesCap,
259
259
  });
260
260
  _emitMetric("tenant.quota.exceeded", 1);
261
- throw new TenantQuotaError("tenantQuota/exceeded",
261
+ throw new TenantQuotaError("tenant-quota/exceeded",
262
262
  "tenantQuota.assert: tenant '" + tenantId + "' is at " +
263
263
  snap.bytesUsed + " of " + snap.bytesCap + " bytes; insert refused");
264
264
  }
@@ -344,21 +344,21 @@ function budget(opts) {
344
344
  ], "tenantQuota.budget");
345
345
 
346
346
  validateOpts.requireNonEmptyString(opts.tenantField,
347
- "tenantQuota.budget: tenantField", TenantQuotaError, "tenantQuota/bad-field");
347
+ "tenantQuota.budget: tenantField", TenantQuotaError, "tenant-quota/bad-field");
348
348
 
349
349
  var qpsCap = (opts.perTenantQpsCap == null) ? DEFAULT_QPS_CAP : opts.perTenantQpsCap;
350
350
  if (typeof qpsCap !== "number" || !isFinite(qpsCap) || qpsCap <= 0) {
351
- throw new TenantQuotaError("tenantQuota/bad-qps",
351
+ throw new TenantQuotaError("tenant-quota/bad-qps",
352
352
  "tenantQuota.budget: perTenantQpsCap must be a positive finite number");
353
353
  }
354
354
  var rowsCap = (opts.perTenantTotalRowsRead == null) ? DEFAULT_ROWS_READ : opts.perTenantTotalRowsRead;
355
355
  if (typeof rowsCap !== "number" || !isFinite(rowsCap) || rowsCap <= 0) {
356
- throw new TenantQuotaError("tenantQuota/bad-rows",
356
+ throw new TenantQuotaError("tenant-quota/bad-rows",
357
357
  "tenantQuota.budget: perTenantTotalRowsRead must be a positive finite number");
358
358
  }
359
359
  var windowMs = (opts.window == null) ? DEFAULT_WINDOW_MS : opts.window;
360
360
  if (typeof windowMs !== "number" || !isFinite(windowMs) || windowMs <= 0) {
361
- throw new TenantQuotaError("tenantQuota/bad-window",
361
+ throw new TenantQuotaError("tenant-quota/bad-window",
362
362
  "tenantQuota.budget: window must be a positive finite number");
363
363
  }
364
364
  var auditOn = opts.audit !== false;
@@ -393,7 +393,7 @@ function budget(opts) {
393
393
 
394
394
  function observe(tenantId, info) {
395
395
  validateOpts.requireNonEmptyString(tenantId,
396
- "tenantQuota.budget.observe: tenantId", TenantQuotaError, "tenantQuota/bad-tenant");
396
+ "tenantQuota.budget.observe: tenantId", TenantQuotaError, "tenant-quota/bad-tenant");
397
397
  info = info || {};
398
398
  var rowsRead = (typeof info.rowsRead === "number" && info.rowsRead >= 0) ? info.rowsRead : 0;
399
399
  var now = Date.now();
@@ -411,7 +411,7 @@ function budget(opts) {
411
411
  windowMs: windowMs,
412
412
  });
413
413
  _emitMetric("tenant.budget.exceeded", 1);
414
- throw new TenantQuotaError("tenantQuota/budget-exceeded",
414
+ throw new TenantQuotaError("tenant-quota/budget-exceeded",
415
415
  "tenantQuota.budget: tenant '" + tenantId + "' exceeded budget " +
416
416
  "(calls=" + c.calls + "/" + maxCalls + ", rowsRead=" + c.rowsRead +
417
417
  "/" + rowsCap + ", windowMs=" + windowMs + ")");
@@ -486,7 +486,7 @@ function budget(opts) {
486
486
  */
487
487
  function instrumentQuery(opts) {
488
488
  if (!opts || typeof opts !== "object") {
489
- throw new TenantQuotaError("tenantQuota/bad-instr",
489
+ throw new TenantQuotaError("tenant-quota/bad-instr",
490
490
  "tenantQuota.instrumentQuery: opts object is required");
491
491
  }
492
492
  validateOpts(opts, [
@@ -494,13 +494,13 @@ function instrumentQuery(opts) {
494
494
  ], "tenantQuota.instrumentQuery");
495
495
 
496
496
  if (!Array.isArray(opts.rows)) {
497
- throw new TenantQuotaError("tenantQuota/bad-rows",
497
+ throw new TenantQuotaError("tenant-quota/bad-rows",
498
498
  "tenantQuota.instrumentQuery: rows must be an array");
499
499
  }
500
500
  validateOpts.requireNonEmptyString(opts.tenantField,
501
- "tenantQuota.instrumentQuery: tenantField", TenantQuotaError, "tenantQuota/bad-field");
501
+ "tenantQuota.instrumentQuery: tenantField", TenantQuotaError, "tenant-quota/bad-field");
502
502
  validateOpts.requireNonEmptyString(opts.tenantId,
503
- "tenantQuota.instrumentQuery: tenantId", TenantQuotaError, "tenantQuota/bad-tenant");
503
+ "tenantQuota.instrumentQuery: tenantId", TenantQuotaError, "tenant-quota/bad-tenant");
504
504
  var auditOn = opts.audit !== false;
505
505
 
506
506
  var crossover = [];
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@blamejs/core",
3
- "version": "0.13.43",
3
+ "version": "0.13.45",
4
4
  "description": "The Node framework that owns its stack.",
5
5
  "license": "Apache-2.0",
6
6
  "author": "blamejs contributors",
package/sbom.cdx.json CHANGED
@@ -2,10 +2,10 @@
2
2
  "$schema": "http://cyclonedx.org/schema/bom-1.5.schema.json",
3
3
  "bomFormat": "CycloneDX",
4
4
  "specVersion": "1.5",
5
- "serialNumber": "urn:uuid:d368c000-a7b7-493e-bb55-a28c08371f7e",
5
+ "serialNumber": "urn:uuid:4f2a624b-9183-4e8a-8807-eb827d32fb2b",
6
6
  "version": 1,
7
7
  "metadata": {
8
- "timestamp": "2026-05-30T00:14:53.721Z",
8
+ "timestamp": "2026-05-30T01:55:57.060Z",
9
9
  "lifecycles": [
10
10
  {
11
11
  "phase": "build"
@@ -19,14 +19,14 @@
19
19
  }
20
20
  ],
21
21
  "component": {
22
- "bom-ref": "@blamejs/core@0.13.43",
22
+ "bom-ref": "@blamejs/core@0.13.45",
23
23
  "type": "application",
24
24
  "name": "blamejs",
25
- "version": "0.13.43",
25
+ "version": "0.13.45",
26
26
  "scope": "required",
27
27
  "author": "blamejs contributors",
28
28
  "description": "The Node framework that owns its stack.",
29
- "purl": "pkg:npm/%40blamejs/core@0.13.43",
29
+ "purl": "pkg:npm/%40blamejs/core@0.13.45",
30
30
  "properties": [],
31
31
  "externalReferences": [
32
32
  {
@@ -54,7 +54,7 @@
54
54
  "components": [],
55
55
  "dependencies": [
56
56
  {
57
- "ref": "@blamejs/core@0.13.43",
57
+ "ref": "@blamejs/core@0.13.45",
58
58
  "dependsOn": []
59
59
  }
60
60
  ]