ngx-histaff-alpha 2.0.1 → 2.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/app/http-interceptors/auth-interceptor.mjs +10 -6
- package/esm2022/lib/app/http-interceptors/base-url-interceptor.mjs +14 -5
- package/esm2022/lib/app/http-interceptors/token-interceptor.mjs +22 -20
- package/esm2022/lib/app/interfaces/IStatisticAuthData.mjs +2 -0
- package/esm2022/lib/app/interfaces/IStatisticSysUser.mjs +2 -0
- package/esm2022/lib/app/libraries/d3/core-statistic-widget/core-statistic-widget.component.mjs +135 -22
- package/esm2022/lib/app/services/app-config.service.mjs +1 -1
- package/esm2022/lib/app/services/app-initialization.service.mjs +3 -2
- package/esm2022/lib/app/services/common-http-request.service.mjs +19 -1
- package/esm2022/lib/app/services/statistic-auth.service.mjs +49 -0
- package/esm2022/lib/app/services/statistic.service.mjs +24 -0
- package/esm2022/public-api.mjs +5 -1
- package/fesm2022/ngx-histaff-alpha.mjs +258 -49
- package/fesm2022/ngx-histaff-alpha.mjs.map +1 -1
- package/lib/app/http-interceptors/auth-interceptor.d.ts +4 -2
- package/lib/app/http-interceptors/base-url-interceptor.d.ts +1 -0
- package/lib/app/http-interceptors/token-interceptor.d.ts +4 -2
- package/lib/app/interfaces/IStatisticAuthData.d.ts +9 -0
- package/lib/app/interfaces/IStatisticSysUser.d.ts +15 -0
- package/lib/app/libraries/d3/core-statistic-widget/core-statistic-widget.component.d.ts +5 -2
- package/lib/app/services/app-config.service.d.ts +1 -1
- package/lib/app/services/common-http-request.service.d.ts +2 -0
- package/lib/app/services/statistic-auth.service.d.ts +21 -0
- package/lib/app/services/statistic.service.d.ts +11 -0
- package/package.json +1 -1
- package/public-api.d.ts +4 -0
|
@@ -1,13 +1,17 @@
|
|
|
1
1
|
import { Injectable } from '@angular/core';
|
|
2
|
+
import { InterceptorStatisticHeader } from './base-url-interceptor';
|
|
2
3
|
import * as i0 from "@angular/core";
|
|
3
4
|
import * as i1 from "../services/auth.service";
|
|
5
|
+
import * as i2 from "../services/statistic-auth.service";
|
|
4
6
|
export class AuthInterceptor {
|
|
5
|
-
constructor(
|
|
6
|
-
this.
|
|
7
|
+
constructor(authService, statisticAuthService) {
|
|
8
|
+
this.authService = authService;
|
|
9
|
+
this.statisticAuthService = statisticAuthService;
|
|
7
10
|
}
|
|
8
11
|
intercept(req, next) {
|
|
12
|
+
const isStatistic = req.headers.has(InterceptorStatisticHeader);
|
|
9
13
|
// Get the auth token from the service.
|
|
10
|
-
const authToken = this.
|
|
14
|
+
const authToken = isStatistic ? this.statisticAuthService.getAuthorizationToken() : this.authService.getAuthorizationToken();
|
|
11
15
|
if (authToken === undefined) {
|
|
12
16
|
return next.handle(req);
|
|
13
17
|
}
|
|
@@ -19,7 +23,7 @@ export class AuthInterceptor {
|
|
|
19
23
|
// send cloned request with header to the next handler.
|
|
20
24
|
return next.handle(authReq);
|
|
21
25
|
}
|
|
22
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: AuthInterceptor, deps: [{ token: i1.AuthService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
26
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: AuthInterceptor, deps: [{ token: i1.AuthService }, { token: i2.StatisticAuthService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
23
27
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: AuthInterceptor, providedIn: 'root' }); }
|
|
24
28
|
}
|
|
25
29
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: AuthInterceptor, decorators: [{
|
|
@@ -27,5 +31,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImpor
|
|
|
27
31
|
args: [{
|
|
28
32
|
providedIn: 'root'
|
|
29
33
|
}]
|
|
30
|
-
}], ctorParameters: () => [{ type: i1.AuthService }] });
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
34
|
+
}], ctorParameters: () => [{ type: i1.AuthService }, { type: i2.StatisticAuthService }] });
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1pbnRlcmNlcHRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL25neC1oaXN0YWZmLWFscGhhL3NyYy9saWIvYXBwL2h0dHAtaW50ZXJjZXB0b3JzL2F1dGgtaW50ZXJjZXB0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQU0zQyxPQUFPLEVBQUUsMEJBQTBCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7OztBQU1wRSxNQUFNLE9BQU8sZUFBZTtJQUV4QixZQUNZLFdBQXdCLEVBQ3hCLG9CQUEwQztRQUQxQyxnQkFBVyxHQUFYLFdBQVcsQ0FBYTtRQUN4Qix5QkFBb0IsR0FBcEIsb0JBQW9CLENBQXNCO0lBQ2xELENBQUM7SUFFTCxTQUFTLENBQUMsR0FBcUIsRUFBRSxJQUFpQjtRQUU5QyxNQUFNLFdBQVcsR0FBRyxHQUFHLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQywwQkFBMEIsQ0FBQyxDQUFDO1FBRWhFLHVDQUF1QztRQUN2QyxNQUFNLFNBQVMsR0FBSSxXQUFXLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxvQkFBb0IsQ0FBQyxxQkFBcUIsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLHFCQUFxQixFQUFFLENBQUM7UUFFOUgsSUFBSSxTQUFTLEtBQUssU0FBUyxFQUFFO1lBQ3pCLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxHQUFHLENBQUMsQ0FBQztTQUMzQjtRQUVELDBEQUEwRDtRQUMxRCxrREFBa0Q7UUFDbEQsTUFBTSxPQUFPLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQztZQUN0QixPQUFPLEVBQUUsR0FBRyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsZUFBZSxFQUFFLFNBQVMsR0FBRyxTQUFTLENBQUM7U0FDbkUsQ0FBQyxDQUFDO1FBRUgsdURBQXVEO1FBQ3ZELE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUNoQyxDQUFDOzhHQTFCUSxlQUFlO2tIQUFmLGVBQWUsY0FGWixNQUFNOzsyRkFFVCxlQUFlO2tCQUgzQixVQUFVO21CQUFDO29CQUNSLFVBQVUsRUFBRSxNQUFNO2lCQUNyQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHtcclxuICAgIEh0dHBJbnRlcmNlcHRvciwgSHR0cEhhbmRsZXIsIEh0dHBSZXF1ZXN0XHJcbn0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uL2h0dHAnO1xyXG5cclxuaW1wb3J0IHsgQXV0aFNlcnZpY2UgfSBmcm9tICcuLi9zZXJ2aWNlcy9hdXRoLnNlcnZpY2UnO1xyXG5pbXBvcnQgeyBJbnRlcmNlcHRvclN0YXRpc3RpY0hlYWRlciB9IGZyb20gJy4vYmFzZS11cmwtaW50ZXJjZXB0b3InO1xyXG5pbXBvcnQgeyBTdGF0aXN0aWNBdXRoU2VydmljZSB9IGZyb20gJy4uL3NlcnZpY2VzL3N0YXRpc3RpYy1hdXRoLnNlcnZpY2UnO1xyXG5cclxuQEluamVjdGFibGUoe1xyXG4gICAgcHJvdmlkZWRJbjogJ3Jvb3QnXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBBdXRoSW50ZXJjZXB0b3IgaW1wbGVtZW50cyBIdHRwSW50ZXJjZXB0b3Ige1xyXG5cclxuICAgIGNvbnN0cnVjdG9yKFxyXG4gICAgICAgIHByaXZhdGUgYXV0aFNlcnZpY2U6IEF1dGhTZXJ2aWNlLFxyXG4gICAgICAgIHByaXZhdGUgc3RhdGlzdGljQXV0aFNlcnZpY2U6IFN0YXRpc3RpY0F1dGhTZXJ2aWNlXHJcbiAgICApIHsgfVxyXG5cclxuICAgIGludGVyY2VwdChyZXE6IEh0dHBSZXF1ZXN0PGFueT4sIG5leHQ6IEh0dHBIYW5kbGVyKSB7XHJcblxyXG4gICAgICAgIGNvbnN0IGlzU3RhdGlzdGljID0gcmVxLmhlYWRlcnMuaGFzKEludGVyY2VwdG9yU3RhdGlzdGljSGVhZGVyKTtcclxuXHJcbiAgICAgICAgLy8gR2V0IHRoZSBhdXRoIHRva2VuIGZyb20gdGhlIHNlcnZpY2UuXHJcbiAgICAgICAgY29uc3QgYXV0aFRva2VuID0gIGlzU3RhdGlzdGljID8gdGhpcy5zdGF0aXN0aWNBdXRoU2VydmljZS5nZXRBdXRob3JpemF0aW9uVG9rZW4oKSA6IHRoaXMuYXV0aFNlcnZpY2UuZ2V0QXV0aG9yaXphdGlvblRva2VuKCk7XHJcblxyXG4gICAgICAgIGlmIChhdXRoVG9rZW4gPT09IHVuZGVmaW5lZCkge1xyXG4gICAgICAgICAgICByZXR1cm4gbmV4dC5oYW5kbGUocmVxKTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIENsb25lIHRoZSByZXF1ZXN0IGFuZCByZXBsYWNlIHRoZSBvcmlnaW5hbCBoZWFkZXJzIHdpdGhcclxuICAgICAgICAvLyBjbG9uZWQgaGVhZGVycywgdXBkYXRlZCB3aXRoIHRoZSBhdXRob3JpemF0aW9uLlxyXG4gICAgICAgIGNvbnN0IGF1dGhSZXEgPSByZXEuY2xvbmUoe1xyXG4gICAgICAgICAgICBoZWFkZXJzOiByZXEuaGVhZGVycy5zZXQoJ0F1dGhvcml6YXRpb24nLCAnQmVhcmVyICcgKyBhdXRoVG9rZW4pXHJcbiAgICAgICAgfSk7XHJcblxyXG4gICAgICAgIC8vIHNlbmQgY2xvbmVkIHJlcXVlc3Qgd2l0aCBoZWFkZXIgdG8gdGhlIG5leHQgaGFuZGxlci5cclxuICAgICAgICByZXR1cm4gbmV4dC5oYW5kbGUoYXV0aFJlcSk7XHJcbiAgICB9XHJcbn0iXX0=
|
|
@@ -2,6 +2,7 @@ import { Injectable } from '@angular/core';
|
|
|
2
2
|
import * as i0 from "@angular/core";
|
|
3
3
|
import * as i1 from "../services/app-config.service";
|
|
4
4
|
export const InterceptorSkipHeader = 'X-Skip-Interceptor';
|
|
5
|
+
export const InterceptorStatisticHeader = 'X-Statistic-Interceptor';
|
|
5
6
|
export class BaseUrlInterceptor {
|
|
6
7
|
constructor(appConfigService) {
|
|
7
8
|
this.appConfigService = appConfigService;
|
|
@@ -14,10 +15,18 @@ export class BaseUrlInterceptor {
|
|
|
14
15
|
if (!this.appConfigService.BASE_URL) {
|
|
15
16
|
console.error("THE APP LAYPOUT had started rendering before appConfigService.BASE_URL reachs a truely value. Please check AppInitializationService initializing$.next method carefully!!!");
|
|
16
17
|
}
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
18
|
+
if (req.headers.has(InterceptorStatisticHeader)) {
|
|
19
|
+
const addBaseUrlReq = req.clone({
|
|
20
|
+
url: this.appConfigService.BASE_URL_FOR_STATISTIC + req.url
|
|
21
|
+
});
|
|
22
|
+
return next.handle(addBaseUrlReq);
|
|
23
|
+
}
|
|
24
|
+
else {
|
|
25
|
+
const addBaseUrlReq = req.clone({
|
|
26
|
+
url: this.appConfigService.BASE_URL + req.url
|
|
27
|
+
});
|
|
28
|
+
return next.handle(addBaseUrlReq);
|
|
29
|
+
}
|
|
21
30
|
}
|
|
22
31
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: BaseUrlInterceptor, deps: [{ token: i1.AppConfigService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
23
32
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: BaseUrlInterceptor, providedIn: 'root' }); }
|
|
@@ -28,4 +37,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImpor
|
|
|
28
37
|
providedIn: 'root'
|
|
29
38
|
}]
|
|
30
39
|
}], ctorParameters: () => [{ type: i1.AppConfigService }] });
|
|
31
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
40
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmFzZS11cmwtaW50ZXJjZXB0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtaGlzdGFmZi1hbHBoYS9zcmMvbGliL2FwcC9odHRwLWludGVyY2VwdG9ycy9iYXNlLXVybC1pbnRlcmNlcHRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFPM0MsTUFBTSxDQUFDLE1BQU0scUJBQXFCLEdBQUcsb0JBQW9CLENBQUM7QUFDMUQsTUFBTSxDQUFDLE1BQU0sMEJBQTBCLEdBQUcseUJBQXlCLENBQUM7QUFLcEUsTUFBTSxPQUFPLGtCQUFrQjtJQUUzQixZQUNZLGdCQUFrQztRQUFsQyxxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQWtCO0lBRTFDLENBQUM7SUFFTCxTQUFTLENBQUMsR0FBcUIsRUFBRSxJQUFpQjtRQUU5QyxJQUFJLEdBQUcsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLHFCQUFxQixDQUFDLEVBQUU7WUFDeEMsTUFBTSxPQUFPLEdBQUcsR0FBRyxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMscUJBQXFCLENBQUMsQ0FBQztZQUMxRCxPQUFPLElBQUksQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUMsQ0FBQztTQUM5QztRQUVELElBQUksQ0FBQyxJQUFJLENBQUMsZ0JBQWdCLENBQUMsUUFBUSxFQUFFO1lBQ2pDLE9BQU8sQ0FBQyxLQUFLLENBQUMsNEtBQTRLLENBQUMsQ0FBQTtTQUM5TDtRQUVELElBQUksR0FBRyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsMEJBQTBCLENBQUMsRUFBRTtZQUM3QyxNQUFNLGFBQWEsR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDO2dCQUM1QixHQUFHLEVBQUUsSUFBSSxDQUFDLGdCQUFnQixDQUFDLHNCQUFzQixHQUFHLEdBQUcsQ0FBQyxHQUFHO2FBQzlELENBQUMsQ0FBQztZQUNILE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztTQUNyQzthQUFNO1lBQ0gsTUFBTSxhQUFhLEdBQUcsR0FBRyxDQUFDLEtBQUssQ0FBQztnQkFDNUIsR0FBRyxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxRQUFRLEdBQUcsR0FBRyxDQUFDLEdBQUc7YUFDaEQsQ0FBQyxDQUFDO1lBQ0gsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLGFBQWEsQ0FBQyxDQUFDO1NBQ3JDO0lBQ0wsQ0FBQzs4R0E3QlEsa0JBQWtCO2tIQUFsQixrQkFBa0IsY0FGZixNQUFNOzsyRkFFVCxrQkFBa0I7a0JBSDlCLFVBQVU7bUJBQUM7b0JBQ1IsVUFBVSxFQUFFLE1BQU07aUJBQ3JCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQge1xyXG4gICAgSHR0cEludGVyY2VwdG9yLCBIdHRwSGFuZGxlciwgSHR0cFJlcXVlc3RcclxufSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XHJcblxyXG5pbXBvcnQgeyBBcHBDb25maWdTZXJ2aWNlIH0gZnJvbSAnLi4vc2VydmljZXMvYXBwLWNvbmZpZy5zZXJ2aWNlJztcclxuXHJcbmV4cG9ydCBjb25zdCBJbnRlcmNlcHRvclNraXBIZWFkZXIgPSAnWC1Ta2lwLUludGVyY2VwdG9yJztcclxuZXhwb3J0IGNvbnN0IEludGVyY2VwdG9yU3RhdGlzdGljSGVhZGVyID0gJ1gtU3RhdGlzdGljLUludGVyY2VwdG9yJztcclxuXHJcbkBJbmplY3RhYmxlKHtcclxuICAgIHByb3ZpZGVkSW46ICdyb290J1xyXG59KVxyXG5leHBvcnQgY2xhc3MgQmFzZVVybEludGVyY2VwdG9yIGltcGxlbWVudHMgSHR0cEludGVyY2VwdG9yIHtcclxuXHJcbiAgICBjb25zdHJ1Y3RvcihcclxuICAgICAgICBwcml2YXRlIGFwcENvbmZpZ1NlcnZpY2U6IEFwcENvbmZpZ1NlcnZpY2UsXHJcblxyXG4gICAgKSB7IH1cclxuXHJcbiAgICBpbnRlcmNlcHQocmVxOiBIdHRwUmVxdWVzdDxhbnk+LCBuZXh0OiBIdHRwSGFuZGxlcikge1xyXG5cclxuICAgICAgICBpZiAocmVxLmhlYWRlcnMuaGFzKEludGVyY2VwdG9yU2tpcEhlYWRlcikpIHtcclxuICAgICAgICAgICAgY29uc3QgaGVhZGVycyA9IHJlcS5oZWFkZXJzLmRlbGV0ZShJbnRlcmNlcHRvclNraXBIZWFkZXIpO1xyXG4gICAgICAgICAgICByZXR1cm4gbmV4dC5oYW5kbGUocmVxLmNsb25lKHsgaGVhZGVycyB9KSk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBpZiAoIXRoaXMuYXBwQ29uZmlnU2VydmljZS5CQVNFX1VSTCkge1xyXG4gICAgICAgICAgICBjb25zb2xlLmVycm9yKFwiVEhFIEFQUCBMQVlQT1VUIGhhZCBzdGFydGVkIHJlbmRlcmluZyBiZWZvcmUgYXBwQ29uZmlnU2VydmljZS5CQVNFX1VSTCByZWFjaHMgYSB0cnVlbHkgdmFsdWUuIFBsZWFzZSBjaGVjayBBcHBJbml0aWFsaXphdGlvblNlcnZpY2UgaW5pdGlhbGl6aW5nJC5uZXh0IG1ldGhvZCBjYXJlZnVsbHkhISFcIilcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmIChyZXEuaGVhZGVycy5oYXMoSW50ZXJjZXB0b3JTdGF0aXN0aWNIZWFkZXIpKSB7XHJcbiAgICAgICAgICAgIGNvbnN0IGFkZEJhc2VVcmxSZXEgPSByZXEuY2xvbmUoe1xyXG4gICAgICAgICAgICAgICAgdXJsOiB0aGlzLmFwcENvbmZpZ1NlcnZpY2UuQkFTRV9VUkxfRk9SX1NUQVRJU1RJQyArIHJlcS51cmxcclxuICAgICAgICAgICAgfSk7XHJcbiAgICAgICAgICAgIHJldHVybiBuZXh0LmhhbmRsZShhZGRCYXNlVXJsUmVxKTtcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICBjb25zdCBhZGRCYXNlVXJsUmVxID0gcmVxLmNsb25lKHtcclxuICAgICAgICAgICAgICAgIHVybDogdGhpcy5hcHBDb25maWdTZXJ2aWNlLkJBU0VfVVJMICsgcmVxLnVybFxyXG4gICAgICAgICAgICB9KTtcclxuICAgICAgICAgICAgcmV0dXJuIG5leHQuaGFuZGxlKGFkZEJhc2VVcmxSZXEpO1xyXG4gICAgICAgIH1cclxuICAgIH1cclxufSJdfQ==
|
|
@@ -1,19 +1,18 @@
|
|
|
1
|
-
/*
|
|
2
|
-
Tannv: This code is based on the original blog post of Alexandru Bereghici
|
|
3
|
-
https://itnext.io/angular-tutorial-implement-refresh-token-with-httpinterceptor-bfa27b966f57
|
|
4
|
-
*/
|
|
5
1
|
import { filter, finalize, switchMap, tap, take } from 'rxjs/operators';
|
|
6
2
|
import { Injectable } from '@angular/core';
|
|
7
3
|
import { BehaviorSubject, throwError, catchError } from 'rxjs';
|
|
4
|
+
import { InterceptorStatisticHeader } from "./base-url-interceptor";
|
|
8
5
|
import { HttpResponse } from '@angular/common/http';
|
|
9
6
|
import * as i0 from "@angular/core";
|
|
10
7
|
import * as i1 from "../services/auth.service";
|
|
11
|
-
import * as i2 from "../services/
|
|
12
|
-
import * as i3 from "
|
|
8
|
+
import * as i2 from "../services/statistic-auth.service";
|
|
9
|
+
import * as i3 from "../services/app-initialization.service";
|
|
10
|
+
import * as i4 from "@angular/router";
|
|
13
11
|
export class TokenInterceptor {
|
|
14
12
|
addAuthenticationToken(request) {
|
|
15
|
-
|
|
16
|
-
const
|
|
13
|
+
const isStatistic = request.headers.has(InterceptorStatisticHeader);
|
|
14
|
+
const authService = isStatistic ? this.statisticAuthService : this.auth;
|
|
15
|
+
const accessToken = authService.getAuthorizationToken();
|
|
17
16
|
// If access token is null this means that user is not logged in
|
|
18
17
|
// And we return the original request
|
|
19
18
|
if (!accessToken) {
|
|
@@ -24,8 +23,9 @@ export class TokenInterceptor {
|
|
|
24
23
|
headers: request.headers.set('Authorization', 'Bearer ' + accessToken)
|
|
25
24
|
});
|
|
26
25
|
}
|
|
27
|
-
constructor(
|
|
28
|
-
this.
|
|
26
|
+
constructor(auth, statisticAuthService, appInitializationService, router) {
|
|
27
|
+
this.auth = auth;
|
|
28
|
+
this.statisticAuthService = statisticAuthService;
|
|
29
29
|
this.appInitializationService = appInitializationService;
|
|
30
30
|
this.router = router;
|
|
31
31
|
this.refreshTokenInProgress = false;
|
|
@@ -34,6 +34,8 @@ export class TokenInterceptor {
|
|
|
34
34
|
this.refreshTokenSubject = new BehaviorSubject(null);
|
|
35
35
|
}
|
|
36
36
|
intercept(req, next) {
|
|
37
|
+
const isStatistic = req.headers.has(InterceptorStatisticHeader);
|
|
38
|
+
const authService = isStatistic ? this.statisticAuthService : this.auth;
|
|
37
39
|
if (req.url === undefined) {
|
|
38
40
|
return next.handle(req);
|
|
39
41
|
}
|
|
@@ -74,9 +76,9 @@ export class TokenInterceptor {
|
|
|
74
76
|
if (req.url.toUpperCase().includes('REFRESH_TOKEN') ||
|
|
75
77
|
req.url.toUpperCase().includes('REFRESHTOKEN')) {
|
|
76
78
|
this.appInitializationService.initializing$.next(false);
|
|
77
|
-
|
|
79
|
+
authService.userLogout().subscribe(x => {
|
|
78
80
|
if (x.ok && x.status === 200) {
|
|
79
|
-
|
|
81
|
+
authService.postLogout();
|
|
80
82
|
}
|
|
81
83
|
});
|
|
82
84
|
}
|
|
@@ -108,7 +110,7 @@ export class TokenInterceptor {
|
|
|
108
110
|
this.refreshTokenSubject.next(null);
|
|
109
111
|
// Call auth.refreshAccessToken
|
|
110
112
|
// (this is an Observable that will be returned)
|
|
111
|
-
const refreshTokenReq =
|
|
113
|
+
const refreshTokenReq = authService.tryRestoreSession();
|
|
112
114
|
return refreshTokenReq.pipe(
|
|
113
115
|
/*
|
|
114
116
|
The main difference between switchMap and other flattening operators is the cancelling effect. On each emission the previous inner observable (the result of the function you supplied) is cancelled and the new observable is subscribed. You can remember this by the phrase switch to a new observable.
|
|
@@ -116,8 +118,8 @@ export class TokenInterceptor {
|
|
|
116
118
|
switchMap((response) => {
|
|
117
119
|
let token = '';
|
|
118
120
|
if (response.ok && response.status === 200) {
|
|
119
|
-
|
|
120
|
-
|
|
121
|
+
authService.data$.next(response.body);
|
|
122
|
+
authService.authenticated$.next(true);
|
|
121
123
|
token = response.body.jwtToken;
|
|
122
124
|
}
|
|
123
125
|
// When the call to refreshToken completes
|
|
@@ -131,9 +133,9 @@ export class TokenInterceptor {
|
|
|
131
133
|
this.refreshTokenInProgress = false;
|
|
132
134
|
console.log("refreshTokenReq catchError", err);
|
|
133
135
|
debugger;
|
|
134
|
-
|
|
136
|
+
authService.userLogout().subscribe(x => {
|
|
135
137
|
if (x.ok && x.status === 200) {
|
|
136
|
-
|
|
138
|
+
authService.postLogout();
|
|
137
139
|
}
|
|
138
140
|
});
|
|
139
141
|
return throwError(() => new Error(err));
|
|
@@ -143,10 +145,10 @@ export class TokenInterceptor {
|
|
|
143
145
|
// Log when response observable either completes or errors
|
|
144
146
|
finalize(() => { }));
|
|
145
147
|
}
|
|
146
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: TokenInterceptor, deps: [{ token: i1.AuthService }, { token: i2.
|
|
148
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: TokenInterceptor, deps: [{ token: i1.AuthService }, { token: i2.StatisticAuthService }, { token: i3.AppInitializationService }, { token: i4.Router }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
147
149
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: TokenInterceptor }); }
|
|
148
150
|
}
|
|
149
151
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.3", ngImport: i0, type: TokenInterceptor, decorators: [{
|
|
150
152
|
type: Injectable
|
|
151
|
-
}], ctorParameters: () => [{ type: i1.AuthService }, { type: i2.
|
|
152
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token-interceptor.js","sourceRoot":"","sources":["../../../../../../projects/ngx-histaff-alpha/src/lib/app/http-interceptors/token-interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["/*\r\nTannv: This code is based on the original blog post of Alexandru Bereghici\r\nhttps://itnext.io/angular-tutorial-implement-refresh-token-with-httpinterceptor-bfa27b966f57\r\n*/\r\n\r\nimport { filter, finalize, switchMap, tap, take } from 'rxjs/operators';\r\nimport { Injectable } from '@angular/core';\r\nimport { Observable, BehaviorSubject, throwError, catchError } from 'rxjs';\r\nimport { HttpEvent } from '@angular/common/http';\r\nimport { Router } from '@angular/router';\r\n\r\nimport { AuthService } from '../services/auth.service';\r\nimport { AppInitializationService } from '../services/app-initialization.service';\r\n\r\nimport {\r\n    HttpInterceptor, HttpHandler, HttpRequest, HttpResponse\r\n} from '@angular/common/http';\r\n\r\n@Injectable()\r\nexport class TokenInterceptor implements HttpInterceptor {\r\n\r\n    addAuthenticationToken(request: HttpRequest<any>): HttpRequest<any> {\r\n        // Get access token from Local Storage\r\n        const accessToken = this.authService.getAuthorizationToken();\r\n\r\n        // If access token is null this means that user is not logged in\r\n        // And we return the original request\r\n        if (!accessToken) {\r\n            return request;\r\n        }\r\n\r\n        // We clone the request, because the original request is immutable\r\n        return request.clone({\r\n            headers: request.headers.set('Authorization', 'Bearer ' + accessToken)\r\n        });\r\n    }\r\n\r\n    private refreshTokenInProgress = false;\r\n    // Refresh Token Subject tracks the current token, \r\n    // or is null if no token is currently available (e.g. refresh pending).\r\n    private refreshTokenSubject: BehaviorSubject<any> = new BehaviorSubject<any>(\r\n        null\r\n    );\r\n\r\n    constructor(\r\n        private authService: AuthService,\r\n        private appInitializationService: AppInitializationService,\r\n        private router: Router\r\n    ) {\r\n    }\r\n\r\n    intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {\r\n\r\n        if (req.url === undefined) {\r\n            return next.handle(req);\r\n        }\r\n\r\n        if (req.url.includes(\"app.config.json\")) {\r\n            console.log(\"app.config.json excluded\")\r\n            return next.handle(req);\r\n        }\r\n\r\n        /* WHY???\r\n        if (req.url.includes(\"GenerateTemplate\")) {\r\n            console.log(\"GenerateTemplate excluded\")\r\n            return next.handle(req);\r\n        }\r\n        if (req.url.includes(\"ExportCorePageListGridToExcel\")) {\r\n            console.log(\"ExportCorePageListGridToExcel excluded\")\r\n            return next.handle(req);\r\n        }\r\n        */\r\n\r\n        let ok: string;\r\n        // extend server response observable with logging\r\n        return next.handle(req)\r\n            .pipe(\r\n                tap({\r\n                    // Succeeds when there is a response; ignore other events\r\n                    next: event => (ok = event instanceof HttpResponse ? 'succeeded' : ''),\r\n                    error: () => { } // we do not implement the logic here\r\n                }),\r\n                // But here:\r\n                // Operation failed; error is an HttpErrorResponse\r\n                catchError((response: any) => {\r\n\r\n                    console.log(\"TokenInterceptor Error: \", req, response)\r\n\r\n                    // We don't want to refresh token for some requests \r\n                    // like login or refresh token itself\r\n                    // So we verify url and we throw an error if it's the case\r\n                    if (\r\n                        req.url.toUpperCase().includes('LOGIN') ||\r\n                        req.url.toUpperCase().includes('REFRESH_TOKEN') ||\r\n                        req.url.toUpperCase().includes('REFRESHTOKEN')\r\n                    ) {\r\n                        // We do another check to see if refresh token failed\r\n                        // In this case we want to logout user and to redirect it to login page\r\n\r\n                        if (\r\n                            req.url.toUpperCase().includes('REFRESH_TOKEN') ||\r\n                            req.url.toUpperCase().includes('REFRESHTOKEN')\r\n\r\n                        ) {\r\n                            this.appInitializationService.initializing$.next(false);\r\n                            this.authService.userLogout().subscribe(x => {\r\n                                if (x.ok && x.status === 200) {\r\n                                    this.authService.postLogout();\r\n                                }\r\n                            })\r\n\r\n                        }\r\n\r\n                        return throwError(() => new Error(JSON.stringify(response)));\r\n\r\n                    }\r\n\r\n                    // If error status is different than 401 \r\n                    // we want to skip refresh token\r\n                    // So we check that and throw the error if it's the case\r\n                    if (response.status !== 401) {\r\n                        return throwError(() => new Error(response.error));\r\n                    }\r\n\r\n                    if (this.refreshTokenInProgress) {\r\n                        // If refreshTokenInProgress is true, we will wait until refreshTokenSubject has a non-null value\r\n                        // – which means the new token is ready and we can retry the request again\r\n                        return this.refreshTokenSubject.pipe(\r\n                            // filter: Emit values that pass the provided condition\r\n                            filter(result => result !== null),\r\n\r\n                            take(1),\r\n                            /*\r\n                            The main difference between switchMap and other flattening operators is the cancelling effect. On each emission the previous inner observable (the result of the function you supplied) is cancelled and the new observable is subscribed. You can remember this by the phrase switch to a new observable.\r\n                            */\r\n                            switchMap(() => next.handle(this.addAuthenticationToken(req)))\r\n                        )\r\n                    } else {\r\n                        console.error(response.error?.message?.toUpperCase() || response);\r\n                        this.refreshTokenInProgress = true;\r\n\r\n                        // Set the refreshTokenSubject to null \r\n                        // so that subsequent API calls will wait\r\n                        // until the new token has been retrieved\r\n                        this.refreshTokenSubject.next(null);\r\n\r\n                        // Call auth.refreshAccessToken\r\n                        // (this is an Observable that will be returned)\r\n\r\n                        const refreshTokenReq = this.authService.tryRestoreSession();\r\n\r\n                        return refreshTokenReq.pipe(\r\n                            /*\r\n                            The main difference between switchMap and other flattening operators is the cancelling effect. On each emission the previous inner observable (the result of the function you supplied) is cancelled and the new observable is subscribed. You can remember this by the phrase switch to a new observable.\r\n                            */\r\n                            switchMap((response: any) => {\r\n                                let token = '';\r\n\r\n                                if (response.ok && response.status === 200) {\r\n                                    this.authService.data$.next(response.body);\r\n                                    this.authService.authenticated$.next(true);\r\n                                    token = response.body.jwtToken;\r\n                                }\r\n\r\n                                // When the call to refreshToken completes \r\n                                // we reset the refreshTokenInProgress to false\r\n                                // for the next time the token needs to be refreshed\r\n                                this.refreshTokenInProgress = false;\r\n                                this.refreshTokenSubject.next(token);\r\n                                const newReq = this.addAuthenticationToken(req);\r\n                                return next.handle(newReq);\r\n                            }),\r\n                            catchError((err: any) => {\r\n                                this.refreshTokenInProgress = false;\r\n\r\n                                console.log(\"refreshTokenReq catchError\", err)\r\n                                debugger\r\n                                this.authService.userLogout().subscribe(x => {\r\n                                    if (x.ok && x.status === 200) {\r\n                                        this.authService.postLogout();\r\n                                    }\r\n                                })\r\n\r\n                                return throwError(() => new Error(err));\r\n                            })\r\n                        )\r\n                    }\r\n                }),\r\n                // Log when response observable either completes or errors\r\n                finalize(() => { })\r\n            );\r\n    }\r\n\r\n}"]}
|
|
153
|
+
}], ctorParameters: () => [{ type: i1.AuthService }, { type: i2.StatisticAuthService }, { type: i3.AppInitializationService }, { type: i4.Router }] });
|
|
154
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token-interceptor.js","sourceRoot":"","sources":["../../../../../../projects/ngx-histaff-alpha/src/lib/app/http-interceptors/token-interceptor.ts"],"names":[],"mappings":"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","sourcesContent":["import { filter, finalize, switchMap, tap, take } from 'rxjs/operators';\r\nimport { Injectable } from '@angular/core';\r\nimport { Observable, BehaviorSubject, throwError, catchError } from 'rxjs';\r\nimport { HttpEvent } from '@angular/common/http';\r\nimport { Router } from '@angular/router';\r\n\r\nimport { AuthService } from '../services/auth.service';\r\nimport { AppInitializationService } from '../services/app-initialization.service';\r\nimport { InterceptorStatisticHeader } from \"./base-url-interceptor\";\r\nimport { StatisticAuthService } from '../services/statistic-auth.service';\r\n\r\nimport {\r\n    HttpInterceptor, HttpHandler, HttpRequest, HttpResponse\r\n} from '@angular/common/http';\r\n\r\n@Injectable()\r\nexport class TokenInterceptor implements HttpInterceptor {\r\n\r\n    addAuthenticationToken(request: HttpRequest<any>): HttpRequest<any> {\r\n\r\n        const isStatistic = request.headers.has(InterceptorStatisticHeader);\r\n        const authService = isStatistic ? this.statisticAuthService : this.auth;\r\n\r\n        const accessToken = authService.getAuthorizationToken();\r\n\r\n        // If access token is null this means that user is not logged in\r\n        // And we return the original request\r\n        if (!accessToken) {\r\n            return request;\r\n        }\r\n\r\n        // We clone the request, because the original request is immutable\r\n        return request.clone({\r\n            headers: request.headers.set('Authorization', 'Bearer ' + accessToken)\r\n        });\r\n    }\r\n\r\n    private refreshTokenInProgress = false;\r\n    // Refresh Token Subject tracks the current token, \r\n    // or is null if no token is currently available (e.g. refresh pending).\r\n    private refreshTokenSubject: BehaviorSubject<any> = new BehaviorSubject<any>(\r\n        null\r\n    );\r\n\r\n    constructor(\r\n        private auth: AuthService,\r\n        private statisticAuthService: StatisticAuthService,\r\n        private appInitializationService: AppInitializationService,\r\n        private router: Router\r\n    ) {\r\n    }\r\n\r\n    intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {\r\n\r\n        const isStatistic = req.headers.has(InterceptorStatisticHeader);\r\n        const authService = isStatistic ? this.statisticAuthService : this.auth;\r\n\r\n\r\n        if (req.url === undefined) {\r\n            return next.handle(req);\r\n        }\r\n\r\n        if (req.url.includes(\"app.config.json\")) {\r\n            console.log(\"app.config.json excluded\")\r\n            return next.handle(req);\r\n        }\r\n\r\n        /* WHY???\r\n        if (req.url.includes(\"GenerateTemplate\")) {\r\n            console.log(\"GenerateTemplate excluded\")\r\n            return next.handle(req);\r\n        }\r\n        if (req.url.includes(\"ExportCorePageListGridToExcel\")) {\r\n            console.log(\"ExportCorePageListGridToExcel excluded\")\r\n            return next.handle(req);\r\n        }\r\n        */\r\n\r\n        let ok: string;\r\n        // extend server response observable with logging\r\n        return next.handle(req)\r\n            .pipe(\r\n                tap({\r\n                    // Succeeds when there is a response; ignore other events\r\n                    next: event => (ok = event instanceof HttpResponse ? 'succeeded' : ''),\r\n                    error: () => { } // we do not implement the logic here\r\n                }),\r\n                // But here:\r\n                // Operation failed; error is an HttpErrorResponse\r\n                catchError((response: any) => {\r\n\r\n                    console.log(\"TokenInterceptor Error: \", req, response)\r\n\r\n                    // We don't want to refresh token for some requests \r\n                    // like login or refresh token itself\r\n                    // So we verify url and we throw an error if it's the case\r\n                    if (\r\n                        req.url.toUpperCase().includes('LOGIN') ||\r\n                        req.url.toUpperCase().includes('REFRESH_TOKEN') ||\r\n                        req.url.toUpperCase().includes('REFRESHTOKEN')\r\n                    ) {\r\n                        // We do another check to see if refresh token failed\r\n                        // In this case we want to logout user and to redirect it to login page\r\n\r\n                        if (\r\n                            req.url.toUpperCase().includes('REFRESH_TOKEN') ||\r\n                            req.url.toUpperCase().includes('REFRESHTOKEN')\r\n\r\n                        ) {\r\n                            this.appInitializationService.initializing$.next(false);\r\n                            authService.userLogout().subscribe(x => {\r\n                                if (x.ok && x.status === 200) {\r\n                                    authService.postLogout();\r\n                                }\r\n                            })\r\n\r\n                        }\r\n\r\n                        return throwError(() => new Error(JSON.stringify(response)));\r\n\r\n                    }\r\n\r\n                    // If error status is different than 401 \r\n                    // we want to skip refresh token\r\n                    // So we check that and throw the error if it's the case\r\n                    if (response.status !== 401) {\r\n                        return throwError(() => new Error(response.error));\r\n                    }\r\n\r\n                    if (this.refreshTokenInProgress) {\r\n                        // If refreshTokenInProgress is true, we will wait until refreshTokenSubject has a non-null value\r\n                        // – which means the new token is ready and we can retry the request again\r\n                        return this.refreshTokenSubject.pipe(\r\n                            // filter: Emit values that pass the provided condition\r\n                            filter(result => result !== null),\r\n\r\n                            take(1),\r\n                            /*\r\n                            The main difference between switchMap and other flattening operators is the cancelling effect. On each emission the previous inner observable (the result of the function you supplied) is cancelled and the new observable is subscribed. You can remember this by the phrase switch to a new observable.\r\n                            */\r\n                            switchMap(() => next.handle(this.addAuthenticationToken(req)))\r\n                        )\r\n                    } else {\r\n                        console.error(response.error?.message?.toUpperCase() || response);\r\n                        this.refreshTokenInProgress = true;\r\n\r\n                        // Set the refreshTokenSubject to null \r\n                        // so that subsequent API calls will wait\r\n                        // until the new token has been retrieved\r\n                        this.refreshTokenSubject.next(null);\r\n\r\n                        // Call auth.refreshAccessToken\r\n                        // (this is an Observable that will be returned)\r\n\r\n                        const refreshTokenReq = authService.tryRestoreSession();\r\n\r\n                        return refreshTokenReq.pipe(\r\n                            /*\r\n                            The main difference between switchMap and other flattening operators is the cancelling effect. On each emission the previous inner observable (the result of the function you supplied) is cancelled and the new observable is subscribed. You can remember this by the phrase switch to a new observable.\r\n                            */\r\n                            switchMap((response: any) => {\r\n                                let token = '';\r\n\r\n                                if (response.ok && response.status === 200) {\r\n                                    authService.data$.next(response.body);\r\n                                    authService.authenticated$.next(true);\r\n                                    token = response.body.jwtToken;\r\n                                }\r\n\r\n                                // When the call to refreshToken completes \r\n                                // we reset the refreshTokenInProgress to false\r\n                                // for the next time the token needs to be refreshed\r\n                                this.refreshTokenInProgress = false;\r\n                                this.refreshTokenSubject.next(token);\r\n                                const newReq = this.addAuthenticationToken(req);\r\n                                return next.handle(newReq);\r\n                            }),\r\n                            catchError((err: any) => {\r\n                                this.refreshTokenInProgress = false;\r\n\r\n                                console.log(\"refreshTokenReq catchError\", err)\r\n                                debugger\r\n                                authService.userLogout().subscribe(x => {\r\n                                    if (x.ok && x.status === 200) {\r\n                                        authService.postLogout();\r\n                                    }\r\n                                })\r\n\r\n                                return throwError(() => new Error(err));\r\n                            })\r\n                        )\r\n                    }\r\n                }),\r\n                // Log when response observable either completes or errors\r\n                finalize(() => { })\r\n            );\r\n    }\r\n\r\n}"]}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiSVN0YXRpc3RpY0F1dGhEYXRhLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvbmd4LWhpc3RhZmYtYWxwaGEvc3JjL2xpYi9hcHAvaW50ZXJmYWNlcy9JU3RhdGlzdGljQXV0aERhdGEudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IElTdGF0aXN0aWNTeXNVc2VyIH0gZnJvbSBcIi4vSVN0YXRpc3RpY1N5c1VzZXJcIjtcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgSVN0YXRpc3RpY0F1dGhEYXRhIHtcclxuICAgIHVzZXI6IElTdGF0aXN0aWNTeXNVc2VyLFxyXG4gICAgYWNjZXNzVG9rZW46IHN0cmluZyxcclxuICAgIGV4cGlyZXM6IERhdGUsXHJcbiAgICByZWZyZXNoVG9rZW46IHsgdG9rZW46IHN0cmluZyB9XHJcbn0iXX0=
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiSVN0YXRpc3RpY1N5c1VzZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9uZ3gtaGlzdGFmZi1hbHBoYS9zcmMvbGliL2FwcC9pbnRlcmZhY2VzL0lTdGF0aXN0aWNTeXNVc2VyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIElTdGF0aXN0aWNTeXNVc2VyIHtcclxuICAgIGlkOiBzdHJpbmcsXHJcbiAgICB0ZW5hbnRJZDogbnVtYmVyLFxyXG4gICAgdXNlcm5hbWU6IHN0cmluZyxcclxuICAgIGVtYWlsOiBzdHJpbmcsXHJcbiAgICBmdWxsbmFtZTogc3RyaW5nLFxyXG4gICAgYXZhdGFyOiBzdHJpbmcsXHJcbiAgICBpc1Jvb3Q/OiBib29sZWFuLFxyXG4gICAgaXNBZG1pbj86IGJvb2xlYW4sXHJcbiAgICBpc0xvY2tlZD86IGJvb2xlYW4sXHJcbiAgICB3ZWJhcHBBbGxvd2VkPzogYm9vbGVhbixcclxuICAgIHBvcnRhbEFsbG93ZWQ/OiBib29sZWFuLFxyXG4gICAgbW9iaWxlQWxsb3dlZD86IGJvb2xlYW4sXHJcbiAgICBpc0ZpcnN0TG9naW4/OiBib29sZWFuLFxyXG59Il19
|