UNPKG

54 kBJavaScriptView Raw
1import { __decorate, __metadata } from 'tslib';
2import { forwardRef, Injectable, InjectionToken, Component, NgModule } from '@angular/core';
3import { Type, plainToClass } from 'class-transformer';
4import { Router, ActivatedRoute, RouterModule } from '@angular/router';
5import { Storage, UbudStorageModule } from '@ubud/storage';
6import { forkJoin, of } from 'rxjs';
7import { fromPromise } from 'rxjs/internal/observable/fromPromise';
8import { map, switchMap, tap, first, catchError } from 'rxjs/internal/operators';
9import { HttpClient } from '@angular/common/http';
10import { PlatformLocation } from '@angular/common';
11import { ErrorObservable } from 'rxjs-compat/observable/ErrorObservable';
12
13/**
14 * @fileoverview added by tsickle
15 * @suppress {checkTypes} checked by tsc
16 */
17class Config {
18 /**
19 * @param {?=} data
20 */
21 constructor(data) {
22 this.endpoint = 'https://account.kemnaker.go.id';
23 Object.assign(this, data);
24 }
25}
26
27/**
28 * @fileoverview added by tsickle
29 * @suppress {checkTypes} checked by tsc
30 */
31class User {
32}
33__decorate([
34 Type(/** @type {?} */ (forwardRef(() => Date))),
35 __metadata("design:type", Date)
36], User.prototype, "updatedAt", void 0);
37
38/**
39 * @fileoverview added by tsickle
40 * @suppress {checkTypes} checked by tsc
41 */
42class NacoService {
43 /**
44 * @param {?} config
45 * @param {?} router
46 * @param {?} storage
47 * @param {?} http
48 */
49 constructor(config, router, storage, http) {
50 this.config = config;
51 this.router = router;
52 this.storage = storage;
53 this.http = http;
54 this.VERSION = 'v1';
55 this.signature = null;
56 this.user = null;
57 this.refreshToken = null;
58 this.clientSecret = null;
59 }
60 /**
61 * @param {?} user
62 * @return {?}
63 */
64 setUser(user) {
65 this.user = user;
66 this.storage.set('user', user);
67 }
68 /**
69 * @return {?}
70 */
71 getUser() {
72 if (null !== this.user) {
73 return of(this.user);
74 }
75 return fromPromise(this.storage.get('user')).pipe(switchMap((user) => {
76 if (null !== user) {
77 return of(user);
78 }
79 const /** @type {?} */ uri = this.getUri('/users/me');
80 return fromPromise(this.getSignature()).pipe(switchMap((signature) => {
81 if (null === signature) {
82 return of(null);
83 }
84 const /** @type {?} */ options = {
85 headers: {
86 Authorization: `${signature.type} ${signature.token}`,
87 },
88 };
89 return this.http.get(uri, options).pipe(map((res) => {
90 if (res.data) {
91 return res.data;
92 }
93 throw new Error('There are no body to be transformed');
94 }), map((data) => {
95 const /** @type {?} */ authUser = plainToClass(User, data);
96 this.setUser(authUser);
97 return authUser;
98 }));
99 }));
100 }));
101 }
102 /**
103 * @return {?}
104 */
105 getSignature() {
106 if (null !== this.signature) {
107 return Promise.resolve(this.signature);
108 }
109 return this.storage.get('signature');
110 }
111 /**
112 * @param {?} signature
113 * @return {?}
114 */
115 setSignature(signature) {
116 this.signature = signature;
117 this.storage.set('signature', signature);
118 }
119 /**
120 * @param {?} secret
121 * @return {?}
122 */
123 setClientSecret(secret) {
124 this.clientSecret = secret;
125 this.storage.set('client_secret', secret);
126 }
127 /**
128 * @param {?} refreshToken
129 * @return {?}
130 */
131 setRefreshToken(refreshToken) {
132 this.refreshToken = refreshToken;
133 this.storage.set('refresh_token', refreshToken);
134 }
135 /**
136 * @param {?=} scopes
137 * @return {?}
138 */
139 refreshUser(scopes) {
140 this.setUser(null);
141 this.setSignature(null);
142 return forkJoin(this.getRefreshToken(), this.getClientSecret()).pipe(switchMap((results) => {
143 const /** @type {?} */ refreshToken = results[0];
144 const /** @type {?} */ clientSecret = results[1];
145 if (refreshToken && clientSecret) {
146 const /** @type {?} */ request = this.http.post(this.getUri('/tokens'), {
147 grant_type: 'refresh_token',
148 scopes: scopes || 'basic email',
149 client_id: this.config.clientId,
150 client_secret: clientSecret,
151 refresh_token: refreshToken.token,
152 });
153 return request.pipe(tap((tokens) => {
154 if (tokens && tokens.data) {
155 this.setSignature({
156 expiresIn: tokens.data.expires_in,
157 type: tokens.data.token_type,
158 token: tokens.data.access_token,
159 });
160 this.setRefreshToken({
161 type: tokens.data.token_type,
162 token: tokens.data.refresh_token,
163 });
164 }
165 }), map((tokens) => tokens && tokens.data), switchMap((hasToken) => {
166 if (hasToken) {
167 return this.getUser();
168 }
169 return of(null);
170 }));
171 }
172 return of(null);
173 }));
174 }
175 /**
176 * @param {?} scopes
177 * @param {?=} redirectUri
178 * @return {?}
179 */
180 login(scopes, redirectUri) {
181 if (!redirectUri) {
182 redirectUri = this.router.url;
183 }
184 const /** @type {?} */ state = this.generateState();
185 this.storage.set('state', state);
186 const /** @type {?} */ query = this.buildQueryString({
187 'response_type': 'token',
188 'scopes': scopes,
189 'client': this.config.clientId,
190 'state': state,
191 'continue': redirectUri,
192 });
193 return this.config.endpoint + '/auth?' + query;
194 }
195 /**
196 * @param {?=} redirectUri
197 * @return {?}
198 */
199 logout(redirectUri) {
200 this.signature = null;
201 this.user = null;
202 return new Promise(resolve => {
203 Promise.all([
204 this.storage.remove('signature'),
205 this.storage.remove('user'),
206 ]).then(() => {
207 let /** @type {?} */ endpoint = this.config.endpoint + '/auth/logout';
208 if (redirectUri) {
209 endpoint += '?continue=' + redirectUri;
210 }
211 resolve(endpoint);
212 });
213 });
214 }
215 /**
216 * @return {?}
217 */
218 getState() {
219 return this.storage.get('state');
220 }
221 /**
222 * @param {?} route
223 * @return {?}
224 */
225 buildNestedUri(route) {
226 let /** @type {?} */ uri = '';
227 if (route.routeConfig && route.routeConfig.path) {
228 uri += '/' + route.routeConfig.path;
229 }
230 if (route.children) {
231 route.children.forEach((item) => {
232 uri += this.buildNestedUri(item);
233 });
234 }
235 return uri;
236 }
237 /**
238 * @return {?}
239 */
240 getRefreshToken() {
241 if (null !== this.refreshToken) {
242 return Promise.resolve(this.refreshToken);
243 }
244 return this.storage.get('refresh_token');
245 }
246 /**
247 * @return {?}
248 */
249 getClientSecret() {
250 if (null !== this.clientSecret) {
251 return Promise.resolve(this.clientSecret);
252 }
253 return this.storage.get('client_secret');
254 }
255 /**
256 * @param {?} path
257 * @return {?}
258 */
259 getUri(path) {
260 return this.config.endpoint + '/api/' + this.VERSION + path;
261 }
262 /**
263 * @return {?}
264 */
265 generateState() {
266 const /** @type {?} */ possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
267 let /** @type {?} */ text = '';
268 for (let /** @type {?} */ i = 0; i < 5; i++) {
269 text += possible.charAt(Math.floor(Math.random() * possible.length));
270 }
271 return text;
272 }
273 /**
274 * @param {?} params
275 * @return {?}
276 */
277 buildQueryString(params) {
278 const /** @type {?} */ queryParams = Object.keys(params).map((key) => {
279 return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]);
280 });
281 return queryParams.join('&');
282 }
283}
284NacoService.decorators = [
285 { type: Injectable },
286];
287/** @nocollapse */
288NacoService.ctorParameters = () => [
289 { type: Config },
290 { type: Router },
291 { type: Storage },
292 { type: HttpClient }
293];
294
295/**
296 * @fileoverview added by tsickle
297 * @suppress {checkTypes} checked by tsc
298 */
299class Signature {
300}
301
302/**
303 * @fileoverview added by tsickle
304 * @suppress {checkTypes} checked by tsc
305 */
306class AuthenticatedGuard {
307 /**
308 * @param {?} naco
309 * @param {?} storage
310 * @param {?} platformLocation
311 */
312 constructor(naco, storage, platformLocation) {
313 this.naco = naco;
314 this.storage = storage;
315 this.platformLocation = platformLocation;
316 }
317 /**
318 * @param {?} route
319 * @return {?}
320 */
321 canActivate(route) {
322 return this.naco.getUser().pipe(first(), map((user) => {
323 if (null === user) {
324 const /** @type {?} */ base = this.rtrim((/** @type {?} */ (this.platformLocation)).location.href);
325 const /** @type {?} */ uri = this.naco.buildNestedUri(route.root);
326 const /** @type {?} */ next = base.replace(uri, '') + '/' + this.ltrim(uri);
327 const /** @type {?} */ origin = base.replace(uri, '');
328 this.storage.set('naco_intended_url', next).then(() => {
329 window.location.href = this.naco.login('basic email', origin + '/auth');
330 });
331 }
332 return !!user;
333 }));
334 }
335 /**
336 * @param {?} route
337 * @return {?}
338 */
339 canActivateChild(route) {
340 return this.canActivate(route);
341 }
342 /**
343 * @param {?} text
344 * @return {?}
345 */
346 ltrim(text) {
347 return text.replace(/^\/+/, '');
348 }
349 /**
350 * @param {?} text
351 * @return {?}
352 */
353 rtrim(text) {
354 return text.replace(/\/+$/, '');
355 }
356}
357AuthenticatedGuard.decorators = [
358 { type: Injectable },
359];
360/** @nocollapse */
361AuthenticatedGuard.ctorParameters = () => [
362 { type: NacoService },
363 { type: Storage },
364 { type: PlatformLocation }
365];
366
367/**
368 * @fileoverview added by tsickle
369 * @suppress {checkTypes} checked by tsc
370 */
371class AuthInterceptor {
372 /**
373 * @param {?} naco
374 */
375 constructor(naco) {
376 this.naco = naco;
377 }
378 /**
379 * @param {?} req
380 * @param {?} next
381 * @return {?}
382 */
383 intercept(req, next) {
384 return fromPromise(this.naco.getSignature()).pipe(switchMap((signature) => {
385 if (!signature) {
386 return next.handle(req);
387 }
388 return next
389 .handle(req.clone({
390 setHeaders: {
391 Authorization: `${signature.type} ${signature.token}`,
392 },
393 }))
394 .pipe(catchError((error) => {
395 if (error.status === 401) {
396 return this.naco.refreshUser().pipe(switchMap(() => fromPromise(this.naco.getSignature())), switchMap((newSignature) => {
397 return next.handle(req.clone({
398 setHeaders: {
399 Authorization: `${newSignature.type} ${newSignature.token}`,
400 },
401 }));
402 }));
403 }
404 return ErrorObservable.create(error);
405 }));
406 }));
407 }
408}
409AuthInterceptor.decorators = [
410 { type: Injectable },
411];
412/** @nocollapse */
413AuthInterceptor.ctorParameters = () => [
414 { type: NacoService }
415];
416
417/**
418 * @fileoverview added by tsickle
419 * @suppress {checkTypes} checked by tsc
420 */
421/**
422 * @param {?} config
423 * @param {?} router
424 * @param {?} storage
425 * @param {?} http
426 * @return {?}
427 */
428function nacoFactory(config, router, storage, http) {
429 return new NacoService(new Config(Object.assign({}, config)), router, storage, http);
430}
431const /** @type {?} */ NACO_CONFIG = new InjectionToken('NACO_CONFIG');
432
433/**
434 * @fileoverview added by tsickle
435 * @suppress {checkTypes} checked by tsc
436 */
437class AuthPage {
438 /**
439 * @param {?} route
440 * @param {?} naco
441 * @param {?} storage
442 */
443 constructor(route, naco, storage) {
444 this.route = route;
445 this.naco = naco;
446 this.storage = storage;
447 this.message = 'Authenticating...';
448 this.route.queryParams.subscribe((params) => this.handle(params));
449 }
450 /**
451 * @param {?} params
452 * @return {?}
453 */
454 handle(params) {
455 if (params["state"]) {
456 this.naco.getState().then((state) => {
457 if (state !== params["state"]) {
458 this.message = 'Invalid CSRF';
459 return;
460 }
461 this.authenticate(params);
462 });
463 }
464 else {
465 this.authenticate(params);
466 }
467 }
468 /**
469 * @param {?} params
470 * @return {?}
471 */
472 authenticate(params) {
473 const /** @type {?} */ signature = {
474 expiresIn: params["expires_in"],
475 type: params["token_type"],
476 token: params["access_token"],
477 };
478 this.naco.setSignature(signature);
479 this.naco.getUser().subscribe((user) => {
480 if (null !== user) {
481 this.storage.get('naco_intended_url').then((url) => {
482 this.storage.remove('naco_intended_url').then(() => {
483 window.location.href = url ? url : '/';
484 });
485 });
486 return;
487 }
488 this.message = 'Unauthenticated!';
489 });
490 }
491}
492AuthPage.decorators = [
493 { type: Component, args: [{
494 selector: 'naker-auth-page',
495 template: '{{ message }}',
496 },] },
497];
498/** @nocollapse */
499AuthPage.ctorParameters = () => [
500 { type: ActivatedRoute },
501 { type: NacoService },
502 { type: Storage }
503];
504
505/**
506 * @fileoverview added by tsickle
507 * @suppress {checkTypes} checked by tsc
508 */
509const /** @type {?} */ routes = [
510 {
511 path: 'auth',
512 component: AuthPage
513 },
514];
515const /** @type {?} */ AUTH_ROUTING = RouterModule.forChild(routes);
516
517/**
518 * @fileoverview added by tsickle
519 * @suppress {checkTypes} checked by tsc
520 */
521class NacoViewModule {
522}
523NacoViewModule.decorators = [
524 { type: NgModule, args: [{
525 imports: [
526 AUTH_ROUTING,
527 ],
528 declarations: [
529 AuthPage,
530 ],
531 },] },
532];
533
534/**
535 * @fileoverview added by tsickle
536 * @suppress {checkTypes} checked by tsc
537 */
538class NacoModule {
539 /**
540 * @param {?} config
541 * @return {?}
542 */
543 static forRoot(config) {
544 return {
545 ngModule: NacoModule,
546 providers: [
547 {
548 provide: NACO_CONFIG,
549 useValue: config,
550 },
551 {
552 provide: NacoService,
553 useFactory: nacoFactory,
554 deps: [NACO_CONFIG, Router, Storage, HttpClient],
555 },
556 AuthenticatedGuard,
557 ],
558 };
559 }
560}
561NacoModule.decorators = [
562 { type: NgModule, args: [{
563 imports: [
564 UbudStorageModule,
565 NacoViewModule,
566 ],
567 },] },
568];
569
570/**
571 * @fileoverview added by tsickle
572 * @suppress {checkTypes} checked by tsc
573 */
574
575/**
576 * @fileoverview added by tsickle
577 * @suppress {checkTypes} checked by tsc
578 */
579
580export { NacoService, User, Signature, Config, AuthenticatedGuard, AuthInterceptor, NacoModule, NACO_CONFIG as ɵe, nacoFactory as ɵd, NacoViewModule as ɵa, AuthPage as ɵc, AUTH_ROUTING as ɵb };
581
582//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"naker-naco.js.map","sources":["ng://@naker/naco/domains/values/config.ts","ng://@naker/naco/domains/models/user.ts","ng://@naker/naco/domains/services/naco.service.ts","ng://@naker/naco/domains/values/signature.ts","ng://@naker/naco/guards/authenticated.guard.ts","ng://@naker/naco/interceptors/auth.interceptor.ts","ng://@naker/naco/factories/naco-factory.ts","ng://@naker/naco/views/pages/auth.page.ts","ng://@naker/naco/views/routes.ts","ng://@naker/naco/views/module.ts","ng://@naker/naco/naco.module.ts"],"sourcesContent":["export class Config {\n    public endpoint: string = 'https://account.kemnaker.go.id';\n    public clientId: string;\n\n    public constructor(data?: Partial<Config>) {\n        Object.assign(this, data);\n    }\n}\n","import { forwardRef } from '@angular/core';\nimport { Type } from 'class-transformer';\n\nexport class User {\n    public id: string;\n    public username: string;\n    public email?: string;\n    public name: string;\n    public status: number;\n\n    @Type(forwardRef(() => Date) as any)\n    public updatedAt: Date;\n}\n","import { Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, Router } from '@angular/router';\nimport { Storage } from '@ubud/storage';\nimport { Config } from '../values/config';\nimport { User } from '../models/user';\nimport { forkJoin, Observable, of } from 'rxjs';\nimport { Signature } from '../values/signature';\nimport { fromPromise } from 'rxjs/internal/observable/fromPromise';\nimport { map, switchMap, tap } from 'rxjs/internal/operators';\nimport { plainToClass } from 'class-transformer';\nimport { HttpClient } from '@angular/common/http';\n\n@Injectable()\nexport class NacoService {\n    public readonly VERSION = 'v1';\n\n    private signature: Signature | null = null;\n    private user: User | null = null;\n    private refreshToken: Signature | null = null;\n    private clientSecret: string | null = null;\n\n    public constructor(private config: Config, private router: Router, private storage: Storage, private http: HttpClient) {\n    }\n\n    public setUser(user: User | null): void {\n        this.user = user;\n        this.storage.set('user', user);\n    }\n\n    public getUser(): Observable<User | null> {\n        if (null !== this.user) {\n            return of(this.user);\n        }\n\n        return fromPromise(this.storage.get('user')).pipe(\n            switchMap((user: User | null) => {\n                if (null !== user) {\n                    return of(user);\n                }\n\n                const uri = this.getUri('/users/me');\n\n                return fromPromise(this.getSignature()).pipe(\n                    switchMap((signature: Signature | null) => {\n                        if (null === signature) {\n                            return of(null);\n                        }\n\n                        const options = {\n                            headers: {\n                                Authorization: `${signature.type} ${signature.token}`,\n                            },\n                        };\n\n                        return this.http.get<any>(uri, options).pipe(\n                            map((res: any) => {\n                                if (res.data) {\n                                    return res.data;\n                                }\n\n                                throw new Error('There are no body to be transformed');\n                            }),\n                            map((data: any) => {\n                                const authUser: any = plainToClass(User, data);\n\n                                this.setUser(authUser);\n\n                                return authUser;\n                            }),\n                        );\n                    }),\n                );\n            }),\n        );\n    }\n\n    public getSignature(): Promise<Signature | null> {\n        if (null !== this.signature) {\n            return Promise.resolve(this.signature);\n        }\n\n        return this.storage.get('signature');\n    }\n\n    public setSignature(signature: Signature): void {\n        this.signature = signature;\n        this.storage.set('signature', signature);\n    }\n\n    public setClientSecret(secret: string): void {\n        this.clientSecret = secret;\n        this.storage.set('client_secret', secret);\n    }\n\n    public setRefreshToken(refreshToken: Signature): void {\n        this.refreshToken = refreshToken;\n        this.storage.set('refresh_token', refreshToken);\n    }\n\n    public refreshUser(scopes?: string): Observable<User | null> {\n        this.setUser(null);\n        this.setSignature(null);\n\n        return forkJoin(this.getRefreshToken(), this.getClientSecret()).pipe(\n            switchMap((results: any) => {\n                const refreshToken: Signature = results[0];\n                const clientSecret: string = results[1];\n\n                if (refreshToken && clientSecret) {\n                    const request = this.http.post(this.getUri('/tokens'), {\n                        grant_type: 'refresh_token',\n                        scopes: scopes || 'basic email',\n                        client_id: this.config.clientId,\n                        client_secret: clientSecret,\n                        refresh_token: refreshToken.token,\n                    });\n\n                    return request.pipe(\n                        tap((tokens: any) => {\n                            if (tokens && tokens.data) {\n                                this.setSignature({\n                                    expiresIn: tokens.data.expires_in,\n                                    type: tokens.data.token_type,\n                                    token: tokens.data.access_token,\n                                });\n\n                                this.setRefreshToken({\n                                    type: tokens.data.token_type,\n                                    token: tokens.data.refresh_token,\n                                });\n                            }\n                        }),\n                        map((tokens: any) => tokens && tokens.data),\n                        switchMap((hasToken: boolean) => {\n                            if (hasToken) {\n                                return this.getUser();\n                            }\n\n                            return of(null);\n                        }),\n                    );\n                }\n\n                return of(null);\n            }),\n        );\n    }\n\n    public login(scopes: string, redirectUri?: string): string {\n        if (!redirectUri) {\n            redirectUri = this.router.url;\n        }\n\n        const state = this.generateState();\n\n        this.storage.set('state', state);\n\n        const query: string = this.buildQueryString(\n            {\n                'response_type': 'token',\n                'scopes': scopes,\n                'client': this.config.clientId,\n                'state': state,\n                'continue': redirectUri,\n            },\n        );\n\n        return this.config.endpoint + '/auth?' + query;\n    }\n\n    public logout(redirectUri?: string): Promise<string> {\n        this.signature = null;\n        this.user = null;\n\n        return new Promise(resolve => {\n            Promise.all([\n                this.storage.remove('signature'),\n                this.storage.remove('user'),\n            ]).then(() => {\n                let endpoint = this.config.endpoint + '/auth/logout';\n\n                if (redirectUri) {\n                    endpoint += '?continue=' + redirectUri;\n                }\n\n                resolve(endpoint);\n            });\n        });\n    }\n\n    public getState() {\n        return this.storage.get('state');\n    }\n\n    public buildNestedUri(route: ActivatedRouteSnapshot): string {\n        let uri = '';\n\n        if (route.routeConfig && route.routeConfig.path) {\n            uri += '/' + route.routeConfig.path;\n        }\n\n        if (route.children) {\n            route.children.forEach((item: ActivatedRouteSnapshot) => {\n                uri += this.buildNestedUri(item);\n            });\n        }\n\n        return uri;\n    }\n\n    private getRefreshToken(): Promise<Signature | null> {\n        if (null !== this.refreshToken) {\n            return Promise.resolve(this.refreshToken);\n        }\n\n        return this.storage.get('refresh_token');\n    }\n\n    private getClientSecret(): Promise<string | null> {\n        if (null !== this.clientSecret) {\n            return Promise.resolve(this.clientSecret);\n        }\n\n        return this.storage.get('client_secret');\n    }\n\n    private getUri(path: string): string {\n        return this.config.endpoint + '/api/' + this.VERSION + path;\n    }\n\n    private generateState(): string {\n        const possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\n        let text = '';\n\n        for (let i = 0; i < 5; i++) {\n            text += possible.charAt(Math.floor(Math.random() * possible.length));\n        }\n\n        return text;\n    }\n\n    private buildQueryString(params: object): string {\n        const queryParams = Object.keys(params).map((key: string) => {\n            return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]);\n        });\n\n        return queryParams.join('&');\n    }\n}\n","export class Signature {\n    public expiresIn?: number;\n    public token: string;\n    public type: string;\n}\n","import { Injectable } from '@angular/core';\nimport { CanActivate, CanActivateChild } from '@angular/router';\nimport { NacoService } from '../domains/services/naco.service';\nimport { first, map } from 'rxjs/internal/operators';\nimport { User } from '../domains/models/user';\nimport { Observable } from 'rxjs';\nimport { Storage } from '@ubud/storage';\nimport { ActivatedRouteSnapshot } from '@angular/router/src/router_state';\nimport { PlatformLocation } from '@angular/common';\n\n@Injectable()\nexport class AuthenticatedGuard implements CanActivate, CanActivateChild {\n    public constructor(private naco: NacoService, private storage: Storage, private platformLocation: PlatformLocation) {\n    }\n\n    public canActivate(route: ActivatedRouteSnapshot): Observable<boolean> {\n        return this.naco.getUser().pipe(\n            first(),\n            map((user: User | null) => {\n                if (null === user) {\n                    const base = this.rtrim((this.platformLocation as any).location.href);\n                    const uri = this.naco.buildNestedUri(route.root);\n                    const next = base.replace(uri, '') + '/' + this.ltrim(uri);\n                    const origin = base.replace(uri, '');\n\n                    this.storage.set('naco_intended_url', next).then(() => {\n                        window.location.href = this.naco.login(\n                            'basic email',\n                            origin + '/auth',\n                        );\n                    });\n                }\n\n                return !!user;\n            }),\n        );\n    }\n\n    public canActivateChild(route: ActivatedRouteSnapshot): Observable<boolean> {\n        return this.canActivate(route);\n    }\n\n    private ltrim(text: string): string {\n        return text.replace(/^\\/+/,'');\n    }\n\n    private rtrim(text: string): string {\n        return text.replace(/\\/+$/,'');\n    }\n}\n","import { HttpErrorResponse, HttpEvent, HttpHandler, HttpInterceptor, HttpRequest } from '@angular/common/http';\nimport { Observable } from 'rxjs';\nimport { NacoService } from '../domains/services/naco.service';\nimport { fromPromise } from 'rxjs/internal/observable/fromPromise';\nimport { Signature } from '../domains/values/signature';\nimport { catchError, switchMap } from 'rxjs/internal/operators';\nimport { ErrorObservable } from 'rxjs-compat/observable/ErrorObservable';\nimport { Injectable } from '@angular/core';\n\n@Injectable()\nexport class AuthInterceptor implements HttpInterceptor {\n    public constructor(private naco: NacoService) {\n    }\n\n    public intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {\n        return fromPromise(this.naco.getSignature()).pipe(\n            switchMap((signature: Signature | null) => {\n                if (!signature) {\n                    return next.handle(req);\n                }\n\n                return next\n                    .handle(\n                        req.clone({\n                            setHeaders: {\n                                Authorization: `${signature.type} ${signature.token}`,\n                            },\n                        }),\n                    )\n                    .pipe(\n                        catchError((error: HttpErrorResponse) => {\n                            if (error.status === 401) {\n                                return this.naco.refreshUser().pipe(\n                                    switchMap(() => fromPromise(this.naco.getSignature())),\n                                    switchMap((newSignature: Signature) => {\n                                        return next.handle(\n                                            req.clone({\n                                                setHeaders: {\n                                                    Authorization: `${newSignature.type} ${newSignature.token}`,\n                                                },\n                                            }),\n                                        );\n                                    }),\n                                );\n                            }\n\n                            return ErrorObservable.create(error);\n                        }),\n                    );\n            }),\n        );\n    }\n}\n","import { NacoService } from '../domains/services/naco.service';\nimport { Config } from '../domains/values/config';\nimport { Router } from '@angular/router';\nimport { Storage } from '@ubud/storage';\nimport { InjectionToken } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\n\nexport function nacoFactory(config: object, router: Router, storage: Storage, http: HttpClient) {\n    return new NacoService(\n        new Config({ ...config }),\n        router,\n        storage,\n        http\n    );\n}\n\nexport const NACO_CONFIG = new InjectionToken<object>('NACO_CONFIG');\n","import { Component } from '@angular/core';\nimport { ActivatedRoute, Params } from '@angular/router';\nimport { NacoService } from '../../domains/services/naco.service';\nimport { Signature } from '../../domains/values/signature';\nimport { User } from '../../domains/models/user';\nimport { Storage } from '@ubud/storage';\n\n@Component({\n    selector: 'naker-auth-page',\n    template: '{{ message }}',\n})\nexport class AuthPage {\n    public message: string = 'Authenticating...';\n\n    public constructor(private route: ActivatedRoute, private naco: NacoService, private storage: Storage) {\n        this.route.queryParams.subscribe((params: Params) => this.handle(params));\n    }\n\n    public handle(params: Params): void {\n        if (params.state) {\n            this.naco.getState().then((state: string | null) => {\n                if (state !== params.state) {\n                    this.message = 'Invalid CSRF';\n\n                    return;\n                }\n\n                this.authenticate(params);\n            });\n        } else {\n            this.authenticate(params);\n        }\n    }\n\n    private authenticate(params: Params): void {\n        const signature: Signature = {\n            expiresIn: params.expires_in,\n            type: params.token_type,\n            token: params.access_token,\n        };\n\n        this.naco.setSignature(signature);\n\n        this.naco.getUser().subscribe((user: User | null) => {\n            if (null !== user) {\n                this.storage.get('naco_intended_url').then((url: any) => {\n                    this.storage.remove('naco_intended_url').then(() => {\n                        window.location.href = url ? url : '/';\n                    });\n                });\n\n                return;\n            }\n\n            this.message = 'Unauthenticated!';\n        });\n    }\n}\n\n","import { ModuleWithProviders } from '@angular/core';\nimport { RouterModule, Routes } from '@angular/router';\nimport { AuthPage } from './pages/auth.page';\n\nconst routes: Routes = [\n    {\n        path: 'auth',\n        component: AuthPage\n    },\n];\n\nexport const AUTH_ROUTING: ModuleWithProviders = RouterModule.forChild(routes);\n","import { NgModule } from '@angular/core';\nimport { AUTH_ROUTING } from './routes';\nimport { AuthPage } from './pages/auth.page';\n\n@NgModule({\n    imports: [\n        AUTH_ROUTING,\n    ],\n    declarations: [\n        AuthPage,\n    ],\n})\nexport class NacoViewModule {\n}\n","import { ModuleWithProviders, NgModule } from '@angular/core';\nimport { NacoService } from './domains/services/naco.service';\nimport { AuthenticatedGuard } from './guards/authenticated.guard';\nimport { NACO_CONFIG, nacoFactory } from './factories/naco-factory';\nimport { Router } from '@angular/router';\nimport { Storage, UbudStorageModule } from '@ubud/storage';\nimport { NacoViewModule } from './views/module';\nimport { HttpClient } from '@angular/common/http';\n\n@NgModule({\n    imports: [\n        UbudStorageModule,\n        NacoViewModule,\n    ],\n})\nexport class NacoModule {\n    public static forRoot(config: { clientId: string, endpoint?: string }): ModuleWithProviders {\n        return {\n            ngModule: NacoModule,\n            providers: [\n                {\n                    provide: NACO_CONFIG,\n                    useValue: config,\n                },\n                {\n                    provide: NacoService,\n                    useFactory: nacoFactory,\n                    deps: [NACO_CONFIG, Router, Storage, HttpClient],\n                },\n                AuthenticatedGuard,\n            ],\n        };\n    }\n}\n"],"names":[],"mappings":"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"}
\No newline at end of file