UNPKG

@sphereon/openid4vci-client

Version:

OpenID for Verifiable Credential Issuance (OpenID4VCI) client

202 lines 18.8 kB
"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.AccessTokenClient = void 0; const ssi_types_1 = require("@sphereon/ssi-types"); const debug_1 = __importDefault(require("debug")); const MetadataClient_1 = require("./MetadataClient"); const functions_1 = require("./functions"); const types_1 = require("./types"); const debug = (0, debug_1.default)('sphereon:openid4vci:token'); class AccessTokenClient { acquireAccessTokenUsingIssuanceInitiation({ issuanceInitiation, asOpts, pin, codeVerifier, code, redirectUri, metadata, }) { return __awaiter(this, void 0, void 0, function* () { const { issuanceInitiationRequest } = issuanceInitiation; const isPinRequired = this.isPinRequiredValue(issuanceInitiationRequest); const issuerOpts = { issuer: issuanceInitiationRequest.issuer }; return yield this.acquireAccessTokenUsingRequest({ accessTokenRequest: yield this.createAccessTokenRequest({ issuanceInitiation, asOpts, codeVerifier, code, redirectUri, pin, }), isPinRequired, metadata, asOpts, issuerOpts, }); }); } acquireAccessTokenUsingRequest({ accessTokenRequest, isPinRequired, metadata, asOpts, issuerOpts, }) { return __awaiter(this, void 0, void 0, function* () { this.validate(accessTokenRequest, isPinRequired); const requestTokenURL = AccessTokenClient.determineTokenURL({ asOpts, issuerOpts, metadata: metadata ? metadata : (issuerOpts === null || issuerOpts === void 0 ? void 0 : issuerOpts.fetchMetadata) ? yield MetadataClient_1.MetadataClient.retrieveAllMetadata(issuerOpts.issuer, { errorOnNotFound: false }) : undefined, }); return this.sendAuthCode(requestTokenURL, accessTokenRequest); }); } createAccessTokenRequest({ issuanceInitiation, asOpts, pin, codeVerifier, code, redirectUri, }) { return __awaiter(this, void 0, void 0, function* () { const issuanceInitiationRequest = issuanceInitiation.issuanceInitiationRequest; issuanceInitiationRequest; const request = {}; if (asOpts === null || asOpts === void 0 ? void 0 : asOpts.clientId) { request.client_id = asOpts.clientId; } this.assertNumericPin(this.isPinRequiredValue(issuanceInitiationRequest), pin); request.user_pin = pin; if (issuanceInitiationRequest[types_1.PRE_AUTH_CODE_LITERAL]) { if (codeVerifier) { throw new Error('Cannot pass a code_verifier when flow type is pre-authorized'); } request.grant_type = types_1.GrantTypes.PRE_AUTHORIZED_CODE; request[types_1.PRE_AUTH_CODE_LITERAL] = issuanceInitiationRequest[types_1.PRE_AUTH_CODE_LITERAL]; } if (issuanceInitiationRequest.op_state) { this.throwNotSupportedFlow(); request.grant_type = types_1.GrantTypes.AUTHORIZATION_CODE; } if (codeVerifier) { request.code_verifier = codeVerifier; request.code = code; request.redirect_uri = redirectUri; request.grant_type = types_1.GrantTypes.AUTHORIZATION_CODE; } if (request.grant_type === types_1.GrantTypes.AUTHORIZATION_CODE && issuanceInitiationRequest[types_1.PRE_AUTH_CODE_LITERAL]) { throw Error('A pre_authorized_code flow cannot have an op_state in the initiation request'); } return request; }); } assertPreAuthorizedGrantType(grantType) { if (types_1.GrantTypes.PRE_AUTHORIZED_CODE !== grantType) { throw new Error("grant type must be 'urn:ietf:params:oauth:grant-type:pre-authorized_code'"); } } assertAuthorizationGrantType(grantType) { if (types_1.GrantTypes.AUTHORIZATION_CODE !== grantType) { throw new Error("grant type must be 'authorization_code'"); } } isPinRequiredValue(issuanceInitiationRequest) { let isPinRequired = false; if (issuanceInitiationRequest !== undefined) { if (typeof issuanceInitiationRequest.user_pin_required === 'string') { isPinRequired = issuanceInitiationRequest.user_pin_required.toLowerCase() === 'true'; } else if (typeof issuanceInitiationRequest.user_pin_required === 'boolean') { isPinRequired = issuanceInitiationRequest.user_pin_required; } } debug(`Pin required for issuer ${issuanceInitiationRequest.issuer}: ${isPinRequired}`); return isPinRequired; } assertNumericPin(isPinRequired, pin) { if (isPinRequired) { if (!pin || !/^\d{1,8}$/.test(pin)) { debug(`Pin is not 1 to 8 digits long`); throw new Error('A valid pin consisting of maximal 8 numeric characters must be present.'); } } else if (pin) { debug(`Pin set, whilst not required`); throw new Error('Cannot set a pin, when the pin is not required.'); } } assertNonEmptyPreAuthorizedCode(accessTokenRequest) { if (!accessTokenRequest[types_1.PRE_AUTH_CODE_LITERAL]) { debug(`No pre-authorized code present, whilst it is required`); throw new Error('Pre-authorization must be proven by presenting the pre-authorized code. Code must be present.'); } } assertNonEmptyCodeVerifier(accessTokenRequest) { if (!accessTokenRequest.code_verifier) { debug('No code_verifier present, whilst it is required'); throw new Error('Authorization flow requires the code_verifier to be present'); } } assertNonEmptyCode(accessTokenRequest) { if (!accessTokenRequest.code) { debug('No code present, whilst it is required'); throw new Error('Authorization flow requires the code to be present'); } } assertNonEmptyRedirectUri(accessTokenRequest) { if (!accessTokenRequest.redirect_uri) { debug('No redirect_uri present, whilst it is required'); throw new Error('Authorization flow requires the redirect_uri to be present'); } } validate(accessTokenRequest, isPinRequired) { if (accessTokenRequest.grant_type === types_1.GrantTypes.PRE_AUTHORIZED_CODE) { this.assertPreAuthorizedGrantType(accessTokenRequest.grant_type); this.assertNonEmptyPreAuthorizedCode(accessTokenRequest); this.assertNumericPin(isPinRequired, accessTokenRequest.user_pin); } else if (accessTokenRequest.grant_type === types_1.GrantTypes.AUTHORIZATION_CODE) { this.assertAuthorizationGrantType(accessTokenRequest.grant_type); this.assertNonEmptyCodeVerifier(accessTokenRequest); this.assertNonEmptyCode(accessTokenRequest); this.assertNonEmptyRedirectUri(accessTokenRequest); } else { this.throwNotSupportedFlow; } } sendAuthCode(requestTokenURL, accessTokenRequest) { return __awaiter(this, void 0, void 0, function* () { return yield (0, functions_1.formPost)(requestTokenURL, (0, functions_1.convertJsonToURI)(accessTokenRequest)); }); } static determineTokenURL({ asOpts, issuerOpts, metadata, }) { if (!asOpts && !(metadata === null || metadata === void 0 ? void 0 : metadata.token_endpoint) && !issuerOpts) { throw new Error('Cannot determine token URL if no issuer, metadata and no Authorization Server values are present'); } const url = asOpts && asOpts.as ? this.creatTokenURLFromURL(asOpts.as, asOpts === null || asOpts === void 0 ? void 0 : asOpts.allowInsecureEndpoints, asOpts.tokenEndpoint) : (metadata === null || metadata === void 0 ? void 0 : metadata.token_endpoint) ? metadata.token_endpoint : this.creatTokenURLFromURL(issuerOpts.issuer, asOpts === null || asOpts === void 0 ? void 0 : asOpts.allowInsecureEndpoints, issuerOpts.tokenEndpoint); if (!url || !ssi_types_1.ObjectUtils.isString(url)) { throw new Error('No authorization server token URL present. Cannot acquire access token'); } debug(`Token endpoint determined to be ${url}`); return url; } static creatTokenURLFromURL(url, allowInsecureEndpoints, tokenEndpoint) { if (allowInsecureEndpoints !== true && url.startsWith('http://')) { throw Error(`Unprotected token endpoints are not allowed ${url}`); } const hostname = url.replace(/https?:\/\//, '').replace(/\/$/, ''); const endpoint = tokenEndpoint ? (tokenEndpoint.startsWith('/') ? tokenEndpoint : tokenEndpoint.substring(1)) : '/token'; // We always require https return `https://${hostname}${endpoint}`; } throwNotSupportedFlow() { debug(`Only pre-authorized flow supported.`); throw new Error('Only pre-authorized-code flow is supported'); } } exports.AccessTokenClient = AccessTokenClient; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AccessTokenClient.js","sourceRoot":"","sources":["../../../lib/AccessTokenClient.ts"],"names":[],"mappings":"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"}