@sphereon/openid4vci-client
Version:
OpenID for Verifiable Credential Issuance (OpenID4VCI) client
230 lines • 19.3 kB
JavaScript
"use strict";
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.OpenID4VCIClient = void 0;
const debug_1 = __importDefault(require("debug"));
const AccessTokenClient_1 = require("./AccessTokenClient");
const CredentialRequestClientBuilder_1 = require("./CredentialRequestClientBuilder");
const IssuanceInitiation_1 = require("./IssuanceInitiation");
const MetadataClient_1 = require("./MetadataClient");
const ProofOfPossessionBuilder_1 = require("./ProofOfPossessionBuilder");
const functions_1 = require("./functions");
const types_1 = require("./types");
const debug = (0, debug_1.default)('sphereon:openid4vci:flow');
class OpenID4VCIClient {
constructor(initiation, flowType, kid, alg, clientId) {
this._flowType = flowType;
this._initiation = initiation;
this._kid = kid;
this._alg = alg;
this._clientId = clientId;
}
static initiateFromURI({ issuanceInitiationURI, flowType, kid, alg, retrieveServerMetadata, clientId, }) {
return __awaiter(this, void 0, void 0, function* () {
const flow = new OpenID4VCIClient(IssuanceInitiation_1.IssuanceInitiation.fromURI(issuanceInitiationURI), flowType, kid, alg, clientId);
if (retrieveServerMetadata !== false) {
yield flow.retrieveServerMetadata();
}
return flow;
});
}
retrieveServerMetadata() {
return __awaiter(this, void 0, void 0, function* () {
this.assertInitiation();
if (!this._serverMetadata) {
this._serverMetadata = yield MetadataClient_1.MetadataClient.retrieveAllMetadataFromInitiation(this._initiation);
}
return this._serverMetadata;
});
}
createAuthorizationRequestUrl({ clientId, codeChallengeMethod, codeChallenge, redirectUri, scope }) {
if (!scope) {
throw Error('Please provide a scope. authorization_details based requests are not supported at this time');
}
if (!this._serverMetadata.openid4vci_metadata.authorization_endpoint) {
throw Error('Server metadata does not contain authorization endpoint');
}
// add 'openid' scope if not present
if (!scope.includes('openid')) {
scope = `openid ${scope}`;
}
const queryObj = {
response_type: types_1.ResponseType.AUTH_CODE,
client_id: clientId,
code_challenge_method: codeChallengeMethod,
code_challenge: codeChallenge,
redirect_uri: redirectUri,
scope: scope,
};
const authRequestUrl = (0, functions_1.convertJsonToURI)(queryObj, {
baseUrl: this._serverMetadata.openid4vci_metadata.authorization_endpoint,
uriTypeProperties: ['redirect_uri', 'scope'],
});
return authRequestUrl;
}
acquireAccessToken({ pin, clientId, codeVerifier, code, redirectUri, }) {
return __awaiter(this, void 0, void 0, function* () {
this.assertInitiation();
if (clientId) {
this._clientId = clientId;
}
if (!this._accessTokenResponse) {
const accessTokenClient = new AccessTokenClient_1.AccessTokenClient();
const response = yield accessTokenClient.acquireAccessTokenUsingIssuanceInitiation({
issuanceInitiation: this._initiation,
metadata: this._serverMetadata,
pin,
codeVerifier,
code,
redirectUri,
asOpts: { clientId: this.clientId },
});
if (response.errorBody) {
debug(`Access token error:\r\n${response.errorBody}`);
throw Error(`Retrieving an access token from ${this._serverMetadata.token_endpoint} for issuer ${this._initiation.issuanceInitiationRequest.issuer} failed with status: ${response.origResponse.status}`);
}
this._accessTokenResponse = response.successBody;
}
return this._accessTokenResponse;
});
}
acquireCredentials({ credentialType, proofCallbacks, format, kid, alg, jti, }) {
var _a;
return __awaiter(this, void 0, void 0, function* () {
if (alg) {
this._alg = alg;
}
if (kid) {
this._kid = kid;
}
const requestBuilder = CredentialRequestClientBuilder_1.CredentialRequestClientBuilder.fromIssuanceInitiation({
initiation: this.initiation,
metadata: this.serverMetadata,
});
requestBuilder.withToken(this.accessTokenResponse.access_token);
if ((_a = this.serverMetadata) === null || _a === void 0 ? void 0 : _a.openid4vci_metadata) {
const metadata = this.serverMetadata.openid4vci_metadata;
const types = Array.isArray(credentialType) ? credentialType : [credentialType];
if (types.some((type) => !metadata.credentials_supported || !metadata.credentials_supported[type])) {
throw Error(`Not all credential types ${JSON.stringify(credentialType)} are supported by issuer ${this.getIssuer()}`);
}
// todo: Format check? We might end up with some disjoint type / format combinations supported by the server
}
const credentialRequestClient = requestBuilder.build();
const proofBuilder = ProofOfPossessionBuilder_1.ProofOfPossessionBuilder.fromAccessTokenResponse({
accessTokenResponse: this.accessTokenResponse,
callbacks: proofCallbacks,
})
.withIssuer(this.getIssuer())
.withAlg(this.alg)
.withJti(jti)
.withClientId(this.clientId)
.withKid(this.kid);
const response = yield credentialRequestClient.acquireCredentialsUsingProof({
proofInput: proofBuilder,
credentialType,
format,
});
if (response.errorBody) {
debug(`Access token error:\r\n${response.errorBody}`);
throw Error(`Retrieving a credential from ${this._serverMetadata.credential_endpoint} for issuer ${this._initiation.issuanceInitiationRequest.issuer} failed with status: ${response.origResponse.status}`);
}
return response.successBody;
});
}
getCredentialsSupported(restrictToInitiationTypes) {
const credentialsSupported = this.serverMetadata.openid4vci_metadata.credentials_supported;
if (restrictToInitiationTypes === false) {
return credentialsSupported;
}
const initiationTypes = this.getCredentialTypesFromInitiation();
const supported = {};
for (const [key, value] of Object.entries(credentialsSupported)) {
if (initiationTypes.includes(key)) {
supported[key] = value;
}
}
return supported;
}
getCredentialMetadata(type) {
return this.getCredentialsSupported(false)[type];
}
getCredentialTypesFromInitiation() {
return typeof this.initiation.issuanceInitiationRequest.credential_type === 'string'
? [this.initiation.issuanceInitiationRequest.credential_type]
: this.initiation.issuanceInitiationRequest.credential_type;
}
get flowType() {
return this._flowType;
}
get initiation() {
return this._initiation;
}
get serverMetadata() {
this.assertServerMetadata();
return this._serverMetadata;
}
get kid() {
this.assertInitiation();
if (!this._kid) {
throw new Error('No value for kid is supplied');
}
return this._kid;
}
get alg() {
this.assertInitiation();
if (!this._alg) {
throw new Error('No value for alg is supplied');
}
return this._alg;
}
get clientId() {
return this._clientId;
}
get accessTokenResponse() {
this.assertAccessToken();
return this._accessTokenResponse;
}
getIssuer() {
this.assertInitiation();
return this._serverMetadata ? this.serverMetadata.issuer : this.initiation.issuanceInitiationRequest.issuer;
}
getAccessTokenEndpoint() {
this.assertInitiation();
return this.serverMetadata
? this.serverMetadata.token_endpoint
: AccessTokenClient_1.AccessTokenClient.determineTokenURL({ issuerOpts: { issuer: this.getIssuer() } });
}
getCredentialEndpoint() {
this.assertInitiation();
return this.serverMetadata ? this.serverMetadata.credential_endpoint : `${this.getIssuer()}/credential`;
}
assertInitiation() {
if (!this._initiation) {
throw Error(`No issuance initiation present`);
}
}
assertServerMetadata() {
if (!this._serverMetadata) {
throw Error('No server metadata');
}
}
assertAccessToken() {
if (!this._accessTokenResponse) {
throw Error(`No access token present`);
}
}
}
exports.OpenID4VCIClient = OpenID4VCIClient;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"OpenID4VCIClient.js","sourceRoot":"","sources":["../../../lib/OpenID4VCIClient.ts"],"names":[],"mappings":"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"}