@blamejs/core 0.12.38 → 0.12.39
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/CHANGELOG.md +2 -0
- package/README.md +1 -0
- package/index.js +1 -0
- package/lib/vc.js +328 -0
- package/package.json +1 -1
- package/sbom.cdx.json +6 -6
package/CHANGELOG.md
CHANGED
|
@@ -8,6 +8,8 @@ upgrading across more than a few patches at a time.
|
|
|
8
8
|
|
|
9
9
|
## v0.12.x
|
|
10
10
|
|
|
11
|
+
- v0.12.39 (2026-05-24) — **`b.vc` — W3C Verifiable Credentials 2.0 (issue / verify, JOSE + COSE securing).** Issue and verify W3C Verifiable Credentials (VC Data Model 2.0, a W3C Recommendation) secured per Securing Verifiable Credentials using JOSE and COSE (VC-JOSE-COSE, also a W3C Recommendation, May 2025). A verifiable credential is a tamper-evident, signed set of claims an issuer makes about a subject — a diploma, a membership, a license, an age assertion. Two securing mechanisms are supported, both signing the credential itself (no JWT/CWT claims wrapper): JOSE produces a compact JWS with the vc+jwt media type, signed with ES256/384/512 or EdDSA; COSE produces a COSE_Sign1 (application/vc+cose) over b.cose, which also accepts ML-DSA-87 for PQC-forward deployments. b.vc.verify auto-detects the form from the input, requires an algorithm allowlist, always refuses the JOSE none algorithm, re-checks the VCDM 2.0 structural rules, and enforces the validFrom / validUntil window. This is the W3C credential model, distinct from the IETF SD-JWT VC already at b.auth.sdJwtVc. Composes b.cose; no new runtime dependency. **Added:** *`b.vc.issue(credential, opts)` / `b.vc.verify(secured, opts)`* — `issue` validates the credential against the VCDM 2.0 structural rules (the `credentials/v2` context first, a `VerifiableCredential` type, an issuer, a credential subject) and signs it: `securing: "jose"` returns a compact JWS string (`typ` header `vc+jwt`), `securing: "cose"` returns COSE_Sign1 bytes (`typ` header `application/vc+cose`, content type `application/vc`) via `b.cose`. The credential is the exact signed payload — no JWT/CWT claims are injected. `verify` auto-detects the securing form from the input (compact-JWS string vs. COSE_Sign1 bytes), verifies the signature against the mandatory `opts.algorithms` allowlist (the JOSE `none` algorithm is always refused), re-checks the structural rules, enforces the `validFrom` / `validUntil` window against `opts.at` (default now; must be a valid Date), and optionally matches `opts.expectedIssuer` against the credential issuer id. Returns `{ credential, securing, alg, issuer }`.
|
|
12
|
+
|
|
11
13
|
- v0.12.38 (2026-05-24) — **`b.tsa` — RFC 3161 trusted timestamping client (build / parse / verify).** A timestamp authority binds a hash of your data to a trusted time, producing a token that proves the data existed at that instant — timestamp a release artifact, an audit-log checkpoint, a b.scitt signed statement, or a contract. b.tsa is the requester/verifier side of RFC 3161: buildRequest produces the DER TimeStampReq (the message imprint plus an optional nonce and a cert request), parseResponse reads the TimeStampResp (PKIStatus, failure-info bits, and the token), and verifyToken checks a token against your data and returns the asserted time. Verification is done in full per §2.4.2 / §2.3: the token is a CMS SignedData (b.cms) whose eContentType must be id-ct-TSTInfo; the message imprint must equal the hash of your data (constant-time); a sent nonce must round-trip; the signer certificate's extendedKeyUsage must be a critical, sole id-kp-timeStamping; and the CMS signature over the signed attributes must verify after the messageDigest attribute is matched to the recomputed eContent digest. An optional trust-anchor set verifies the certificate chain and validity at the asserted time. The HTTP transport to the TSA is the operator's to make. Composes b.cms and the in-tree ASN.1 DER codec; no new runtime dependency. **Added:** *`b.tsa.buildRequest(data, opts?)` / `b.tsa.parseResponse(der)` / `b.tsa.verifyToken(token, opts)`* — `buildRequest` returns `{ der, nonce, hashAlg, messageImprint }`; the imprint hash defaults to SHA-512 and may be SHA-256/384/512 or SHA3-256/512, a random 64-bit nonce and a certificate request are included by default, and a pre-hashed input is accepted with `hashed: true`. `parseResponse` returns `{ granted, status, statusString, failInfo, token }`, decoding the PKIFailureInfo bits for a non-granted response rather than throwing. `verifyToken` enforces the imprint match (`opts.data` or `opts.hash`), the nonce round-trip, the critical/sole `id-kp-timeStamping` EKU, and the CMS signature, returning `{ genTime, policy, serialHex, accuracy, hashAlg, signerCertPem }`; pass `opts.trustAnchorsPem` to also verify the certificate chain and validity at the asserted time. Timestamp tokens are third-party artifacts, so verification accepts the classical RSA (PKCS#1 v1.5 and PSS) and ECDSA-over-SHA-2 signatures that public TSAs emit — the same consume-what-exists posture as `b.cose` verification, not a framework signing default.
|
|
12
14
|
|
|
13
15
|
- v0.12.37 (2026-05-24) — **`b.scitt.signStatement` / `b.scitt.verifyStatement` — SCITT signed statements over COSE (RFC 9052 + RFC 9597).** A SCITT signed statement is a signed, attributable claim about an artifact — a signed SBOM, a build attestation, a release approval. It is a COSE_Sign1 (b.cose) whose integrity-protected CWT_Claims header (label 15, RFC 9597) binds the issuer (who makes the statement) and the subject (the artifact it is about); the artifact, or a hash/reference to it, is the payload. signStatement places iss/sub in the protected header and declares the payload media type; verifyStatement checks the COSE signature (the algorithm allowlist is mandatory) and refuses any statement that lacks the iss/sub binding, with optional expected-issuer/subject matching. Signing uses the same algorithms as b.cose — classical ES256/384/512 + EdDSA (final COSE ids, interoperable today) plus ML-DSA-87 (PQC-forward). This is the issuer side of SCITT, buildable today on finalized RFCs; the transparency receipt (an inclusion proof from a transparency service, the COSE Receipts draft) is not yet shipped — a statement produced here is the input a transparency service registers, and the receipt format is the part still in flux. It opts in when COSE Receipts publishes. **Added:** *`b.scitt.signStatement(payload, opts)` / `b.scitt.verifyStatement(statement, opts)`* — `signStatement` produces a COSE_Sign1 whose protected CWT_Claims header (label 15) carries `iss` (`opts.issuer`) and `sub` (`opts.subject`), with the payload media type declared via `opts.contentType` and extra CWT claims allowed by integer label (iss/sub cannot be overridden through `opts.claims`). `verifyStatement` verifies the signature through `b.cose.verify` (passing `opts.algorithms` as the mandatory allowlist), then requires a CWT_Claims header with both `iss` and `sub` — a bare COSE_Sign1 with no such binding is refused with `scitt/missing-cwt-claims` — and enforces `expectedIssuer` / `expectedSubject` when given. Returns `{ payload, issuer, subject, cwtClaims, alg, protectedHeaders, unprotectedHeaders }`. Because the identity binding lives in the integrity-protected header it is covered by the signature and cannot be substituted without detection. **Changed:** *`b.cose.sign` accepts `protectedHeaders` and a media-type-string `contentType`* — `opts.protectedHeaders` (a numeric-keyed object or Map) adds extra integrity-protected header parameters — the CWT_Claims map (label 15) is the SCITT case. Label 1 (alg) is reserved and managed via `opts.alg`; setting it through `protectedHeaders` is refused with `cose/reserved-header`. `opts.contentType` now accepts a media-type string (RFC 9052 §3.1 tstr form, e.g. `"application/spdx+json"`) in addition to a CoAP Content-Format uint; a string was previously dropped.
|
package/README.md
CHANGED
|
@@ -131,6 +131,7 @@ The framework bundles the surface a typical Node app reaches for. Every primitiv
|
|
|
131
131
|
- **Entity Attestation Token** — `b.eat` EAT sign/verify (RFC 9711) over `b.cwt`: device + software attestation claims (ueid / oemid / hwmodel / measurements / submods) with verifier-nonce freshness binding, `dbgstat` debug-status policy, and `eat_profile` pinning
|
|
132
132
|
- **SCITT signed statements** — `b.scitt` sign/verify a signed, attributable claim about an artifact (signed SBOM, build attestation, release approval) over `b.cose`: the issuer + subject bind in the integrity-protected CWT_Claims header (RFC 9597); verification refuses any statement missing the iss/sub binding. The issuer side, on finalized RFCs; the transparency receipt (COSE Receipts draft) opts in on publication
|
|
133
133
|
- **Trusted timestamping** — `b.tsa` RFC 3161 timestamp client: `buildRequest` a TimeStampReq, `parseResponse`, and `verifyToken` against your data — the message imprint, sent nonce, critical/sole `id-kp-timeStamping` EKU, and CMS signature are all checked, with optional certificate-chain verification. Timestamp a release artifact, audit checkpoint, or signed statement against any RFC 3161 TSA. Composes `b.cms` + the in-tree ASN.1 DER codec
|
|
134
|
+
- **Verifiable Credentials** — `b.vc` W3C Verifiable Credentials Data Model 2.0 (VC-JOSE-COSE): `issue` / `verify` a signed credential as a compact JWS (`vc+jwt`, ES256/384/512 + EdDSA) or a COSE_Sign1 (`vc+cose`, + ML-DSA-87) over `b.cose`. VCDM structural + `validFrom`/`validUntil` checks; the JOSE `none` algorithm is always refused. The W3C model, distinct from the IETF SD-JWT VC at `b.auth.sdJwtVc`
|
|
134
135
|
- **Document parsers** — `b.parsers` (XML / TOML / YAML / .env); `b.config` (schema-validated env)
|
|
135
136
|
- **File-type detection** — `b.fileType` magic-byte content classification with deny-on-upload categories (image / document / archive / executable / etc.)
|
|
136
137
|
### Content-safety gates
|
package/index.js
CHANGED
package/lib/vc.js
ADDED
|
@@ -0,0 +1,328 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* @module b.vc
|
|
4
|
+
* @nav Crypto
|
|
5
|
+
* @title Verifiable Credentials (W3C VCDM 2.0)
|
|
6
|
+
*
|
|
7
|
+
* @intro
|
|
8
|
+
* Issue and verify W3C Verifiable Credentials (VC Data Model 2.0, a
|
|
9
|
+
* W3C Recommendation) secured per "Securing Verifiable Credentials
|
|
10
|
+
* using JOSE and COSE" (VC-JOSE-COSE, also a W3C Recommendation). A
|
|
11
|
+
* verifiable credential is a tamper-evident, cryptographically-signed
|
|
12
|
+
* set of claims an issuer makes about a subject — a diploma, a
|
|
13
|
+
* membership, a license, an age assertion.
|
|
14
|
+
*
|
|
15
|
+
* Two securing mechanisms are supported, both putting the credential
|
|
16
|
+
* itself (not a JWT/CWT claims wrapper) as the signed payload:
|
|
17
|
+
* <strong>JOSE</strong> produces a compact JWS with the <code>vc+jwt</code>
|
|
18
|
+
* media type (<code>typ</code> header <code>"vc+jwt"</code>), signed
|
|
19
|
+
* with the classical ES256 / 384 / 512 or EdDSA JOSE algorithms;
|
|
20
|
+
* <strong>COSE</strong> produces a COSE_Sign1 (<code>application/vc+cose</code>)
|
|
21
|
+
* over <code>b.cose</code>, adding ML-DSA-87 (PQC-forward) to that set.
|
|
22
|
+
* <code>b.vc.verify</code> auto-detects the form from the input (a
|
|
23
|
+
* compact-JWS string vs. COSE_Sign1 bytes).
|
|
24
|
+
*
|
|
25
|
+
* <code>b.vc.issue(credential, opts)</code> validates the credential
|
|
26
|
+
* against the VCDM 2.0 structural rules (the <code>credentials/v2</code>
|
|
27
|
+
* context first, a <code>VerifiableCredential</code> type, an issuer,
|
|
28
|
+
* a credential subject) and signs it. <code>b.vc.verify(secured, opts)</code>
|
|
29
|
+
* verifies the signature (the algorithm allowlist is mandatory; the
|
|
30
|
+
* JOSE <code>none</code> algorithm is always refused), re-checks the
|
|
31
|
+
* structural rules, and enforces the <code>validFrom</code> /
|
|
32
|
+
* <code>validUntil</code> validity window. This is the W3C model and
|
|
33
|
+
* is distinct from the IETF SD-JWT VC at <code>b.auth.sdJwtVc</code>.
|
|
34
|
+
*
|
|
35
|
+
* @card
|
|
36
|
+
* W3C Verifiable Credentials 2.0 (VC-JOSE-COSE) — issue / verify a
|
|
37
|
+
* signed credential as a compact JWS (vc+jwt) or a COSE_Sign1
|
|
38
|
+
* (vc+cose), with VCDM structural + validity-window checks. Composes
|
|
39
|
+
* b.cose; the JOSE alg `none` is always refused.
|
|
40
|
+
*/
|
|
41
|
+
|
|
42
|
+
var nodeCrypto = require("node:crypto");
|
|
43
|
+
var cose = require("./cose");
|
|
44
|
+
var safeJson = require("./safe-json");
|
|
45
|
+
var validateOpts = require("./validate-opts");
|
|
46
|
+
var { defineClass } = require("./framework-error");
|
|
47
|
+
|
|
48
|
+
var VcError = defineClass("VcError", { alwaysPermanent: true });
|
|
49
|
+
|
|
50
|
+
var VCDM_V2_CONTEXT = "https://www.w3.org/ns/credentials/v2";
|
|
51
|
+
var JOSE_TYP = "vc+jwt";
|
|
52
|
+
var COSE_TYP = "application/vc+cose";
|
|
53
|
+
var COSE_CONTENT_TYPE = "application/vc";
|
|
54
|
+
var HDR_COSE_TYP = 16; // allow:raw-byte-literal — COSE "typ" header label (RFC 9596)
|
|
55
|
+
|
|
56
|
+
// JOSE signature algorithms (final RFC 7518 / 8037), mapped to node
|
|
57
|
+
// verify parameters. ECDSA uses the IEEE-P1363 fixed-width encoding JOSE
|
|
58
|
+
// mandates (not ASN.1 DER). There is no signing default — the caller
|
|
59
|
+
// names the algorithm, mirroring b.cose.
|
|
60
|
+
var JOSE_ALGS = {
|
|
61
|
+
"ES256": { nodeHash: "sha256", dsaEncoding: "ieee-p1363" },
|
|
62
|
+
"ES384": { nodeHash: "sha384", dsaEncoding: "ieee-p1363" },
|
|
63
|
+
"ES512": { nodeHash: "sha512", dsaEncoding: "ieee-p1363" },
|
|
64
|
+
"EdDSA": { nodeHash: null },
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
function _b64urlJson(obj) {
|
|
68
|
+
return Buffer.from(JSON.stringify(obj), "utf8").toString("base64url");
|
|
69
|
+
}
|
|
70
|
+
function _toKey(key, kind) {
|
|
71
|
+
if (key && typeof key === "object" && typeof key.asymmetricKeyType === "string") return key;
|
|
72
|
+
try {
|
|
73
|
+
return kind === "private" ? nodeCrypto.createPrivateKey(key) : nodeCrypto.createPublicKey(key);
|
|
74
|
+
} catch (e) {
|
|
75
|
+
throw new VcError("vc/bad-key", "vc: could not load " + kind + " key: " + ((e && e.message) || e));
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function _issuerId(cred) {
|
|
80
|
+
if (typeof cred.issuer === "string") return cred.issuer;
|
|
81
|
+
if (cred.issuer && typeof cred.issuer === "object" && typeof cred.issuer.id === "string") return cred.issuer.id;
|
|
82
|
+
return undefined;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// VCDM 2.0 structural rules; temporal checks only on verify.
|
|
86
|
+
function _validateVcdm(cred, opts) {
|
|
87
|
+
if (!cred || typeof cred !== "object" || Array.isArray(cred)) {
|
|
88
|
+
throw new VcError("vc/bad-credential", "vc: credential must be a JSON object");
|
|
89
|
+
}
|
|
90
|
+
var ctx = cred["@context"];
|
|
91
|
+
if (!Array.isArray(ctx) || ctx[0] !== VCDM_V2_CONTEXT) {
|
|
92
|
+
throw new VcError("vc/bad-context",
|
|
93
|
+
"vc: @context must be an array whose first element is '" + VCDM_V2_CONTEXT + "'");
|
|
94
|
+
}
|
|
95
|
+
var types = Array.isArray(cred.type) ? cred.type : [cred.type];
|
|
96
|
+
if (types.indexOf("VerifiableCredential") === -1) {
|
|
97
|
+
throw new VcError("vc/bad-type", "vc: type must include 'VerifiableCredential'");
|
|
98
|
+
}
|
|
99
|
+
if (_issuerId(cred) === undefined) {
|
|
100
|
+
throw new VcError("vc/no-issuer", "vc: issuer is required (a URL string or an object with an id)");
|
|
101
|
+
}
|
|
102
|
+
if (cred.credentialSubject === undefined || cred.credentialSubject === null) {
|
|
103
|
+
throw new VcError("vc/no-subject", "vc: credentialSubject is required");
|
|
104
|
+
}
|
|
105
|
+
// validFrom / validUntil, when present, MUST be valid XSD dateTimes
|
|
106
|
+
// (VCDM 2.0 §4.9). A malformed value fails closed at both issue and
|
|
107
|
+
// verify rather than silently skipping the window check.
|
|
108
|
+
var vf = _parseValidityField(cred, "validFrom");
|
|
109
|
+
var vu = _parseValidityField(cred, "validUntil");
|
|
110
|
+
if (opts && opts.temporal) {
|
|
111
|
+
var nowMs = opts.at.getTime();
|
|
112
|
+
if (vf !== null && nowMs < vf) {
|
|
113
|
+
throw new VcError("vc/not-yet-valid", "vc.verify: credential validFrom (" + cred.validFrom + ") is in the future");
|
|
114
|
+
}
|
|
115
|
+
if (vu !== null && nowMs > vu) {
|
|
116
|
+
throw new VcError("vc/expired", "vc.verify: credential validUntil (" + cred.validUntil + ") has passed");
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function _parseValidityField(cred, name) {
|
|
122
|
+
if (cred[name] === undefined) return null;
|
|
123
|
+
if (typeof cred[name] !== "string") {
|
|
124
|
+
throw new VcError("vc/bad-validity", "vc: " + name + " must be an XSD dateTime string");
|
|
125
|
+
}
|
|
126
|
+
var ms = Date.parse(cred[name]);
|
|
127
|
+
if (!isFinite(ms)) {
|
|
128
|
+
throw new VcError("vc/bad-validity", "vc: " + name + " is not a valid dateTime: " + cred[name]);
|
|
129
|
+
}
|
|
130
|
+
return ms;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* @primitive b.vc.issue
|
|
135
|
+
* @signature b.vc.issue(credential, opts)
|
|
136
|
+
* @since 0.12.39
|
|
137
|
+
* @status experimental
|
|
138
|
+
* @compliance gdpr, soc2
|
|
139
|
+
* @related b.vc.verify, b.cose.sign
|
|
140
|
+
*
|
|
141
|
+
* Validate a credential against the VCDM 2.0 structural rules and secure
|
|
142
|
+
* it. <code>securing: "jose"</code> returns a compact JWS string (media
|
|
143
|
+
* type <code>vc+jwt</code>) signed with an ES256/384/512 or EdDSA key;
|
|
144
|
+
* <code>securing: "cose"</code> returns COSE_Sign1 bytes (media type
|
|
145
|
+
* <code>application/vc+cose</code>) over <code>b.cose</code>, which also
|
|
146
|
+
* accepts <code>"ML-DSA-87"</code>. The credential itself is the signed
|
|
147
|
+
* payload — no JWT/CWT claims wrapper is added.
|
|
148
|
+
*
|
|
149
|
+
* @opts
|
|
150
|
+
* {
|
|
151
|
+
* securing: string, // "jose" (compact JWS) | "cose" (COSE_Sign1)
|
|
152
|
+
* alg: string, // JOSE: ES256/384/512 | EdDSA. COSE: + ML-DSA-87
|
|
153
|
+
* privateKey: object, // matching KeyObject or PEM
|
|
154
|
+
* kid: string, // optional key id (header)
|
|
155
|
+
* cty: string, // optional JOSE cty (e.g. "vc")
|
|
156
|
+
* }
|
|
157
|
+
*
|
|
158
|
+
* @example
|
|
159
|
+
* var jws = await b.vc.issue(credential, { securing: "jose", alg: "ES256", privateKey: key });
|
|
160
|
+
* // → a compact JWS string with typ "vc+jwt"
|
|
161
|
+
*/
|
|
162
|
+
async function issue(credential, opts) {
|
|
163
|
+
validateOpts.requireObject(opts, "vc.issue", VcError);
|
|
164
|
+
validateOpts(opts, ["securing", "alg", "privateKey", "kid", "cty"], "vc.issue");
|
|
165
|
+
_validateVcdm(credential, null);
|
|
166
|
+
if (!opts.privateKey) throw new VcError("vc/no-key", "vc.issue: opts.privateKey is required");
|
|
167
|
+
|
|
168
|
+
if (opts.securing === "cose") {
|
|
169
|
+
var protectedHeaders = {};
|
|
170
|
+
protectedHeaders[HDR_COSE_TYP] = COSE_TYP;
|
|
171
|
+
return cose.sign(Buffer.from(JSON.stringify(credential), "utf8"), {
|
|
172
|
+
alg: opts.alg,
|
|
173
|
+
privateKey: opts.privateKey,
|
|
174
|
+
kid: opts.kid,
|
|
175
|
+
contentType: COSE_CONTENT_TYPE,
|
|
176
|
+
protectedHeaders: protectedHeaders,
|
|
177
|
+
});
|
|
178
|
+
}
|
|
179
|
+
if (opts.securing === "jose") {
|
|
180
|
+
var params = JOSE_ALGS[opts.alg];
|
|
181
|
+
if (!params) {
|
|
182
|
+
throw new VcError("vc/bad-alg", "vc.issue: JOSE securing requires alg ES256/384/512 or EdDSA (got " + opts.alg + ")");
|
|
183
|
+
}
|
|
184
|
+
var key = _toKey(opts.privateKey, "private");
|
|
185
|
+
var header = { alg: opts.alg, typ: JOSE_TYP };
|
|
186
|
+
if (typeof opts.kid === "string") header.kid = opts.kid;
|
|
187
|
+
if (typeof opts.cty === "string") header.cty = opts.cty;
|
|
188
|
+
var signingInput = _b64urlJson(header) + "." + _b64urlJson(credential);
|
|
189
|
+
var sig = params.nodeHash === null
|
|
190
|
+
? nodeCrypto.sign(null, Buffer.from(signingInput, "ascii"), key)
|
|
191
|
+
: nodeCrypto.sign(params.nodeHash, Buffer.from(signingInput, "ascii"), { key: key, dsaEncoding: params.dsaEncoding });
|
|
192
|
+
return signingInput + "." + sig.toString("base64url");
|
|
193
|
+
}
|
|
194
|
+
throw new VcError("vc/bad-securing", "vc.issue: securing must be 'jose' or 'cose'");
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
function _verifyJose(token, opts) {
|
|
198
|
+
var parts = token.split(".");
|
|
199
|
+
if (parts.length !== 3) {
|
|
200
|
+
throw new VcError("vc/malformed", "vc.verify: not a compact JWS (expected three dot-separated segments)");
|
|
201
|
+
}
|
|
202
|
+
var header;
|
|
203
|
+
try { header = safeJson.parse(Buffer.from(parts[0], "base64url").toString("utf8")); }
|
|
204
|
+
catch (_e) { throw new VcError("vc/malformed", "vc.verify: JWS header is not valid base64url-JSON"); }
|
|
205
|
+
if (!header || header.typ !== JOSE_TYP) {
|
|
206
|
+
throw new VcError("vc/bad-typ", "vc.verify: JWS typ must be '" + JOSE_TYP + "'");
|
|
207
|
+
}
|
|
208
|
+
// crit-bypass defense (RFC 7515 §4.1.11): a `crit` header marks
|
|
209
|
+
// extensions the verifier MUST understand and process. This verifier
|
|
210
|
+
// implements no critical extensions, so any `crit` is refused rather
|
|
211
|
+
// than ignored — accepting it would mean honoring a credential under
|
|
212
|
+
// weaker semantics than the issuer marked mandatory.
|
|
213
|
+
if (header.crit !== undefined) {
|
|
214
|
+
throw new VcError("vc/crit-unsupported",
|
|
215
|
+
"vc.verify: JWS 'crit' header lists extensions this verifier does not support (RFC 7515 §4.1.11)");
|
|
216
|
+
}
|
|
217
|
+
if (header.alg === "none" || !JOSE_ALGS[header.alg]) {
|
|
218
|
+
throw new VcError("vc/bad-alg", "vc.verify: unsupported or unsecured JWS alg '" + header.alg + "'");
|
|
219
|
+
}
|
|
220
|
+
if (opts.algorithms.indexOf(header.alg) === -1) {
|
|
221
|
+
throw new VcError("vc/alg-not-allowed", "vc.verify: alg '" + header.alg + "' is not in the allowlist");
|
|
222
|
+
}
|
|
223
|
+
var params = JOSE_ALGS[header.alg];
|
|
224
|
+
var pub = opts.publicKey ? _toKey(opts.publicKey, "public") : _toKey(opts.keyResolver(header), "public");
|
|
225
|
+
var signingInput = parts[0] + "." + parts[1];
|
|
226
|
+
var sig = Buffer.from(parts[2], "base64url");
|
|
227
|
+
var ok = params.nodeHash === null
|
|
228
|
+
? nodeCrypto.verify(null, Buffer.from(signingInput, "ascii"), pub, sig)
|
|
229
|
+
: nodeCrypto.verify(params.nodeHash, Buffer.from(signingInput, "ascii"), { key: pub, dsaEncoding: params.dsaEncoding }, sig);
|
|
230
|
+
if (!ok) throw new VcError("vc/bad-signature", "vc.verify: JWS signature did not verify");
|
|
231
|
+
var credential;
|
|
232
|
+
try { credential = safeJson.parse(Buffer.from(parts[1], "base64url").toString("utf8")); }
|
|
233
|
+
catch (_e) { throw new VcError("vc/malformed", "vc.verify: JWS payload is not valid base64url-JSON"); }
|
|
234
|
+
return { credential: credential, alg: header.alg };
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
async function _verifyCose(bytes, opts) {
|
|
238
|
+
var algorithms = opts.algorithms.filter(function (a) { return a in cose.ALGORITHMS; });
|
|
239
|
+
if (!algorithms.length) {
|
|
240
|
+
throw new VcError("vc/no-cose-alg", "vc.verify: opts.algorithms has no COSE algorithm for a vc+cose credential");
|
|
241
|
+
}
|
|
242
|
+
var out = await cose.verify(bytes, {
|
|
243
|
+
algorithms: algorithms,
|
|
244
|
+
publicKey: opts.publicKey,
|
|
245
|
+
keyResolver: opts.keyResolver,
|
|
246
|
+
});
|
|
247
|
+
var typ = out.protectedHeaders.get(HDR_COSE_TYP);
|
|
248
|
+
if (typ !== undefined && typ !== COSE_TYP) {
|
|
249
|
+
throw new VcError("vc/bad-typ", "vc.verify: COSE typ header is '" + typ + "', expected '" + COSE_TYP + "'");
|
|
250
|
+
}
|
|
251
|
+
var credential;
|
|
252
|
+
try { credential = safeJson.parse(out.payload.toString("utf8")); }
|
|
253
|
+
catch (_e) { throw new VcError("vc/malformed", "vc.verify: COSE payload is not valid JSON"); }
|
|
254
|
+
return { credential: credential, alg: out.alg };
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
/**
|
|
258
|
+
* @primitive b.vc.verify
|
|
259
|
+
* @signature b.vc.verify(secured, opts)
|
|
260
|
+
* @since 0.12.39
|
|
261
|
+
* @status experimental
|
|
262
|
+
* @compliance gdpr, soc2
|
|
263
|
+
* @related b.vc.issue, b.cose.verify
|
|
264
|
+
*
|
|
265
|
+
* Verify a secured verifiable credential and return the credential. The
|
|
266
|
+
* securing form is auto-detected (a compact-JWS string vs. COSE_Sign1
|
|
267
|
+
* bytes); the algorithm allowlist is mandatory and the JOSE
|
|
268
|
+
* <code>none</code> algorithm is always refused. After the signature,
|
|
269
|
+
* the VCDM 2.0 structural rules are re-checked and the
|
|
270
|
+
* <code>validFrom</code> / <code>validUntil</code> window is enforced
|
|
271
|
+
* against <code>opts.at</code> (default: now).
|
|
272
|
+
*
|
|
273
|
+
* @opts
|
|
274
|
+
* {
|
|
275
|
+
* algorithms: string[], // required — accepted alg names (allowlist)
|
|
276
|
+
* publicKey: object, // verification key (KeyObject / PEM)
|
|
277
|
+
* keyResolver: function, // (header) → key (alternative to publicKey)
|
|
278
|
+
* expectedIssuer: string, // require the credential issuer (id) to match
|
|
279
|
+
* at: Date, // validity instant (default: now); must be a valid Date
|
|
280
|
+
* }
|
|
281
|
+
*
|
|
282
|
+
* @example
|
|
283
|
+
* var out = await b.vc.verify(jws, { algorithms: ["ES256"], publicKey: issuerPub, expectedIssuer: "did:example:123" });
|
|
284
|
+
* // → { credential, securing: "jose", alg: "ES256", issuer: "did:example:123" }
|
|
285
|
+
*/
|
|
286
|
+
async function verify(secured, opts) {
|
|
287
|
+
validateOpts.requireObject(opts, "vc.verify", VcError);
|
|
288
|
+
validateOpts(opts, ["algorithms", "publicKey", "keyResolver", "expectedIssuer", "at"], "vc.verify");
|
|
289
|
+
if (!Array.isArray(opts.algorithms) || opts.algorithms.length === 0) {
|
|
290
|
+
throw new VcError("vc/algorithms-required", "vc.verify: opts.algorithms is required (name the accepted algorithms)");
|
|
291
|
+
}
|
|
292
|
+
if (!opts.publicKey && typeof opts.keyResolver !== "function") {
|
|
293
|
+
throw new VcError("vc/no-key", "vc.verify: pass publicKey or keyResolver");
|
|
294
|
+
}
|
|
295
|
+
var at = new Date();
|
|
296
|
+
if (opts.at !== undefined && opts.at !== null) {
|
|
297
|
+
if (!(opts.at instanceof Date) || !isFinite(opts.at.getTime())) {
|
|
298
|
+
throw new VcError("vc/bad-at", "vc.verify: opts.at must be a valid Date");
|
|
299
|
+
}
|
|
300
|
+
at = opts.at;
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
var securing, result;
|
|
304
|
+
if (typeof secured === "string") {
|
|
305
|
+
securing = "jose";
|
|
306
|
+
result = _verifyJose(secured, opts);
|
|
307
|
+
} else if (Buffer.isBuffer(secured) || secured instanceof Uint8Array) {
|
|
308
|
+
securing = "cose";
|
|
309
|
+
result = await _verifyCose(Buffer.from(secured), opts);
|
|
310
|
+
} else {
|
|
311
|
+
throw new VcError("vc/bad-input", "vc.verify: secured must be a compact-JWS string or COSE_Sign1 bytes");
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
_validateVcdm(result.credential, { temporal: true, at: at });
|
|
315
|
+
var issuer = _issuerId(result.credential);
|
|
316
|
+
if (opts.expectedIssuer !== undefined && issuer !== opts.expectedIssuer) {
|
|
317
|
+
throw new VcError("vc/issuer-mismatch", "vc.verify: credential issuer does not match expectedIssuer");
|
|
318
|
+
}
|
|
319
|
+
return { credential: result.credential, securing: securing, alg: result.alg, issuer: issuer };
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
module.exports = {
|
|
323
|
+
issue: issue,
|
|
324
|
+
verify: verify,
|
|
325
|
+
JOSE_ALGS: JOSE_ALGS,
|
|
326
|
+
VCDM_V2_CONTEXT: VCDM_V2_CONTEXT,
|
|
327
|
+
VcError: VcError,
|
|
328
|
+
};
|
package/package.json
CHANGED
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:
|
|
5
|
+
"serialNumber": "urn:uuid:a7d5bcb5-4579-407d-b2ac-9baa683fb8f4",
|
|
6
6
|
"version": 1,
|
|
7
7
|
"metadata": {
|
|
8
|
-
"timestamp": "2026-05-
|
|
8
|
+
"timestamp": "2026-05-25T03:32:36.788Z",
|
|
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.12.
|
|
22
|
+
"bom-ref": "@blamejs/core@0.12.39",
|
|
23
23
|
"type": "application",
|
|
24
24
|
"name": "blamejs",
|
|
25
|
-
"version": "0.12.
|
|
25
|
+
"version": "0.12.39",
|
|
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.12.
|
|
29
|
+
"purl": "pkg:npm/%40blamejs/core@0.12.39",
|
|
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.12.
|
|
57
|
+
"ref": "@blamejs/core@0.12.39",
|
|
58
58
|
"dependsOn": []
|
|
59
59
|
}
|
|
60
60
|
]
|