angular-simple-oidc
Version:
Angular Library implementing Open Id Connect specification. Code Flow, Refresh Tokens, Session Management, Discovery Document.
79 lines • 13.4 kB
JavaScript
import { Injectable } from '@angular/core';
import { HttpClient, HttpHeaders } from '@angular/common/http';
import { switchMap, take, map, tap } from 'rxjs/operators';
import { TokenValidationService, TokenHelperService, SimpleOidcError } from 'angular-simple-oidc/core';
import { OidcDiscoveryDocClient } from './discovery-document/oidc-discovery-doc-client.service';
import { TokenEndpointError, TokenEndpointUnexpectedError } from './errors';
import { TokensObtainedEvent } from './auth.events';
import { EventsService, SimpleOidcInfoEvent } from 'angular-simple-oidc/events';
export class TokenEndpointClientService {
constructor(http, discoveryDocumentClient, tokenValidation, tokenHelper, events) {
this.http = http;
this.discoveryDocumentClient = discoveryDocumentClient;
this.tokenValidation = tokenValidation;
this.tokenHelper = tokenHelper;
this.events = events;
}
call(payload) {
const headers = new HttpHeaders()
.set('Content-Type', 'application/x-www-form-urlencoded');
return this.discoveryDocumentClient.current$
.pipe(take(1), switchMap(({ token_endpoint }) => {
this.events.dispatch(new SimpleOidcInfoEvent(`Executing Token Endpoint`, { url: token_endpoint, payload }));
return this.http.post(token_endpoint, payload, { headers: headers });
}), tap({
error: (e) => {
if (e instanceof SimpleOidcError) {
return;
}
if (e.status === 400) {
// https://tools.ietf.org/html/rfc6749#section-5.2
throw new TokenEndpointError(e.error.error, e);
}
else {
throw new TokenEndpointUnexpectedError(e);
}
}
}), map(response => {
let expiresAt;
if (response.expires_in) {
expiresAt = this.tokenHelper.getExpirationFromExpiresIn(response.expires_in);
}
else {
this.events.dispatch(new SimpleOidcInfoEvent(`Token Response did not contain expires_in`, response));
}
let decodedToken;
if (response.id_token) {
this.events.dispatch(new SimpleOidcInfoEvent(`Validating Identity Token format`, response.id_token));
this.tokenValidation.validateIdTokenFormat(response.id_token);
decodedToken = this.tokenHelper.getPayloadFromToken(response.id_token);
this.events.dispatch(new SimpleOidcInfoEvent(`Identity Token Payload decoded`, decodedToken));
}
else {
this.events.dispatch(new SimpleOidcInfoEvent(`Token Response did not contain id_token`, response));
}
const result = {
accessToken: response.access_token,
accessTokenExpiresIn: response.expires_in,
accessTokenExpiresAt: expiresAt ? expiresAt.getTime() : null,
error: response.error,
idToken: response.id_token,
refreshToken: response.refresh_token,
decodedIdToken: decodedToken
};
this.events.dispatch(new TokensObtainedEvent(result));
return result;
}));
}
}
TokenEndpointClientService.decorators = [
{ type: Injectable }
];
TokenEndpointClientService.ctorParameters = () => [
{ type: HttpClient },
{ type: OidcDiscoveryDocClient },
{ type: TokenValidationService },
{ type: TokenHelperService },
{ type: EventsService }
];
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token-endpoint-client.service.js","sourceRoot":"","sources":["../../../../projects/angular-simple-oidc/src/lib/token-endpoint-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { HttpClient, HttpHeaders, HttpErrorResponse } from '@angular/common/http';\nimport { switchMap, take, map, catchError, tap } from 'rxjs/operators';\nimport {\n    TokenRequestResult,\n    DecodedIdentityToken,\n    TokenValidationService,\n    TokenHelperService,\n    SimpleOidcError\n} from 'angular-simple-oidc/core';\nimport { OidcDiscoveryDocClient } from './discovery-document/oidc-discovery-doc-client.service';\nimport { TokenEndpointError, TokenEndpointUnexpectedError } from './errors';\nimport { TokensObtainedEvent } from './auth.events';\nimport { EventsService, SimpleOidcInfoEvent } from 'angular-simple-oidc/events';\n\ninterface TokenEndpointResponse {\n    access_token?: string;\n    expires_in?: number;\n    id_token?: string;\n    error?: string;\n    refresh_token?: string;\n}\n\n@Injectable()\nexport class TokenEndpointClientService {\n\n    constructor(\n        protected readonly http: HttpClient,\n        protected readonly discoveryDocumentClient: OidcDiscoveryDocClient,\n        protected readonly tokenValidation: TokenValidationService,\n        protected readonly tokenHelper: TokenHelperService,\n        protected readonly events: EventsService,\n    ) { }\n\n    public call(payload: string) {\n        const headers: HttpHeaders = new HttpHeaders()\n            .set('Content-Type', 'application/x-www-form-urlencoded');\n        return this.discoveryDocumentClient.current$\n            .pipe(\n                take(1),\n                switchMap(({ token_endpoint }) => {\n                    this.events.dispatch(new SimpleOidcInfoEvent(`Executing Token Endpoint`,\n                        { url: token_endpoint, payload }));\n\n                    return this.http.post<TokenEndpointResponse>(token_endpoint, payload, { headers: headers });\n                }),\n                tap({\n                    error: (e: HttpErrorResponse) => {\n                        if (e instanceof SimpleOidcError) {\n                            return;\n                        }\n\n                        if (e.status === 400) {\n                            // https://tools.ietf.org/html/rfc6749#section-5.2\n                            throw new TokenEndpointError(e.error.error, e);\n                        } else {\n                            throw new TokenEndpointUnexpectedError(e);\n                        }\n                    }\n                }),\n                map(response => {\n                    let expiresAt: Date;\n                    if (response.expires_in) {\n                        expiresAt = this.tokenHelper.getExpirationFromExpiresIn(response.expires_in);\n                    } else {\n                        this.events.dispatch(new SimpleOidcInfoEvent(`Token Response did not contain expires_in`,\n                            response));\n                    }\n\n                    let decodedToken: DecodedIdentityToken;\n                    if (response.id_token) {\n                        this.events.dispatch(new SimpleOidcInfoEvent(`Validating Identity Token format`,\n                            response.id_token));\n                        this.tokenValidation.validateIdTokenFormat(response.id_token);\n\n                        decodedToken = this.tokenHelper.getPayloadFromToken(response.id_token);\n                        this.events.dispatch(new SimpleOidcInfoEvent(`Identity Token Payload decoded`,\n                            decodedToken));\n                    } else {\n                        this.events.dispatch(new SimpleOidcInfoEvent(`Token Response did not contain id_token`,\n                            response));\n                    }\n\n                    const result: TokenRequestResult = {\n                        accessToken: response.access_token,\n                        accessTokenExpiresIn: response.expires_in,\n                        accessTokenExpiresAt: expiresAt ? expiresAt.getTime() : null,\n                        error: response.error,\n                        idToken: response.id_token,\n                        refreshToken: response.refresh_token,\n                        decodedIdToken: decodedToken\n                    };\n\n                    this.events.dispatch(new TokensObtainedEvent(result));\n\n                    return result;\n                })\n            );\n    }\n}\n"]}