UNPKG

65.9 kBJavaScriptView Raw
1import { __decorate, __metadata, __awaiter } 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, Observable, of } from 'rxjs';
7import { fromPromise } from 'rxjs/internal/observable/fromPromise';
8import { catchError, map, switchMap, tap, first } from 'rxjs/internal/operators';
9import { HttpClient } from '@angular/common/http';
10import { Location } 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 * @param {?} location
49 */
50 constructor(config, router, storage, http, location) {
51 this.config = config;
52 this.router = router;
53 this.storage = storage;
54 this.http = http;
55 this.location = location;
56 this.VERSION = 'v1';
57 this.signature = null;
58 this.user = null;
59 this.refreshToken = null;
60 this.clientSecret = null;
61 }
62 /**
63 * @param {?} user
64 * @return {?}
65 */
66 setUser(user) {
67 this.user = user;
68 }
69 /**
70 * @return {?}
71 */
72 getUser() {
73 if (null !== this.user) {
74 return of(this.user);
75 }
76 const /** @type {?} */ uri = this.getUri('/users/me');
77 return fromPromise(this.getSignature()).pipe(switchMap((signature) => {
78 if (null === signature) {
79 return this.refreshUser();
80 }
81 const /** @type {?} */ options = {
82 headers: {
83 Authorization: `${signature.type} ${signature.token}`,
84 },
85 };
86 return this.http.get(uri, options).pipe(map((res) => {
87 if (res.data) {
88 return res.data;
89 }
90 throw new Error('There are no body to be transformed');
91 }), map((data) => {
92 const /** @type {?} */ authUser = plainToClass(User, data);
93 this.setUser(authUser);
94 return authUser;
95 }), catchError((e) => {
96 if (401 === e.status) {
97 return this.refreshUser().pipe(switchMap((refreshUser) => {
98 if (refreshUser) {
99 return of(refreshUser);
100 }
101 return ErrorObservable.create(e);
102 }));
103 }
104 return ErrorObservable.create(e);
105 }));
106 }));
107 }
108 /**
109 * @return {?}
110 */
111 getSignature() {
112 if (this.signature) {
113 return Promise.resolve(this.signature);
114 }
115 return this.storage.get('signature');
116 }
117 /**
118 * @param {?} signature
119 * @return {?}
120 */
121 setSignature(signature) {
122 this.signature = signature;
123 this.storage.set('signature', signature);
124 }
125 /**
126 * @param {?} secret
127 * @return {?}
128 */
129 setClientSecret(secret) {
130 this.clientSecret = secret;
131 this.storage.set('client_secret', secret);
132 }
133 /**
134 * @param {?} refreshToken
135 * @return {?}
136 */
137 setRefreshToken(refreshToken) {
138 this.refreshToken = refreshToken;
139 this.storage.set('refresh_token', refreshToken);
140 }
141 /**
142 * @param {?=} scopes
143 * @return {?}
144 */
145 refreshUser(scopes) {
146 this.setUser(null);
147 this.setSignature(null);
148 return forkJoin(this.getRefreshToken(), this.getClientSecret()).pipe(switchMap((results) => {
149 const /** @type {?} */ refreshToken = results[0];
150 const /** @type {?} */ clientSecret = results[1];
151 if (refreshToken && clientSecret) {
152 return this.refreshWithRefreshToken(clientSecret, refreshToken.token, scopes);
153 }
154 return this.refreshWithSilentAuth();
155 }));
156 }
157 /**
158 * @param {?} clientSecret
159 * @param {?} refreshToken
160 * @param {?=} scopes
161 * @return {?}
162 */
163 refreshWithRefreshToken(clientSecret, refreshToken, scopes) {
164 const /** @type {?} */ request = this.http.post(this.getUri('/tokens'), {
165 grant_type: 'refresh_token',
166 scopes: scopes || 'basic email',
167 client_id: this.config.clientId,
168 client_secret: clientSecret,
169 refresh_token: refreshToken,
170 });
171 return request.pipe(tap((tokens) => {
172 if (tokens && tokens.data) {
173 this.setSignature({
174 expiresIn: tokens.data.expires_in,
175 type: tokens.data.token_type,
176 token: tokens.data.access_token,
177 });
178 this.setRefreshToken({
179 type: tokens.data.token_type,
180 token: tokens.data.refresh_token,
181 });
182 }
183 }), map((tokens) => tokens && tokens.data), switchMap((hasToken) => {
184 if (hasToken) {
185 return this.getUser();
186 }
187 return of(null);
188 }));
189 }
190 /**
191 * @return {?}
192 */
193 refreshWithSilentAuth() {
194 return new Observable((observer) => {
195 const /** @type {?} */ redirectUri = this.getCurrentOrigin('auth/silent');
196 const /** @type {?} */ uri = this.login('basic email', redirectUri, false);
197 const /** @type {?} */ iframe = document.createElement('iframe');
198 iframe.setAttribute('src', uri);
199 iframe.style.display = 'none';
200 iframe.addEventListener('load', () => __awaiter(this, void 0, void 0, function* () {
201 try {
202 const /** @type {?} */ queryString = iframe.contentWindow.location.search.substring(1);
203 iframe.parentNode.removeChild(iframe);
204 if (queryString) {
205 const /** @type {?} */ queryParts = queryString.split('&');
206 const /** @type {?} */ params = {};
207 for (const /** @type {?} */ item of queryParts) {
208 const /** @type {?} */ itemParts = item.split('=');
209 params[itemParts[0]] = itemParts[1];
210 }
211 if (params.hasOwnProperty('access_token')) {
212 this.setSignature({
213 token: params['access_token'],
214 type: params['token_type'],
215 expiresIn: params['ttl'],
216 });
217 const /** @type {?} */ user = yield this.getUser().toPromise();
218 if (user) {
219 this.setUser(user);
220 }
221 observer.next(user);
222 }
223 else {
224 observer.next(null);
225 }
226 return;
227 }
228 observer.next(null);
229 }
230 catch (/** @type {?} */ e) {
231 observer.next(null);
232 }
233 }), false);
234 document.body.appendChild(iframe);
235 });
236 }
237 /**
238 * @param {?} scopes
239 * @param {?=} redirectUri
240 * @param {?=} needState
241 * @return {?}
242 */
243 login(scopes, redirectUri, needState = true) {
244 if (!redirectUri) {
245 redirectUri = this.router.url;
246 }
247 const /** @type {?} */ params = {
248 'response_type': 'token',
249 'scopes': scopes,
250 'client': this.config.clientId,
251 'silent': true,
252 'continue': encodeURIComponent(redirectUri),
253 };
254 if (needState) {
255 const /** @type {?} */ state = this.generateState();
256 this.storage.set('state', state);
257 params.state = state;
258 }
259 const /** @type {?} */ query = this.buildQueryString(params);
260 return this.config.endpoint + '/auth?' + query;
261 }
262 /**
263 * @param {?=} redirectUri
264 * @return {?}
265 */
266 logout(redirectUri) {
267 this.signature = null;
268 this.user = null;
269 return new Promise(resolve => {
270 Promise.all([
271 this.storage.remove('signature'),
272 this.storage.remove('user'),
273 ]).then(() => {
274 let /** @type {?} */ endpoint = this.config.endpoint + '/auth/logout';
275 if (redirectUri) {
276 endpoint += '?continue=' + encodeURIComponent(redirectUri);
277 }
278 resolve(endpoint);
279 });
280 });
281 }
282 /**
283 * @return {?}
284 */
285 getState() {
286 return this.storage.get('state');
287 }
288 /**
289 * @param {?} route
290 * @return {?}
291 */
292 buildNestedUri(route) {
293 let /** @type {?} */ uri = '';
294 if (route.routeConfig && route.routeConfig.path) {
295 let /** @type {?} */ routePath = route.routeConfig.path;
296 for (const /** @type {?} */ key in route.params) {
297 if (key) {
298 const /** @type {?} */ re = new RegExp('\:' + key);
299 routePath = routePath.replace(re, route.params[key]);
300 }
301 }
302 uri += '/' + routePath;
303 }
304 if (route.children) {
305 route.children.forEach((item) => {
306 uri += this.buildNestedUri(item);
307 });
308 }
309 return uri;
310 }
311 /**
312 * @return {?}
313 */
314 getRefreshToken() {
315 if (null !== this.refreshToken) {
316 return Promise.resolve(this.refreshToken);
317 }
318 return this.storage.get('refresh_token');
319 }
320 /**
321 * @return {?}
322 */
323 getClientSecret() {
324 if (null !== this.clientSecret) {
325 return Promise.resolve(this.clientSecret);
326 }
327 return this.storage.get('client_secret');
328 }
329 /**
330 * @param {?} path
331 * @return {?}
332 */
333 getUri(path) {
334 return this.config.endpoint + '/api/' + this.VERSION + path;
335 }
336 /**
337 * @param {?=} path
338 * @return {?}
339 */
340 getCurrentOrigin(path) {
341 return window.location.origin + this.location.prepareExternalUrl(path || '');
342 }
343 /**
344 * @return {?}
345 */
346 generateState() {
347 const /** @type {?} */ possible = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';
348 let /** @type {?} */ text = '';
349 for (let /** @type {?} */ i = 0; i < 5; i++) {
350 text += possible.charAt(Math.floor(Math.random() * possible.length));
351 }
352 return text;
353 }
354 /**
355 * @param {?} params
356 * @return {?}
357 */
358 buildQueryString(params) {
359 const /** @type {?} */ queryParams = Object.keys(params).map((key) => {
360 return encodeURIComponent(key) + '=' + encodeURIComponent(params[key]);
361 });
362 return queryParams.join('&');
363 }
364}
365NacoService.decorators = [
366 { type: Injectable },
367];
368/** @nocollapse */
369NacoService.ctorParameters = () => [
370 { type: Config },
371 { type: Router },
372 { type: Storage },
373 { type: HttpClient },
374 { type: Location }
375];
376
377/**
378 * @fileoverview added by tsickle
379 * @suppress {checkTypes} checked by tsc
380 */
381class Signature {
382}
383
384/**
385 * @fileoverview added by tsickle
386 * @suppress {checkTypes} checked by tsc
387 */
388class AuthenticatedGuard {
389 /**
390 * @param {?} naco
391 * @param {?} storage
392 */
393 constructor(naco, storage) {
394 this.naco = naco;
395 this.storage = storage;
396 }
397 /**
398 * @param {?} route
399 * @return {?}
400 */
401 canActivate(route) {
402 return this.naco.getUser().pipe(first(), map((user) => {
403 if (null === user) {
404 const /** @type {?} */ uri = this.naco.buildNestedUri(route.root);
405 const /** @type {?} */ next = this.naco.getCurrentOrigin('/' + this.ltrim(uri));
406 this.storage.set('naco_intended_url', next).then(() => {
407 window.location.href = this.naco.login('basic email', this.naco.getCurrentOrigin('/auth'));
408 });
409 }
410 return !!user;
411 }));
412 }
413 /**
414 * @param {?} route
415 * @return {?}
416 */
417 canActivateChild(route) {
418 return this.canActivate(route);
419 }
420 /**
421 * @param {?} text
422 * @return {?}
423 */
424 ltrim(text) {
425 return text.replace(/^\/+/, '');
426 }
427 /**
428 * @param {?} text
429 * @return {?}
430 */
431 rtrim(text) {
432 return text.replace(/\/+$/, '');
433 }
434}
435AuthenticatedGuard.decorators = [
436 { type: Injectable },
437];
438/** @nocollapse */
439AuthenticatedGuard.ctorParameters = () => [
440 { type: NacoService },
441 { type: Storage }
442];
443
444/**
445 * @fileoverview added by tsickle
446 * @suppress {checkTypes} checked by tsc
447 */
448class AuthInterceptor {
449 /**
450 * @param {?} naco
451 */
452 constructor(naco) {
453 this.naco = naco;
454 }
455 /**
456 * @param {?} req
457 * @param {?} next
458 * @return {?}
459 */
460 intercept(req, next) {
461 return fromPromise(this.naco.getSignature()).pipe(switchMap((signature) => {
462 if (!signature) {
463 return next.handle(req);
464 }
465 return next
466 .handle(req.clone({
467 setHeaders: {
468 Authorization: `${signature.type} ${signature.token}`,
469 },
470 }))
471 .pipe(catchError((error) => {
472 if (error.status === 401) {
473 return this.naco.refreshUser().pipe(switchMap((user) => {
474 if (user) {
475 return fromPromise(this.naco.getSignature()).pipe(switchMap((newSignature) => {
476 return next.handle(req.clone({
477 setHeaders: {
478 Authorization: `${newSignature.type} ${newSignature.token}`,
479 },
480 }));
481 }));
482 }
483 return ErrorObservable.create(error);
484 }));
485 }
486 return ErrorObservable.create(error);
487 }));
488 }));
489 }
490}
491AuthInterceptor.decorators = [
492 { type: Injectable },
493];
494/** @nocollapse */
495AuthInterceptor.ctorParameters = () => [
496 { type: NacoService }
497];
498
499/**
500 * @fileoverview added by tsickle
501 * @suppress {checkTypes} checked by tsc
502 */
503/**
504 * @param {?} config
505 * @param {?} router
506 * @param {?} storage
507 * @param {?} http
508 * @param {?} location
509 * @return {?}
510 */
511function nacoFactory(config, router, storage, http, location) {
512 return new NacoService(new Config(Object.assign({}, config)), router, storage, http, location);
513}
514const /** @type {?} */ NACO_CONFIG = new InjectionToken('NACO_CONFIG');
515
516/**
517 * @fileoverview added by tsickle
518 * @suppress {checkTypes} checked by tsc
519 */
520class AuthPage {
521 /**
522 * @param {?} route
523 * @param {?} naco
524 * @param {?} storage
525 */
526 constructor(route, naco, storage) {
527 this.route = route;
528 this.naco = naco;
529 this.storage = storage;
530 this.message = 'Authenticating...';
531 this.route.queryParams.subscribe((params) => this.handle(params));
532 }
533 /**
534 * @param {?} params
535 * @return {?}
536 */
537 handle(params) {
538 if (params["state"]) {
539 this.naco.getState().then((state) => {
540 if (state !== params["state"]) {
541 this.message = 'Invalid CSRF';
542 return;
543 }
544 this.authenticate(params);
545 });
546 }
547 else {
548 this.authenticate(params);
549 }
550 }
551 /**
552 * @param {?} params
553 * @return {?}
554 */
555 authenticate(params) {
556 const /** @type {?} */ signature = {
557 expiresIn: params["expires_in"],
558 type: params["token_type"],
559 token: params["access_token"],
560 };
561 this.naco.setUser(null);
562 this.naco.setSignature(signature);
563 this.naco.getUser().subscribe((user) => {
564 if (null !== user) {
565 this.storage.get('naco_intended_url').then((url) => {
566 this.storage.remove('naco_intended_url').then(() => {
567 window.location.href = url ? url : '/';
568 });
569 });
570 return;
571 }
572 this.message = 'Unauthenticated!';
573 });
574 }
575}
576AuthPage.decorators = [
577 { type: Component, args: [{
578 selector: 'naker-auth-page',
579 template: '{{ message }}',
580 },] },
581];
582/** @nocollapse */
583AuthPage.ctorParameters = () => [
584 { type: ActivatedRoute },
585 { type: NacoService },
586 { type: Storage }
587];
588
589/**
590 * @fileoverview added by tsickle
591 * @suppress {checkTypes} checked by tsc
592 */
593class AuthSilentPage {
594}
595AuthSilentPage.decorators = [
596 { type: Component, args: [{
597 selector: 'naker-auth-silent',
598 template: 'Processing silent auth...',
599 },] },
600];
601
602/**
603 * @fileoverview added by tsickle
604 * @suppress {checkTypes} checked by tsc
605 */
606const /** @type {?} */ routes = [
607 {
608 path: 'auth',
609 component: AuthPage
610 },
611 {
612 path: 'auth/silent',
613 component: AuthSilentPage
614 },
615];
616const /** @type {?} */ AUTH_ROUTING = RouterModule.forChild(routes);
617
618/**
619 * @fileoverview added by tsickle
620 * @suppress {checkTypes} checked by tsc
621 */
622class NacoViewModule {
623}
624NacoViewModule.decorators = [
625 { type: NgModule, args: [{
626 imports: [
627 AUTH_ROUTING,
628 ],
629 declarations: [
630 AuthPage,
631 AuthSilentPage,
632 ],
633 },] },
634];
635
636/**
637 * @fileoverview added by tsickle
638 * @suppress {checkTypes} checked by tsc
639 */
640class NacoModule {
641 /**
642 * @param {?} config
643 * @return {?}
644 */
645 static forRoot(config) {
646 return {
647 ngModule: NacoModule,
648 providers: [
649 {
650 provide: NACO_CONFIG,
651 useValue: config,
652 },
653 {
654 provide: NacoService,
655 useFactory: nacoFactory,
656 deps: [NACO_CONFIG, Router, Storage, HttpClient, Location],
657 },
658 AuthenticatedGuard,
659 ],
660 };
661 }
662}
663NacoModule.decorators = [
664 { type: NgModule, args: [{
665 imports: [
666 UbudStorageModule,
667 NacoViewModule,
668 ],
669 },] },
670];
671
672/**
673 * @fileoverview added by tsickle
674 * @suppress {checkTypes} checked by tsc
675 */
676
677/**
678 * @fileoverview added by tsickle
679 * @suppress {checkTypes} checked by tsc
680 */
681
682export { NacoService, User, Signature, Config, AuthenticatedGuard, AuthInterceptor, NacoModule, NACO_CONFIG as ɵf, nacoFactory as ɵe, NacoViewModule as ɵa, AuthSilentPage as ɵd, AuthPage as ɵc, AUTH_ROUTING as ɵb };
683
684//# 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/pages/auth-silent.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, Observer, of } from 'rxjs';\nimport { Signature } from '../values/signature';\nimport { fromPromise } from 'rxjs/internal/observable/fromPromise';\nimport { catchError, map, switchMap, tap } from 'rxjs/internal/operators';\nimport { plainToClass } from 'class-transformer';\nimport { HttpClient, HttpErrorResponse } from '@angular/common/http';\nimport { Location } from '@angular/common';\nimport { ErrorObservable } from 'rxjs-compat/observable/ErrorObservable';\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, private location: Location) {\n    }\n\n    public setUser(user: User | null): void {\n        this.user = user;\n    }\n\n    public getUser(): Observable<User | null> {\n        if (null !== this.user) {\n            return of(this.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 this.refreshUser();\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                    catchError((e: HttpErrorResponse) => {\n                        if (401 === e.status) {\n                            return this.refreshUser().pipe(\n                                switchMap((refreshUser: User | null) => {\n                                    if (refreshUser) {\n                                        return of(refreshUser);\n                                    }\n\n                                    return ErrorObservable.create(e);\n                                }),\n                            );\n                        }\n\n                        return ErrorObservable.create(e);\n                    }),\n                );\n            }),\n        );\n    }\n\n    public getSignature(): Promise<Signature | null> {\n        if (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                    return this.refreshWithRefreshToken(clientSecret, refreshToken.token, scopes);\n                }\n\n                return this.refreshWithSilentAuth();\n            }),\n        );\n    }\n\n    public refreshWithRefreshToken(clientSecret: string, refreshToken: string, scopes?: string): Observable<User | null> {\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,\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    public refreshWithSilentAuth(): Observable<User | null> {\n        return new Observable<User | null>((observer: Observer<User | null>) => {\n            const redirectUri = this.getCurrentOrigin('auth/silent');\n            const uri = this.login('basic email', redirectUri, false);\n\n            const iframe = document.createElement('iframe');\n            iframe.setAttribute('src', uri);\n            iframe.style.display = 'none';\n\n            iframe.addEventListener('load', async () => {\n                try {\n                    const queryString = iframe.contentWindow.location.search.substring(1);\n\n                    iframe.parentNode.removeChild(iframe);\n\n                    if (queryString) {\n                        const queryParts = queryString.split('&');\n                        const params = {};\n\n                        for (const item of queryParts) {\n                            const itemParts = item.split('=');\n\n                            params[itemParts[0]] = itemParts[1];\n                        }\n\n                        if (params.hasOwnProperty('access_token')) {\n                            this.setSignature({\n                                token: params['access_token'],\n                                type: params['token_type'],\n                                expiresIn: params['ttl'],\n                            });\n\n                            const user = await this.getUser().toPromise();\n\n                            if (user) {\n                                this.setUser(user);\n                            }\n\n                            observer.next(user);\n                        } else {\n                            observer.next(null);\n                        }\n\n                        return;\n                    }\n\n                    observer.next(null);\n                } catch (e) {\n                    observer.next(null);\n                }\n            }, false);\n\n            document.body.appendChild(iframe);\n        });\n    }\n\n    public login(scopes: string, redirectUri?: string, needState: boolean = true): string {\n        if (!redirectUri) {\n            redirectUri = this.router.url;\n        }\n\n        const params: any = {\n            'response_type': 'token',\n            'scopes': scopes,\n            'client': this.config.clientId,\n            'silent': true,\n            'continue': encodeURIComponent(redirectUri),\n        };\n\n        if (needState) {\n            const state = this.generateState();\n\n            this.storage.set('state', state);\n            params.state = state;\n        }\n\n        const query: string = this.buildQueryString(params);\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=' + encodeURIComponent(redirectUri);\n                }\n\n                resolve(endpoint);\n            });\n        });\n    }\n\n    public getState(): Promise<string | null> {\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            let routePath = route.routeConfig.path;\n\n            for (const key in route.params) {\n                if (key) {\n                    const re = new RegExp('\\:' + key);\n\n                    routePath = routePath.replace(re, route.params[key]);\n                }\n            }\n\n            uri += '/' + routePath;\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    public getCurrentOrigin(path?: string): string {\n        return window.location.origin + this.location.prepareExternalUrl(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';\n\n@Injectable()\nexport class AuthenticatedGuard implements CanActivate, CanActivateChild {\n    public constructor(private naco: NacoService, private storage: Storage) {\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 uri = this.naco.buildNestedUri(route.root);\n                    const next = this.naco.getCurrentOrigin('/' + this.ltrim(uri));\n\n                    this.storage.set('naco_intended_url', next).then(() => {\n                        window.location.href = this.naco.login(\n                            'basic email',\n                            this.naco.getCurrentOrigin('/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((user: any) => {\n                                        if (user) {\n                                            return fromPromise(this.naco.getSignature()).pipe(\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                            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';\nimport { Location } from '@angular/common';\n\nexport function nacoFactory(config: object, router: Router, storage: Storage, http: HttpClient, location: Location) {\n    return new NacoService(\n        new Config({ ...config }),\n        router,\n        storage,\n        http,\n        location\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.setUser(null);\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 { Component } from '@angular/core';\n\n@Component({\n    selector: 'naker-auth-silent',\n    template: 'Processing silent auth...',\n})\nexport class AuthSilentPage {\n}\n","import { ModuleWithProviders } from '@angular/core';\nimport { RouterModule, Routes } from '@angular/router';\nimport { AuthPage } from './pages/auth.page';\nimport { AuthSilentPage } from './pages/auth-silent.page';\n\nconst routes: Routes = [\n    {\n        path: 'auth',\n        component: AuthPage\n    },\n    {\n        path: 'auth/silent',\n        component: AuthSilentPage\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';\nimport { AuthSilentPage } from './pages/auth-silent.page';\n\n@NgModule({\n    imports: [\n        AUTH_ROUTING,\n    ],\n    declarations: [\n        AuthPage,\n        AuthSilentPage,\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';\nimport { Location } from '@angular/common';\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, Location],\n                },\n                AuthenticatedGuard,\n            ],\n        };\n    }\n}\n"],"names":[],"mappings":"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"}
\No newline at end of file