@blamejs/core 0.12.39 → 0.12.40
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/mdoc.js +305 -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.40 (2026-05-24) — **`b.mdoc` — ISO 18013-5 mdoc / mDL issuer-data verification.** Verify the issuer-signed data of an ISO/IEC 18013-5 mdoc — the credential format behind mobile driving licences (mDL) and the ISO track of the EU Digital Identity Wallet. This is the relying-party side: confirm that the data elements a holder presents were signed by the issuer and have not been altered. An mdoc's IssuerSigned carries the disclosed data elements and an issuerAuth that is a COSE_Sign1 (b.cose) over a Mobile Security Object (MSO) holding a per-element digest. b.mdoc.verifyIssuerSigned verifies the COSE signature with the issuer certificate from the COSE x5chain header, parses the MSO, enforces its validityInfo window, and recomputes each disclosed element's digest (the full Tag-24 IssuerSignedItemBytes) to match it against the MSO constant-time — the integrity check that makes selective disclosure trustworthy. An absent or mismatched digest is refused. Signing algorithms follow b.cose verification (the classical ES256/384/512 + EdDSA that real mDL issuers use; the caller names the allowlist); opts.trustAnchorsPem additionally verifies the issuer certificate chain. This completes the credential trio alongside W3C VCDM (b.vc) and IETF SD-JWT VC (b.auth.sdJwtVc). Composes b.cose + b.cbor; no new runtime dependency. **Added:** *`b.mdoc.verifyIssuerSigned(issuerSigned, opts)`* — Takes the CBOR `IssuerSigned` map (the operator extracts it from the device response / QR) and returns `{ docType, version, digestAlgorithm, validityInfo, namespaces, signerCert, alg }`. Verifies the COSE_Sign1 `issuerAuth` against the mandatory `opts.algorithms` allowlist using the issuer certificate from its `x5chain` (label 33) header; parses the Tag-24 Mobile Security Object; enforces the MSO `validityInfo` window against `opts.at` (default now; must be a valid Date; malformed dates fail closed); and recomputes the digest of every disclosed `IssuerSignedItem` (over the full Tag-24 bytes, with the MSO `digestAlgorithm` — SHA-256/384/512) to match the MSO `valueDigests` constant-time — an absent or mismatched digest is refused with `mdoc/digest-mismatch`. `opts.expectedDocType` pins the document type; `opts.trustAnchorsPem` (a PEM string or array) additionally verifies the issuer certificate chain and validity at the asserted time. A malformed `x5chain` certificate is refused with a clean `mdoc/bad-cert`. The mdoc device-authentication half (the SessionTranscript-bound holder-binding proof) is a presentation-protocol concern and is not part of issuer-data verification.
|
|
12
|
+
|
|
11
13
|
- 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
14
|
|
|
13
15
|
- 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.
|
package/README.md
CHANGED
|
@@ -132,6 +132,7 @@ The framework bundles the surface a typical Node app reaches for. Every primitiv
|
|
|
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
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`
|
|
135
|
+
- **Mobile credentials (mDL)** — `b.mdoc` ISO/IEC 18013-5 issuer-data verification: `verifyIssuerSigned` checks the COSE_Sign1 IssuerAuth (issuer cert from the `x5chain` header), the Mobile Security Object validity window, and every disclosed element's digest against the MSO `valueDigests` (the selective-disclosure integrity check), with optional issuer-chain verification. The ISO credential ecosystem alongside `b.vc` and `b.auth.sdJwtVc`. Composes `b.cose` + `b.cbor`
|
|
135
136
|
- **Document parsers** — `b.parsers` (XML / TOML / YAML / .env); `b.config` (schema-validated env)
|
|
136
137
|
- **File-type detection** — `b.fileType` magic-byte content classification with deny-on-upload categories (image / document / archive / executable / etc.)
|
|
137
138
|
### Content-safety gates
|
package/index.js
CHANGED
package/lib/mdoc.js
ADDED
|
@@ -0,0 +1,305 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* @module b.mdoc
|
|
4
|
+
* @nav Crypto
|
|
5
|
+
* @title ISO mdoc / mDL (ISO 18013-5)
|
|
6
|
+
*
|
|
7
|
+
* @intro
|
|
8
|
+
* Verify the issuer-signed data of an ISO/IEC 18013-5 mdoc — the
|
|
9
|
+
* credential format behind mobile driving licences (mDL) and the ISO
|
|
10
|
+
* track of the EU Digital Identity Wallet. This is the relying-party
|
|
11
|
+
* side: confirm that the data elements a holder presents were signed
|
|
12
|
+
* by the issuer and have not been altered.
|
|
13
|
+
*
|
|
14
|
+
* An mdoc's <code>IssuerSigned</code> structure carries the disclosed
|
|
15
|
+
* data elements (<code>nameSpaces</code>) and an <code>issuerAuth</code>
|
|
16
|
+
* that is a COSE_Sign1 (<code>b.cose</code>) over a Mobile Security
|
|
17
|
+
* Object (MSO). The MSO holds, per namespace, a SHA-256/384/512 digest
|
|
18
|
+
* of every issued element. <code>b.mdoc.verifyIssuerSigned</code>
|
|
19
|
+
* verifies the COSE signature with the issuer certificate carried in
|
|
20
|
+
* the COSE <code>x5chain</code> (label 33), parses the MSO, enforces
|
|
21
|
+
* its <code>validityInfo</code> window, and — the integrity check that
|
|
22
|
+
* makes selective disclosure trustworthy — recomputes the digest of
|
|
23
|
+
* every disclosed element (the full Tag-24 <code>IssuerSignedItemBytes</code>)
|
|
24
|
+
* and matches it against the MSO, constant-time. A disclosed element
|
|
25
|
+
* whose digest is absent or mismatched is refused.
|
|
26
|
+
*
|
|
27
|
+
* Signing algorithms follow <code>b.cose</code> verification: the
|
|
28
|
+
* classical ES256 / 384 / 512 and EdDSA that real mDL issuers use are
|
|
29
|
+
* accepted (consume-what-exists; the caller names the allowlist).
|
|
30
|
+
* <code>opts.trustAnchorsPem</code> additionally verifies the issuer
|
|
31
|
+
* certificate chain and its validity at the asserted time.
|
|
32
|
+
*
|
|
33
|
+
* <strong>Scope.</strong> This is issuer-data authentication
|
|
34
|
+
* (ISO 18013-5 §9.1.2.4) — the data is genuine and issuer-signed. The
|
|
35
|
+
* mdoc <em>device authentication</em> half (DeviceSigned / the
|
|
36
|
+
* SessionTranscript-bound holder-binding proof, §9.1.3) is deferred:
|
|
37
|
+
* it needs the live session transcript a verifier negotiates, so it is
|
|
38
|
+
* a presentation-protocol concern rather than a credential check.
|
|
39
|
+
* Composes <code>b.cose</code> + <code>b.cbor</code>; no new runtime
|
|
40
|
+
* dependency. Distinct from W3C VCDM (<code>b.vc</code>) and IETF
|
|
41
|
+
* SD-JWT VC (<code>b.auth.sdJwtVc</code>) — the three credential
|
|
42
|
+
* ecosystems.
|
|
43
|
+
*
|
|
44
|
+
* @card
|
|
45
|
+
* ISO 18013-5 mdoc / mDL issuer-data verification — checks the
|
|
46
|
+
* COSE_Sign1 IssuerAuth, the MSO validity window, and every disclosed
|
|
47
|
+
* element's digest against the Mobile Security Object. Composes
|
|
48
|
+
* b.cose + b.cbor; device-auth holder-binding deferred.
|
|
49
|
+
*/
|
|
50
|
+
|
|
51
|
+
var nodeCrypto = require("node:crypto");
|
|
52
|
+
var C = require("./constants");
|
|
53
|
+
var cbor = require("./cbor");
|
|
54
|
+
var cose = require("./cose");
|
|
55
|
+
var bCrypto = require("./crypto");
|
|
56
|
+
var validateOpts = require("./validate-opts");
|
|
57
|
+
var { defineClass } = require("./framework-error");
|
|
58
|
+
|
|
59
|
+
var MdocError = defineClass("MdocError", { alwaysPermanent: true });
|
|
60
|
+
|
|
61
|
+
var HDR_X5CHAIN = 33; // allow:raw-byte-literal allow:raw-time-literal — x5chain COSE header label (RFC 9360 is a spec number, not a size/duration)
|
|
62
|
+
var TAG_ENCODED_CBOR = 24; // allow:raw-byte-literal — RFC 8949 §3.4.5.1 embedded-CBOR tag
|
|
63
|
+
// Tags ISO 18013-5 uses in issuer data: tdate(0), epoch(1), embedded
|
|
64
|
+
// CBOR(24), full-date(1004, RFC 8943). Bounded — others are refused.
|
|
65
|
+
var ALLOWED_TAGS = [0, 1, TAG_ENCODED_CBOR, 1004];
|
|
66
|
+
var DIGEST_ALGS = { "SHA-256": "sha256", "SHA-384": "sha384", "SHA-512": "sha512" };
|
|
67
|
+
|
|
68
|
+
function _bytes(x, what) {
|
|
69
|
+
if (Buffer.isBuffer(x)) return x;
|
|
70
|
+
if (x instanceof Uint8Array) return Buffer.from(x);
|
|
71
|
+
throw new MdocError("mdoc/bad-input", "mdoc: " + what + " must be a Buffer / Uint8Array of CBOR");
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// validityInfo dates are tdate (Tag 0, an RFC 3339 string) or epoch
|
|
75
|
+
// (Tag 1). Returns epoch-ms; fails closed on a malformed value.
|
|
76
|
+
function _validityMs(v, name) {
|
|
77
|
+
var raw = (v instanceof cbor.Tag) ? v.value : v;
|
|
78
|
+
if (typeof raw === "string") {
|
|
79
|
+
var ms = Date.parse(raw);
|
|
80
|
+
if (!isFinite(ms)) throw new MdocError("mdoc/bad-validity", "mdoc: validityInfo." + name + " is not a valid date: " + raw);
|
|
81
|
+
return ms;
|
|
82
|
+
}
|
|
83
|
+
if (typeof raw === "number" && isFinite(raw)) return raw * C.TIME.seconds(1); // epoch seconds → ms
|
|
84
|
+
throw new MdocError("mdoc/bad-validity", "mdoc: validityInfo." + name + " is missing or malformed");
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
function _mapGet(m, k) { return m instanceof Map ? m.get(k) : (m ? m[k] : undefined); }
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* @primitive b.mdoc.verifyIssuerSigned
|
|
91
|
+
* @signature b.mdoc.verifyIssuerSigned(issuerSigned, opts)
|
|
92
|
+
* @since 0.12.40
|
|
93
|
+
* @status experimental
|
|
94
|
+
* @compliance gdpr, soc2
|
|
95
|
+
* @related b.cose.verify, b.vc.verify
|
|
96
|
+
*
|
|
97
|
+
* Verify the issuer-signed data of an ISO 18013-5 mdoc and return the
|
|
98
|
+
* disclosed elements. <code>issuerSigned</code> is the CBOR
|
|
99
|
+
* <code>IssuerSigned</code> map (the operator extracts it from the
|
|
100
|
+
* device response / QR). The COSE_Sign1 <code>issuerAuth</code> is
|
|
101
|
+
* verified with the issuer certificate from its <code>x5chain</code>
|
|
102
|
+
* header against the mandatory <code>opts.algorithms</code> allowlist;
|
|
103
|
+
* the MSO <code>validityInfo</code> window is enforced; and every
|
|
104
|
+
* disclosed element's digest is matched against the Mobile Security
|
|
105
|
+
* Object (a mismatch or absence is refused). Pass
|
|
106
|
+
* <code>opts.trustAnchorsPem</code> to also verify the issuer
|
|
107
|
+
* certificate chain.
|
|
108
|
+
*
|
|
109
|
+
* @opts
|
|
110
|
+
* {
|
|
111
|
+
* algorithms: string[], // required — accepted COSE alg names (ES256/384/512, EdDSA)
|
|
112
|
+
* trustAnchorsPem: string|string[], // optional issuer roots — enables chain + validity verification
|
|
113
|
+
* expectedDocType: string, // require the MSO docType to match (e.g. "org.iso.18013.5.1.mDL")
|
|
114
|
+
* at: Date, // validity instant (default now); must be a valid Date
|
|
115
|
+
* maxBytes: number, // forwarded to b.cbor.decode
|
|
116
|
+
* maxDepth: number,
|
|
117
|
+
* }
|
|
118
|
+
*
|
|
119
|
+
* @example
|
|
120
|
+
* var out = await b.mdoc.verifyIssuerSigned(issuerSignedBytes, {
|
|
121
|
+
* algorithms: ["ES256"], expectedDocType: "org.iso.18013.5.1.mDL",
|
|
122
|
+
* });
|
|
123
|
+
* // → { docType, validityInfo, namespaces: { "org.iso.18013.5.1": { family_name, age_over_18, … } }, signerCert, alg }
|
|
124
|
+
*/
|
|
125
|
+
async function verifyIssuerSigned(issuerSigned, opts) {
|
|
126
|
+
validateOpts.requireObject(opts, "mdoc.verifyIssuerSigned", MdocError);
|
|
127
|
+
validateOpts(opts, ["algorithms", "trustAnchorsPem", "expectedDocType", "at", "maxBytes", "maxDepth"], "mdoc.verifyIssuerSigned");
|
|
128
|
+
if (!Array.isArray(opts.algorithms) || opts.algorithms.length === 0) {
|
|
129
|
+
throw new MdocError("mdoc/algorithms-required", "mdoc.verifyIssuerSigned: opts.algorithms is required");
|
|
130
|
+
}
|
|
131
|
+
var at = new Date();
|
|
132
|
+
if (opts.at !== undefined && opts.at !== null) {
|
|
133
|
+
if (!(opts.at instanceof Date) || !isFinite(opts.at.getTime())) {
|
|
134
|
+
throw new MdocError("mdoc/bad-at", "mdoc.verifyIssuerSigned: opts.at must be a valid Date");
|
|
135
|
+
}
|
|
136
|
+
at = opts.at;
|
|
137
|
+
}
|
|
138
|
+
var decodeOpts = { allowedTags: ALLOWED_TAGS, maxBytes: opts.maxBytes, maxDepth: opts.maxDepth };
|
|
139
|
+
|
|
140
|
+
var top = cbor.decode(_bytes(issuerSigned, "issuerSigned"), decodeOpts);
|
|
141
|
+
var nameSpaces = _mapGet(top, "nameSpaces");
|
|
142
|
+
var issuerAuth = _mapGet(top, "issuerAuth");
|
|
143
|
+
if (!Array.isArray(issuerAuth) || issuerAuth.length !== 4) {
|
|
144
|
+
throw new MdocError("mdoc/malformed", "mdoc.verifyIssuerSigned: issuerAuth must be a COSE_Sign1 (4-element array)");
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// The signer certificate rides in the COSE x5chain (label 33): a
|
|
148
|
+
// single cert bstr or an array of bstrs, leaf first.
|
|
149
|
+
var unprotected = issuerAuth[1];
|
|
150
|
+
var x5 = _mapGet(unprotected, HDR_X5CHAIN);
|
|
151
|
+
var chain = Array.isArray(x5) ? x5 : (x5 != null ? [x5] : []);
|
|
152
|
+
if (!chain.length || !Buffer.isBuffer(chain[0])) {
|
|
153
|
+
throw new MdocError("mdoc/no-cert", "mdoc.verifyIssuerSigned: issuerAuth has no x5chain certificate (label 33)");
|
|
154
|
+
}
|
|
155
|
+
// The x5chain certificate is attacker-controlled — a malformed DER
|
|
156
|
+
// must surface as a clean error, not a raw OpenSSL throw.
|
|
157
|
+
var signerCert;
|
|
158
|
+
try { signerCert = new nodeCrypto.X509Certificate(chain[0]); }
|
|
159
|
+
catch (e) {
|
|
160
|
+
throw new MdocError("mdoc/bad-cert", "mdoc.verifyIssuerSigned: x5chain certificate is not valid DER: " + ((e && e.message) || e));
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
// Verify the COSE_Sign1 signature with the embedded signer key.
|
|
164
|
+
var coseBytes = cbor.encode(issuerAuth);
|
|
165
|
+
var verified = await cose.verify(coseBytes, {
|
|
166
|
+
algorithms: opts.algorithms,
|
|
167
|
+
keyResolver: function () { return signerCert.publicKey; },
|
|
168
|
+
maxBytes: opts.maxBytes,
|
|
169
|
+
maxDepth: opts.maxDepth,
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
// payload = Tag 24 ( bstr .cbor MSO ).
|
|
173
|
+
var payloadTag = cbor.decode(verified.payload, decodeOpts);
|
|
174
|
+
var msoBytes = (payloadTag instanceof cbor.Tag && payloadTag.tag === TAG_ENCODED_CBOR) ? payloadTag.value : null;
|
|
175
|
+
if (!Buffer.isBuffer(msoBytes)) {
|
|
176
|
+
throw new MdocError("mdoc/malformed", "mdoc.verifyIssuerSigned: issuerAuth payload is not a Tag-24 MobileSecurityObject");
|
|
177
|
+
}
|
|
178
|
+
var mso = cbor.decode(msoBytes, decodeOpts);
|
|
179
|
+
|
|
180
|
+
var digestAlgName = _mapGet(mso, "digestAlgorithm");
|
|
181
|
+
var digestNode = DIGEST_ALGS[digestAlgName];
|
|
182
|
+
if (!digestNode) {
|
|
183
|
+
throw new MdocError("mdoc/bad-digest-alg", "mdoc.verifyIssuerSigned: unsupported MSO digestAlgorithm '" + digestAlgName + "'");
|
|
184
|
+
}
|
|
185
|
+
var docType = _mapGet(mso, "docType");
|
|
186
|
+
if (opts.expectedDocType !== undefined && docType !== opts.expectedDocType) {
|
|
187
|
+
throw new MdocError("mdoc/doctype-mismatch", "mdoc.verifyIssuerSigned: MSO docType '" + docType + "' does not match expectedDocType");
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
// validityInfo window (fail closed on malformed dates).
|
|
191
|
+
var vi = _mapGet(mso, "validityInfo");
|
|
192
|
+
if (!(vi instanceof Map) && (!vi || typeof vi !== "object")) {
|
|
193
|
+
throw new MdocError("mdoc/malformed", "mdoc.verifyIssuerSigned: MSO has no validityInfo");
|
|
194
|
+
}
|
|
195
|
+
var nowMs = at.getTime();
|
|
196
|
+
var validFromMs = _validityMs(_mapGet(vi, "validFrom"), "validFrom");
|
|
197
|
+
var validUntilMs = _validityMs(_mapGet(vi, "validUntil"), "validUntil");
|
|
198
|
+
if (nowMs < validFromMs) throw new MdocError("mdoc/not-yet-valid", "mdoc.verifyIssuerSigned: credential not yet valid");
|
|
199
|
+
if (nowMs > validUntilMs) throw new MdocError("mdoc/expired", "mdoc.verifyIssuerSigned: credential validity has passed");
|
|
200
|
+
|
|
201
|
+
// Match every disclosed element's digest against the MSO. The digest
|
|
202
|
+
// covers the full Tag-24 IssuerSignedItemBytes (ISO 18013-5 §9.1.2.5).
|
|
203
|
+
var valueDigests = _mapGet(mso, "valueDigests");
|
|
204
|
+
var out = {};
|
|
205
|
+
if (nameSpaces instanceof Map) {
|
|
206
|
+
var nsNames = Array.from(nameSpaces.keys());
|
|
207
|
+
for (var ni = 0; ni < nsNames.length; ni += 1) {
|
|
208
|
+
var ns = nsNames[ni];
|
|
209
|
+
var items = nameSpaces.get(ns);
|
|
210
|
+
var nsDigests = _mapGet(valueDigests, ns);
|
|
211
|
+
if (!Array.isArray(items) || !(nsDigests instanceof Map)) {
|
|
212
|
+
throw new MdocError("mdoc/malformed", "mdoc.verifyIssuerSigned: namespace '" + ns + "' has no matching valueDigests");
|
|
213
|
+
}
|
|
214
|
+
out[ns] = {};
|
|
215
|
+
var seen = Object.create(null); // dup-elementIdentifier guard (proto-safe)
|
|
216
|
+
for (var ii = 0; ii < items.length; ii += 1) {
|
|
217
|
+
var item = items[ii];
|
|
218
|
+
if (!(item instanceof cbor.Tag) || item.tag !== TAG_ENCODED_CBOR || !Buffer.isBuffer(item.value)) {
|
|
219
|
+
throw new MdocError("mdoc/malformed", "mdoc.verifyIssuerSigned: IssuerSignedItem is not a Tag-24 byte string");
|
|
220
|
+
}
|
|
221
|
+
var itemBytes = cbor.encode(new cbor.Tag(TAG_ENCODED_CBOR, item.value));
|
|
222
|
+
var digest = nodeCrypto.createHash(digestNode).update(itemBytes).digest();
|
|
223
|
+
var inner = cbor.decode(item.value, decodeOpts);
|
|
224
|
+
var digestID = _mapGet(inner, "digestID");
|
|
225
|
+
var expected = nsDigests.get(digestID);
|
|
226
|
+
if (!Buffer.isBuffer(expected) || !bCrypto.timingSafeEqual(digest, expected)) {
|
|
227
|
+
throw new MdocError("mdoc/digest-mismatch",
|
|
228
|
+
"mdoc.verifyIssuerSigned: disclosed element (digestID " + digestID + ", namespace " + ns + ") does not match the MSO");
|
|
229
|
+
}
|
|
230
|
+
// Refuse a duplicate elementIdentifier within a namespace — two
|
|
231
|
+
// signed values for one element is ambiguous; fail closed rather
|
|
232
|
+
// than silently keep the last.
|
|
233
|
+
var elementId = _mapGet(inner, "elementIdentifier");
|
|
234
|
+
if (seen[elementId]) {
|
|
235
|
+
throw new MdocError("mdoc/duplicate-element",
|
|
236
|
+
"mdoc.verifyIssuerSigned: namespace '" + ns + "' has duplicate elementIdentifier '" + elementId + "'");
|
|
237
|
+
}
|
|
238
|
+
seen[elementId] = true;
|
|
239
|
+
out[ns][elementId] = _mapGet(inner, "elementValue");
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
// Optional issuer chain + validity at the asserted time.
|
|
245
|
+
if (opts.trustAnchorsPem !== undefined && opts.trustAnchorsPem !== null) {
|
|
246
|
+
var anchors = typeof opts.trustAnchorsPem === "string" ? [opts.trustAnchorsPem] : opts.trustAnchorsPem;
|
|
247
|
+
if (!Array.isArray(anchors) || anchors.length === 0 ||
|
|
248
|
+
!anchors.every(function (a) { return typeof a === "string" && a.length > 0; })) {
|
|
249
|
+
throw new MdocError("mdoc/bad-trust-anchors", "mdoc.verifyIssuerSigned: trustAnchorsPem must be a non-empty PEM string or array");
|
|
250
|
+
}
|
|
251
|
+
_verifyChain(chain, anchors, at);
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
return {
|
|
255
|
+
docType: docType,
|
|
256
|
+
version: _mapGet(mso, "version"),
|
|
257
|
+
digestAlgorithm: digestAlgName,
|
|
258
|
+
validityInfo: { validFrom: new Date(validFromMs), validUntil: new Date(validUntilMs) },
|
|
259
|
+
namespaces: out,
|
|
260
|
+
signerCert: signerCert.toString(),
|
|
261
|
+
alg: verified.alg,
|
|
262
|
+
};
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
// Verify the leaf (chain[0]) chains to a supplied anchor and every cert
|
|
266
|
+
// is valid at `at`. Intermediates in the x5chain are consulted.
|
|
267
|
+
function _verifyChain(chainDer, anchorsPem, at) {
|
|
268
|
+
var anchors = anchorsPem.map(function (p) { return new nodeCrypto.X509Certificate(p); });
|
|
269
|
+
var pool = chainDer.map(function (d) { return new nodeCrypto.X509Certificate(d); });
|
|
270
|
+
var current = pool[0];
|
|
271
|
+
var atMs = at.getTime();
|
|
272
|
+
var steps = 0;
|
|
273
|
+
while (steps <= pool.length + 1) {
|
|
274
|
+
_assertValidAt(current, atMs);
|
|
275
|
+
for (var a = 0; a < anchors.length; a += 1) {
|
|
276
|
+
if (_issued(anchors[a], current)) { _assertValidAt(anchors[a], atMs); return; }
|
|
277
|
+
if (current.fingerprint256 === anchors[a].fingerprint256) return;
|
|
278
|
+
}
|
|
279
|
+
var parent = null;
|
|
280
|
+
for (var p = 0; p < pool.length; p += 1) {
|
|
281
|
+
if (pool[p].fingerprint256 !== current.fingerprint256 && _issued(pool[p], current)) { parent = pool[p]; break; }
|
|
282
|
+
}
|
|
283
|
+
if (!parent) {
|
|
284
|
+
throw new MdocError("mdoc/untrusted-chain", "mdoc.verifyIssuerSigned: issuer certificate does not chain to a supplied trust anchor");
|
|
285
|
+
}
|
|
286
|
+
current = parent;
|
|
287
|
+
steps += 1;
|
|
288
|
+
}
|
|
289
|
+
throw new MdocError("mdoc/chain-loop", "mdoc.verifyIssuerSigned: certificate chain did not terminate");
|
|
290
|
+
}
|
|
291
|
+
function _issued(issuer, subject) {
|
|
292
|
+
try { return subject.checkIssued(issuer) && subject.verify(issuer.publicKey); }
|
|
293
|
+
catch (_e) { return false; }
|
|
294
|
+
}
|
|
295
|
+
function _assertValidAt(cert, atMs) {
|
|
296
|
+
if (atMs < cert.validFromDate.getTime() || atMs > cert.validToDate.getTime()) {
|
|
297
|
+
throw new MdocError("mdoc/cert-expired", "mdoc.verifyIssuerSigned: certificate '" + cert.subject + "' is not valid at the asserted time");
|
|
298
|
+
}
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
module.exports = {
|
|
302
|
+
verifyIssuerSigned: verifyIssuerSigned,
|
|
303
|
+
DIGEST_ALGS: DIGEST_ALGS,
|
|
304
|
+
MdocError: MdocError,
|
|
305
|
+
};
|
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:6a95f7d3-c5ac-4994-955e-1fb82f6bb11b",
|
|
6
6
|
"version": 1,
|
|
7
7
|
"metadata": {
|
|
8
|
-
"timestamp": "2026-05-
|
|
8
|
+
"timestamp": "2026-05-25T04:51:15.499Z",
|
|
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.40",
|
|
23
23
|
"type": "application",
|
|
24
24
|
"name": "blamejs",
|
|
25
|
-
"version": "0.12.
|
|
25
|
+
"version": "0.12.40",
|
|
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.40",
|
|
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.40",
|
|
58
58
|
"dependsOn": []
|
|
59
59
|
}
|
|
60
60
|
]
|