@sambath999/localize-token 12.4.11 → 12.4.13
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/bundles/sambath999-localize-token.umd.js +400 -1125
- package/bundles/sambath999-localize-token.umd.js.map +1 -1
- package/{localize-logindlg/localize-logindlg.component.d.ts → components/localize-token-login-dialog.component.d.ts} +14 -17
- package/esm2015/components/localize-token-login-dialog.component.js +395 -0
- package/esm2015/models/localize-token-models.js +2 -0
- package/esm2015/models/localize-token.module.js +26 -0
- package/esm2015/public-api.js +6 -15
- package/esm2015/sambath999-localize-token.js +1 -1
- package/esm2015/services/localize-token-configure.service.js +32 -0
- package/esm2015/services/localize-token-dialog.service.js +64 -0
- package/fesm2015/sambath999-localize-token.js +64 -906
- package/fesm2015/sambath999-localize-token.js.map +1 -1
- package/{localize-logindlg/localize-logindlg.service.d.ts → models/localize-token-models.d.ts} +1 -10
- package/package.json +2 -2
- package/public-api.d.ts +5 -13
- package/sambath999-localize-token.metadata.json +1 -1
- package/services/localize-token-configure.service.d.ts +9 -0
- package/services/localize-token-dialog.service.d.ts +10 -0
- package/README.md +0 -57
- package/esm2015/localize-api-token/localize-api-token.module.js +0 -21
- package/esm2015/localize-api-token/localize-api-token.service.js +0 -34
- package/esm2015/localize-logindlg/localize-logindlg.component.js +0 -423
- package/esm2015/localize-logindlg/localize-logindlg.module.js +0 -28
- package/esm2015/localize-logindlg/localize-logindlg.service.js +0 -64
- package/esm2015/localize-token/helpers/interfaces.js +0 -19
- package/esm2015/localize-token/helpers/localize.api.assets.js +0 -20
- package/esm2015/localize-token/helpers/loccalize.api.helper.js +0 -267
- package/esm2015/localize-token/localize.api.service.js +0 -242
- package/esm2015/localize-token/localize.token.js +0 -60
- package/esm2015/localize-token/localize.token.module.js +0 -14
- package/esm2015/localize-token/localize.token.service.js +0 -94
- package/esm2015/localize-token/localize.token.storage.js +0 -107
- package/localize-api-token/localize-api-token.module.d.ts +0 -2
- package/localize-api-token/localize-api-token.service.d.ts +0 -12
- package/localize-logindlg/localize-logindlg.module.d.ts +0 -2
- package/localize-token/helpers/interfaces.d.ts +0 -89
- package/localize-token/helpers/localize.api.assets.d.ts +0 -5
- package/localize-token/helpers/loccalize.api.helper.d.ts +0 -32
- package/localize-token/localize.api.service.d.ts +0 -55
- package/localize-token/localize.token.d.ts +0 -55
- package/localize-token/localize.token.service.d.ts +0 -36
- package/localize-token/localize.token.storage.d.ts +0 -61
- /package/{localize-token/localize.token.module.d.ts → models/localize-token.module.d.ts} +0 -0
|
@@ -1,20 +0,0 @@
|
|
|
1
|
-
export const LOCALIZE_API_ASSETS = {
|
|
2
|
-
network: {
|
|
3
|
-
noConnection: `<?xml version="1.0" encoding="UTF-8"?>
|
|
4
|
-
<svg id="lze-no-connection" data-name="Layer 2" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 77.29 70.98">
|
|
5
|
-
<defs>
|
|
6
|
-
<style> .cls-1 { fill: #fff; } .cls-2, .cls-3 { fill: #e30613; } .cls-3 { stroke: #e30613; stroke-miterlimit: 10; stroke-width: .25px; } </style>
|
|
7
|
-
</defs>
|
|
8
|
-
<g id="Layer_3" data-name="Layer 3">
|
|
9
|
-
<g>
|
|
10
|
-
<path class="cls-1" d="m73.29,35c-1.2,0-2.33-.53-3.09-1.46-.48-.57-11.23-13.2-31.55-13.2s-31.11,12.66-31.56,13.2c-1.4,1.71-3.92,1.95-5.63.55-1.71-1.4-1.95-3.92-.55-5.63.54-.66,13.47-16.12,37.74-16.12s37.2,15.46,37.74,16.12c1.4,1.71,1.15,4.23-.56,5.62-.71.58-1.61.9-2.53.9Z"/>
|
|
11
|
-
<path class="cls-1" d="m63.96,45.66c-1.19,0-2.32-.53-3.08-1.44-5.79-6.05-13.86-9.39-22.24-9.21-8.38-.18-16.45,3.16-22.24,9.22-1.46,1.65-3.99,1.81-5.64.35-1.57-1.39-1.8-3.77-.52-5.43,7.32-7.89,17.64-12.29,28.4-12.12,10.76-.17,21.08,4.24,28.4,12.12,1.4,1.71,1.15,4.23-.56,5.62-.71.58-1.6.9-2.53.9Z"/>
|
|
12
|
-
<path class="cls-1" d="m53.3,56.32c-1.24,0-2.41-.57-3.16-1.55-5.73-6.35-15.52-6.85-21.87-1.13-.4.36-.77.73-1.13,1.13-1.36,1.73-3.88,2.03-5.61.67-1.71-1.34-2.03-3.8-.73-5.54,8.39-9.85,23.18-11.04,33.03-2.65.95.81,1.84,1.69,2.65,2.65,1.34,1.75,1,4.26-.75,5.6-.7.53-1.55.82-2.43.82Z"/>
|
|
13
|
-
</g>
|
|
14
|
-
<path class="cls-2" d="m47.21,9.45l-4.06,41.36c-.64,5.42-8.39,5.39-9.01,0,0,0-4.06-41.36-4.06-41.36-.46-4.73,2.99-8.94,7.72-9.4,5.33-.58,9.97,4.09,9.4,9.4h0Z"/>
|
|
15
|
-
<circle class="cls-3" cx="38.64" cy="64.79" r="6.07"/>
|
|
16
|
-
</g>
|
|
17
|
-
</svg>`
|
|
18
|
-
}
|
|
19
|
-
};
|
|
20
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9jYWxpemUuYXBpLmFzc2V0cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL0FOR1VMQVIvTElCUkFSSUVTL2xvY2FsaXplLnRva2VuLjEyLjEuMC9kZXYvc3JjL2xvY2FsaXplLXRva2VuL2hlbHBlcnMvbG9jYWxpemUuYXBpLmFzc2V0cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRztJQUMvQixPQUFPLEVBQUU7UUFDTCxZQUFZLEVBQUU7Ozs7Ozs7Ozs7Ozs7O21CQWNIO0tBQ2Q7Q0FDSixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IExPQ0FMSVpFX0FQSV9BU1NFVFMgPSB7XHJcbiAgICBuZXR3b3JrOiB7XHJcbiAgICAgICAgbm9Db25uZWN0aW9uOiBgPD94bWwgdmVyc2lvbj1cIjEuMFwiIGVuY29kaW5nPVwiVVRGLThcIj8+XHJcbiAgICAgICAgICAgIDxzdmcgaWQ9XCJsemUtbm8tY29ubmVjdGlvblwiIGRhdGEtbmFtZT1cIkxheWVyIDJcIiB4bWxucz1cImh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnXCIgdmlld0JveD1cIjAgMCA3Ny4yOSA3MC45OFwiPlxyXG4gICAgICAgICAgICA8ZGVmcz5cclxuICAgICAgICAgICAgICAgIDxzdHlsZT4gLmNscy0xIHsgZmlsbDogI2ZmZjsgfSAuY2xzLTIsIC5jbHMtMyB7IGZpbGw6ICNlMzA2MTM7IH0gLmNscy0zIHsgc3Ryb2tlOiAjZTMwNjEzOyBzdHJva2UtbWl0ZXJsaW1pdDogMTA7IHN0cm9rZS13aWR0aDogLjI1cHg7IH0gPC9zdHlsZT5cclxuICAgICAgICAgICAgPC9kZWZzPlxyXG4gICAgICAgICAgICA8ZyBpZD1cIkxheWVyXzNcIiBkYXRhLW5hbWU9XCJMYXllciAzXCI+XHJcbiAgICAgICAgICAgICAgICA8Zz5cclxuICAgICAgICAgICAgICAgIDxwYXRoIGNsYXNzPVwiY2xzLTFcIiBkPVwibTczLjI5LDM1Yy0xLjIsMC0yLjMzLS41My0zLjA5LTEuNDYtLjQ4LS41Ny0xMS4yMy0xMy4yLTMxLjU1LTEzLjJzLTMxLjExLDEyLjY2LTMxLjU2LDEzLjJjLTEuNCwxLjcxLTMuOTIsMS45NS01LjYzLjU1LTEuNzEtMS40LTEuOTUtMy45Mi0uNTUtNS42My41NC0uNjYsMTMuNDctMTYuMTIsMzcuNzQtMTYuMTJzMzcuMiwxNS40NiwzNy43NCwxNi4xMmMxLjQsMS43MSwxLjE1LDQuMjMtLjU2LDUuNjItLjcxLjU4LTEuNjEuOS0yLjUzLjlaXCIvPlxyXG4gICAgICAgICAgICAgICAgPHBhdGggY2xhc3M9XCJjbHMtMVwiIGQ9XCJtNjMuOTYsNDUuNjZjLTEuMTksMC0yLjMyLS41My0zLjA4LTEuNDQtNS43OS02LjA1LTEzLjg2LTkuMzktMjIuMjQtOS4yMS04LjM4LS4xOC0xNi40NSwzLjE2LTIyLjI0LDkuMjItMS40NiwxLjY1LTMuOTksMS44MS01LjY0LjM1LTEuNTctMS4zOS0xLjgtMy43Ny0uNTItNS40Myw3LjMyLTcuODksMTcuNjQtMTIuMjksMjguNC0xMi4xMiwxMC43Ni0uMTcsMjEuMDgsNC4yNCwyOC40LDEyLjEyLDEuNCwxLjcxLDEuMTUsNC4yMy0uNTYsNS42Mi0uNzEuNTgtMS42LjktMi41My45WlwiLz5cclxuICAgICAgICAgICAgICAgIDxwYXRoIGNsYXNzPVwiY2xzLTFcIiBkPVwibTUzLjMsNTYuMzJjLTEuMjQsMC0yLjQxLS41Ny0zLjE2LTEuNTUtNS43My02LjM1LTE1LjUyLTYuODUtMjEuODctMS4xMy0uNC4zNi0uNzcuNzMtMS4xMywxLjEzLTEuMzYsMS43My0zLjg4LDIuMDMtNS42MS42Ny0xLjcxLTEuMzQtMi4wMy0zLjgtLjczLTUuNTQsOC4zOS05Ljg1LDIzLjE4LTExLjA0LDMzLjAzLTIuNjUuOTUuODEsMS44NCwxLjY5LDIuNjUsMi42NSwxLjM0LDEuNzUsMSw0LjI2LS43NSw1LjYtLjcuNTMtMS41NS44Mi0yLjQzLjgyWlwiLz5cclxuICAgICAgICAgICAgICAgIDwvZz5cclxuICAgICAgICAgICAgICAgIDxwYXRoIGNsYXNzPVwiY2xzLTJcIiBkPVwibTQ3LjIxLDkuNDVsLTQuMDYsNDEuMzZjLS42NCw1LjQyLTguMzksNS4zOS05LjAxLDAsMCwwLTQuMDYtNDEuMzYtNC4wNi00MS4zNi0uNDYtNC43MywyLjk5LTguOTQsNy43Mi05LjQsNS4zMy0uNTgsOS45Nyw0LjA5LDkuNCw5LjRoMFpcIi8+XHJcbiAgICAgICAgICAgICAgICA8Y2lyY2xlIGNsYXNzPVwiY2xzLTNcIiBjeD1cIjM4LjY0XCIgY3k9XCI2NC43OVwiIHI9XCI2LjA3XCIvPlxyXG4gICAgICAgICAgICA8L2c+XHJcbiAgICAgICAgICAgIDwvc3ZnPmBcclxuICAgIH1cclxufSJdfQ==
|
|
@@ -1,267 +0,0 @@
|
|
|
1
|
-
import { __awaiter } from "tslib";
|
|
2
|
-
import { takeUntil, catchError, throwError, map } from "rxjs";
|
|
3
|
-
import { waitFor } from "../localize.token";
|
|
4
|
-
import { LOCALIZE_API_ASSETS } from "./localize.api.assets";
|
|
5
|
-
class LocalizeApiHelper {
|
|
6
|
-
constructor() {
|
|
7
|
-
this.defaultRetryOptions = {
|
|
8
|
-
connectionError: {
|
|
9
|
-
message: 'Connection error occurred. Please wait',
|
|
10
|
-
blockScreen: true,
|
|
11
|
-
blockScreenZIndex: 10000
|
|
12
|
-
}
|
|
13
|
-
};
|
|
14
|
-
}
|
|
15
|
-
performRetry(options) {
|
|
16
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
17
|
-
let attempts = 0;
|
|
18
|
-
let lastError;
|
|
19
|
-
let consoleCount = 0;
|
|
20
|
-
// Merge default retry options with provided options
|
|
21
|
-
options = Object.assign(Object.assign({}, this.defaultRetryOptions), options);
|
|
22
|
-
let styleElement;
|
|
23
|
-
while (attempts < options.maxRetries()) {
|
|
24
|
-
try {
|
|
25
|
-
const result = yield options.callback();
|
|
26
|
-
this.removeBlocker(styleElement);
|
|
27
|
-
return result;
|
|
28
|
-
}
|
|
29
|
-
catch (error) {
|
|
30
|
-
lastError = error;
|
|
31
|
-
if (consoleCount >= 7) {
|
|
32
|
-
console.clear();
|
|
33
|
-
consoleCount = 0;
|
|
34
|
-
}
|
|
35
|
-
if (options.retryUnless && !options.retryUnless(error))
|
|
36
|
-
throw error; // If the error should not be retried, rethrow it
|
|
37
|
-
// Handle connection error
|
|
38
|
-
styleElement = yield this.onConnectionError(options, error);
|
|
39
|
-
if (options.onError)
|
|
40
|
-
yield this.invokeHook(options.onError.bind(this, error));
|
|
41
|
-
if (attempts >= options.maxRetries() - 1)
|
|
42
|
-
throw error;
|
|
43
|
-
attempts++;
|
|
44
|
-
consoleCount++;
|
|
45
|
-
console.warn(`Attempt ${attempts} failed. Retrying...`, error);
|
|
46
|
-
yield waitFor(options.delay);
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
console.warn(`Failed after ${options.maxRetries()} attempts`);
|
|
50
|
-
throw lastError;
|
|
51
|
-
});
|
|
52
|
-
}
|
|
53
|
-
performRequestWithRetry(options, config, performRequest) {
|
|
54
|
-
var _a, _b, _c, _d;
|
|
55
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
56
|
-
const retryUnless = ((_a = config.retryOptions) === null || _a === void 0 ? void 0 : _a.retryFunction)
|
|
57
|
-
|| this.isConnectionError;
|
|
58
|
-
return yield this.performRetry({
|
|
59
|
-
connectionError: (_b = config.retryOptions) === null || _b === void 0 ? void 0 : _b.onConnectionError,
|
|
60
|
-
maxRetries: () => { var _a, _b; return (_b = (_a = config.retryOptions) === null || _a === void 0 ? void 0 : _a.maxRetries) !== null && _b !== void 0 ? _b : 1000; },
|
|
61
|
-
delay: (_d = (_c = config.retryOptions) === null || _c === void 0 ? void 0 : _c.delay) !== null && _d !== void 0 ? _d : 500,
|
|
62
|
-
callback: () => performRequest(options),
|
|
63
|
-
retryUnless: retryUnless,
|
|
64
|
-
});
|
|
65
|
-
});
|
|
66
|
-
}
|
|
67
|
-
buildUrl(baseUrl, path) {
|
|
68
|
-
const normalizedUrl = `${baseUrl.trim().replace(/\/?$/, '/')}${path.trim().replace(/^\//, '')}`;
|
|
69
|
-
return normalizedUrl.endsWith('/')
|
|
70
|
-
? normalizedUrl.slice(0, -1)
|
|
71
|
-
: normalizedUrl;
|
|
72
|
-
}
|
|
73
|
-
invokeHook(callback) {
|
|
74
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
75
|
-
if (!callback)
|
|
76
|
-
return;
|
|
77
|
-
const result = callback();
|
|
78
|
-
if (result instanceof Promise) {
|
|
79
|
-
yield result;
|
|
80
|
-
}
|
|
81
|
-
});
|
|
82
|
-
}
|
|
83
|
-
createRequest(instance, method, url, body, options, onInvalidResponseBody) {
|
|
84
|
-
const request$ = instance.client.request(method, url, Object.assign(Object.assign({}, options), { body })).pipe(takeUntil(instance.destroy$()), map((body) => extractJsonFromResponse(body, onInvalidResponseBody)), catchError((error) => {
|
|
85
|
-
// Convert to a non-observable error to handle in the promise
|
|
86
|
-
return throwError(() => error);
|
|
87
|
-
}));
|
|
88
|
-
return request$;
|
|
89
|
-
}
|
|
90
|
-
defaultRetryFunction(error) {
|
|
91
|
-
// Don't retry for other errors (like 400, 401, 403, etc.)
|
|
92
|
-
if (!this.isConnectionError(error))
|
|
93
|
-
throw error;
|
|
94
|
-
return true;
|
|
95
|
-
}
|
|
96
|
-
isConnectionError(error) {
|
|
97
|
-
const isNetworkError = error.status === 0;
|
|
98
|
-
const isServerError = error.status >= 1000 && error.status < 600;
|
|
99
|
-
return isNetworkError || isServerError;
|
|
100
|
-
}
|
|
101
|
-
onConnectionError(options, error) {
|
|
102
|
-
var _a;
|
|
103
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
104
|
-
if (!options.connectionError)
|
|
105
|
-
return;
|
|
106
|
-
let styleElement;
|
|
107
|
-
if (this.isConnectionError(error)) {
|
|
108
|
-
styleElement = this.screenBlocker(options, error, true);
|
|
109
|
-
yield this.invokeHook((_a = options.connectionError.callback) === null || _a === void 0 ? void 0 : _a.bind(this, error));
|
|
110
|
-
return styleElement;
|
|
111
|
-
}
|
|
112
|
-
else {
|
|
113
|
-
this.screenBlocker(options, error, false);
|
|
114
|
-
styleElement === null || styleElement === void 0 ? void 0 : styleElement.remove();
|
|
115
|
-
}
|
|
116
|
-
});
|
|
117
|
-
}
|
|
118
|
-
screenBlocker(optons, error, add = true) {
|
|
119
|
-
var _a, _b, _c, _d, _e;
|
|
120
|
-
if (!((_a = optons.connectionError) === null || _a === void 0 ? void 0 : _a.blockScreen))
|
|
121
|
-
return;
|
|
122
|
-
const message = ((_b = optons.connectionError) === null || _b === void 0 ? void 0 : _b.message)
|
|
123
|
-
|| 'Connection error occurred. Please wait';
|
|
124
|
-
const errorMessage = ((_c = error === null || error === void 0 ? void 0 : error.error) === null || _c === void 0 ? void 0 : _c.message) || 'An error occurred';
|
|
125
|
-
const suggestinMessage = ((_d = optons.connectionError) === null || _d === void 0 ? void 0 : _d.suggestionMessage)
|
|
126
|
-
|| 'Please check your internet connection or the server status.';
|
|
127
|
-
const zIndex = ((_e = optons.connectionError) === null || _e === void 0 ? void 0 : _e.blockScreenZIndex) || 10000;
|
|
128
|
-
const body = document.body;
|
|
129
|
-
const blcokerHtml = `
|
|
130
|
-
<div class="lze-blocker">
|
|
131
|
-
${LOCALIZE_API_ASSETS.network.noConnection}
|
|
132
|
-
<div class="lze-blocker__message">
|
|
133
|
-
${message}
|
|
134
|
-
<span class="lze-blocker__dotting">
|
|
135
|
-
<span class="lze-blocker__dot"></span>
|
|
136
|
-
<span class="lze-blocker__dot"></span>
|
|
137
|
-
<span class="lze-blocker__dot"></span>
|
|
138
|
-
</span>
|
|
139
|
-
</div>
|
|
140
|
-
<div class="lze-blocker__error">${errorMessage}</div>
|
|
141
|
-
<div class="lze-blocker__error_suggestion">${suggestinMessage}</div>
|
|
142
|
-
</div>
|
|
143
|
-
`;
|
|
144
|
-
const style = `
|
|
145
|
-
div.lze-blocker {
|
|
146
|
-
position: fixed;
|
|
147
|
-
top: 0;
|
|
148
|
-
left: 0;
|
|
149
|
-
width: 100%;
|
|
150
|
-
height: 100%;
|
|
151
|
-
background: rgba(0, 0, 0, 0.85) !important;
|
|
152
|
-
z-index: ${zIndex};
|
|
153
|
-
display: flex;
|
|
154
|
-
align-items: center;
|
|
155
|
-
justify-content: center;
|
|
156
|
-
flex-direction: column;
|
|
157
|
-
color: #fff !important;
|
|
158
|
-
font-family: Arial, sans-serif;
|
|
159
|
-
text-align: center;
|
|
160
|
-
padding: 20px;
|
|
161
|
-
box-sizing: border-box;
|
|
162
|
-
overflow: hidden;
|
|
163
|
-
user-select: none;
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
svg#lze-no-connection {
|
|
167
|
-
width: 75px;
|
|
168
|
-
height: 75px;
|
|
169
|
-
margin-bottom: 20px;
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
div.lze-blocker__message {
|
|
173
|
-
color: #fff !important;
|
|
174
|
-
font-size: 18px !important;
|
|
175
|
-
margin-bottom: 10px;
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
.lze-blocker__dotting {
|
|
179
|
-
display: inline-block;
|
|
180
|
-
vertical-align: middle;
|
|
181
|
-
}
|
|
182
|
-
span.lze-blocker__dot {
|
|
183
|
-
display: inline-block;
|
|
184
|
-
width: 7px;
|
|
185
|
-
height: 7px;
|
|
186
|
-
background-color: #ffffff !important;
|
|
187
|
-
border-radius: 50%;
|
|
188
|
-
margin-left: 3px;
|
|
189
|
-
opacity: 0.3;
|
|
190
|
-
animation: dotting 1s infinite;
|
|
191
|
-
}
|
|
192
|
-
.lze-blocker__dot:nth-child(1) {
|
|
193
|
-
animation-delay: 0s;
|
|
194
|
-
opacity: 1;
|
|
195
|
-
}
|
|
196
|
-
.lze-blocker__dot:nth-child(2) {
|
|
197
|
-
animation-delay: 0.2s;
|
|
198
|
-
}
|
|
199
|
-
.lze-blocker__dot:nth-child(3) {
|
|
200
|
-
animation-delay: 0.4s;
|
|
201
|
-
}
|
|
202
|
-
|
|
203
|
-
@keyframes dotting {
|
|
204
|
-
0%, 80%, 100% { opacity: 0.3; }
|
|
205
|
-
40% { opacity: 1; }
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
div.lze-blocker__error {
|
|
209
|
-
color: #f00;
|
|
210
|
-
font-size: 14px !important;
|
|
211
|
-
margin-bottom: 10px;
|
|
212
|
-
text-shadow: 0 0 1px #ff5f5f !important;
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
div.lze-blocker__error_suggestion {
|
|
216
|
-
color: #ccc !important;
|
|
217
|
-
font-size: 14px !important;
|
|
218
|
-
margin-top: 10px;
|
|
219
|
-
}
|
|
220
|
-
|
|
221
|
-
@keyframes spin {
|
|
222
|
-
0% { transform: rotate(0deg); }
|
|
223
|
-
100% { transform: rotate(360deg); }
|
|
224
|
-
}
|
|
225
|
-
`;
|
|
226
|
-
const styleElement = document.createElement('style');
|
|
227
|
-
if (add) {
|
|
228
|
-
if (!document.querySelector('.lze-blocker')) {
|
|
229
|
-
styleElement.innerHTML = style;
|
|
230
|
-
document.head.appendChild(styleElement);
|
|
231
|
-
body.insertAdjacentHTML('beforeend', blcokerHtml);
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
else {
|
|
235
|
-
this.removeBlocker(styleElement);
|
|
236
|
-
}
|
|
237
|
-
return styleElement;
|
|
238
|
-
}
|
|
239
|
-
removeBlocker(styleElement) {
|
|
240
|
-
const blocker = document.querySelector('.lze-blocker');
|
|
241
|
-
blocker === null || blocker === void 0 ? void 0 : blocker.remove();
|
|
242
|
-
styleElement === null || styleElement === void 0 ? void 0 : styleElement.remove();
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
function extractJsonFromResponse(body, onError) {
|
|
246
|
-
// If already an object, just return
|
|
247
|
-
if (typeof body === 'object' && body !== null)
|
|
248
|
-
return body;
|
|
249
|
-
// If it's a string, try to extract JSON portion
|
|
250
|
-
if (typeof body === 'string') {
|
|
251
|
-
const firstBrace = body.indexOf('{');
|
|
252
|
-
if (firstBrace !== -1) {
|
|
253
|
-
const jsonString = body.substring(firstBrace);
|
|
254
|
-
try {
|
|
255
|
-
return JSON.parse(jsonString);
|
|
256
|
-
}
|
|
257
|
-
catch (e) {
|
|
258
|
-
// If parsing fails, return original string
|
|
259
|
-
onError === null || onError === void 0 ? void 0 : onError(body);
|
|
260
|
-
return body;
|
|
261
|
-
}
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
return body;
|
|
265
|
-
}
|
|
266
|
-
export const ApiHelper = new LocalizeApiHelper();
|
|
267
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"loccalize.api.helper.js","sourceRoot":"","sources":["../../../../../../../../ANGULAR/LIBRARIES/localize.token.12.1.0/dev/src/localize-token/helpers/loccalize.api.helper.ts"],"names":[],"mappings":";AAEA,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,UAAU,EAAW,GAAG,EAAE,MAAM,MAAM,CAAC;AACvE,OAAO,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAC5C,OAAO,EAAE,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;AAgB5D,MAAM,iBAAiB;IAAvB;QAEa,wBAAmB,GAA2B;YACnD,eAAe,EAAE;gBACb,OAAO,EAAE,wCAAwC;gBACjD,WAAW,EAAE,IAAI;gBACjB,iBAAiB,EAAE,KAAK;aAC3B;SACJ,CAAC;IAoRN,CAAC;IAlRS,YAAY,CAAU,OAAsB;;YAC9C,IAAI,QAAQ,GAAG,CAAC,CAAC;YACjB,IAAI,SAAc,CAAC;YACnB,IAAI,YAAY,GAAG,CAAC,CAAC;YAErB,oDAAoD;YACpD,OAAO,mCAAQ,IAAI,CAAC,mBAAmB,GAAK,OAAO,CAAE,CAAA;YAErD,IAAI,YAA0C,CAAC;YAE/C,OAAO,QAAQ,GAAG,OAAO,CAAC,UAAU,EAAE,EAAE;gBACpC,IAAI;oBACA,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,QAAQ,EAAE,CAAC;oBACxC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,CAAC;oBACjC,OAAO,MAAM,CAAC;iBAEjB;gBAAC,OAAO,KAAU,EAAE;oBACjB,SAAS,GAAG,KAAK,CAAC;oBAElB,IAAI,YAAY,IAAI,CAAC,EAAE;wBACnB,OAAO,CAAC,KAAK,EAAE,CAAC;wBAChB,YAAY,GAAG,CAAC,CAAC;qBACpB;oBAED,IAAI,OAAO,CAAC,WAAW,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,KAAK,CAAC;wBAClD,MAAM,KAAK,CAAC,CAAC,iDAAiD;oBAElE,0BAA0B;oBAC1B,YAAY,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;oBAE5D,IAAI,OAAO,CAAC,OAAO;wBACf,MAAM,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;oBAE7D,IAAI,QAAQ,IAAI,OAAO,CAAC,UAAU,EAAE,GAAG,CAAC;wBACpC,MAAM,KAAK,CAAC;oBAEhB,QAAQ,EAAE,CAAC;oBACX,YAAY,EAAE,CAAC;oBACf,OAAO,CAAC,IAAI,CAAC,WAAW,QAAQ,sBAAsB,EAAE,KAAK,CAAC,CAAC;oBAC/D,MAAM,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;iBAChC;aACJ;YAED,OAAO,CAAC,IAAI,CAAC,gBAAgB,OAAO,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;YAC9D,MAAM,SAAS,CAAC;QACpB,CAAC;KAAA;IAEK,uBAAuB,CACzB,OAAoB,EACpB,MAA2B,EAC3B,cAA6D;;;YAG7D,MAAM,WAAW,GAAG,CAAA,MAAA,MAAM,CAAC,YAAY,0CAAE,aAAa;mBAC/C,IAAI,CAAC,iBAAiB,CAAC;YAE9B,OAAO,MAAM,IAAI,CAAC,YAAY,CAAC;gBAC3B,eAAe,EAAE,MAAA,MAAM,CAAC,YAAY,0CAAE,iBAAiB;gBACvD,UAAU,EAAE,GAAG,EAAE,eAAC,OAAA,MAAA,MAAA,MAAM,CAAC,YAAY,0CAAE,UAAU,mCAAI,IAAI,CAAA,EAAA;gBACzD,KAAK,EAAE,MAAA,MAAA,MAAM,CAAC,YAAY,0CAAE,KAAK,mCAAI,GAAG;gBACxC,QAAQ,EAAE,GAAG,EAAE,CAAC,cAAc,CAAI,OAAO,CAAC;gBAC1C,WAAW,EAAE,WAAW;aAC3B,CAAC,CAAA;;KACL;IAED,QAAQ,CAAC,OAAe,EAAE,IAAY;QAClC,MAAM,aAAa,GAAG,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,EAAE,CAAC;QAChG,OAAO,aAAa,CAAC,QAAQ,CAAC,GAAG,CAAC;YAC9B,CAAC,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YAC5B,CAAC,CAAC,aAAa,CAAC;IACxB,CAAC;IAEK,UAAU,CAAC,QAAmC;;YAChD,IAAI,CAAC,QAAQ;gBAAE,OAAO;YAEtB,MAAM,MAAM,GAAG,QAAQ,EAAE,CAAC;YAC1B,IAAI,MAAM,YAAY,OAAO,EAAE;gBAC3B,MAAM,MAAM,CAAC;aAChB;QACL,CAAC;KAAA;IAED,aAAa,CACT,QAAoC,EACpC,MAAc,EACd,GAAW,EACX,IAAS,EACT,OAAiC,EACjC,qBAA2C;QAE3C,MAAM,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,OAAO,CAAI,MAAM,EAAE,GAAG,kCAChD,OAAO,KACV,IAAI,IACN,CAAC,IAAI,CACH,SAAS,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC,EAC9B,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,uBAAuB,CAAC,IAAI,EAAE,qBAAqB,CAAC,CAAC,EACnE,UAAU,CAAC,CAAC,KAAwB,EAAE,EAAE;YACpC,6DAA6D;YAC7D,OAAO,UAAU,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,CAAC;QACnC,CAAC,CAAC,CACL,CAAC;QAEF,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED,oBAAoB,CAAC,KAAwB;QAEzC,0DAA0D;QAC1D,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC;YAC9B,MAAM,KAAK,CAAC;QAEhB,OAAO,IAAI,CAAC;IAChB,CAAC;IAED,iBAAiB,CAAC,KAAwB;QACtC,MAAM,cAAc,GAAG,KAAK,CAAC,MAAM,KAAK,CAAC,CAAC;QAC1C,MAAM,aAAa,GAAG,KAAK,CAAC,MAAM,IAAI,IAAI,IAAI,KAAK,CAAC,MAAM,GAAG,GAAG,CAAC;QACjE,OAAO,cAAc,IAAI,aAAa,CAAC;IAC3C,CAAC;IAEK,iBAAiB,CACnB,OAAsB,EACtB,KAAU;;;YAGV,IAAI,CAAC,OAAO,CAAC,eAAe;gBACxB,OAAO;YAEX,IAAI,YAA0C,CAAC;YAC/C,IAAI,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC,EAAE;gBAE/B,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;gBAExD,MAAM,IAAI,CAAC,UAAU,CAAC,MAAA,OAAO,CAAC,eAAe,CAAC,QAAQ,0CAAE,IAAI,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;gBAC3E,OAAO,YAAY,CAAC;aACvB;iBAAM;gBACH,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;gBAC1C,YAAY,aAAZ,YAAY,uBAAZ,YAAY,CAAE,MAAM,EAAE,CAAC;aAC1B;;KACJ;IAED,aAAa,CAAC,MAAqB,EAC/B,KAAyB,EACzB,MAAe,IAAI;;QAEnB,IAAI,CAAC,CAAA,MAAA,MAAM,CAAC,eAAe,0CAAE,WAAW,CAAA;YACpC,OAAO;QAEX,MAAM,OAAO,GAAG,CAAA,MAAA,MAAM,CAAC,eAAe,0CAAE,OAAO;eACxC,wCAAwC,CAAC;QAChD,MAAM,YAAY,GAAG,CAAA,MAAA,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,KAAK,0CAAE,OAAO,KAAI,mBAAmB,CAAC;QAClE,MAAM,gBAAgB,GAAG,CAAA,MAAA,MAAM,CAAC,eAAe,0CAAE,iBAAiB;eAC3D,6DAA6D,CAAC;QAErE,MAAM,MAAM,GAAG,CAAA,MAAA,MAAM,CAAC,eAAe,0CAAE,iBAAiB,KAAI,KAAK,CAAC;QAElE,MAAM,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC;QAC3B,MAAM,WAAW,GAAG;;cAEd,mBAAmB,CAAC,OAAO,CAAC,YAAY;;kBAEpC,OAAO;;;;;;;8CAOqB,YAAY;yDACD,gBAAgB;;SAEhE,CAAC;QACF,MAAM,KAAK,GAAG;;;;;;;;uBAQC,MAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;SAyEpB,CAAC;QAEF,MAAM,YAAY,GAAG,QAAQ,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QAErD,IAAI,GAAG,EAAE;YACL,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,cAAc,CAAC,EAAE;gBACzC,YAAY,CAAC,SAAS,GAAG,KAAK,CAAC;gBAC/B,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC;gBACxC,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC;aACrD;SACJ;aAAM;YACH,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,CAAC;SACpC;QAED,OAAO,YAAY,CAAC;IACxB,CAAC;IAEO,aAAa,CAAC,YAA0C;QAC5D,MAAM,OAAO,GAAG,QAAQ,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC;QACvD,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,MAAM,EAAE,CAAC;QAClB,YAAY,aAAZ,YAAY,uBAAZ,YAAY,CAAE,MAAM,EAAE,CAAC;IAC3B,CAAC;CACJ;AAED,SAAS,uBAAuB,CAAC,IAAS,EAAE,OAA6B;IACrE,oCAAoC;IACpC,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,IAAI;QAAE,OAAO,IAAI,CAAC;IAE3D,gDAAgD;IAChD,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE;QAC1B,MAAM,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACrC,IAAI,UAAU,KAAK,CAAC,CAAC,EAAE;YACnB,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;YAC9C,IAAI;gBACA,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;aACjC;YAAC,OAAO,CAAC,EAAE;gBACR,2CAA2C;gBAC3C,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAG,IAAI,CAAC,CAAC;gBAChB,OAAO,IAAI,CAAC;aACf;SACJ;KACJ;IAED,OAAO,IAAI,CAAC;AAChB,CAAC;AAED,MAAM,CAAC,MAAM,SAAS,GAAG,IAAI,iBAAiB,EAAE,CAAC","sourcesContent":["import { HttpClient, HttpErrorResponse, HttpHeaders } from \"@angular/common/http\";\r\nimport { IApiOptions, ILocalizeApiConfigs, IOnConnectionError } from \"./interfaces\";\r\nimport { takeUntil, catchError, throwError, Subject, map } from \"rxjs\";\r\nimport { waitFor } from \"../localize.token\";\r\nimport { LOCALIZE_API_ASSETS } from \"./localize.api.assets\";\r\n\r\ninterface IRequestInstance {\r\n    client: HttpClient;\r\n    destroy$: () => Subject<void>;\r\n}\r\n\r\ninterface IRetryOptions {\r\n    maxRetries: () => number;\r\n    delay: number;\r\n    callback: () => Promise<any> | any;\r\n    retryUnless?: (error: any) => boolean;\r\n    onError?: (error: any) => Promise<void> | any;\r\n    connectionError?: IOnConnectionError\r\n}\r\n\r\nclass LocalizeApiHelper {\r\n\r\n    readonly defaultRetryOptions: Partial<IRetryOptions> = {\r\n        connectionError: {\r\n            message: 'Connection error occurred. Please wait',\r\n            blockScreen: true,\r\n            blockScreenZIndex: 10000\r\n        }\r\n    };\r\n\r\n    async performRetry<T = any>(options: IRetryOptions): Promise<T> {\r\n        let attempts = 0;\r\n        let lastError: any;\r\n        let consoleCount = 0;\r\n\r\n        // Merge default retry options with provided options\r\n        options = { ...this.defaultRetryOptions, ...options }\r\n\r\n        let styleElement: HTMLStyleElement | undefined;\r\n\r\n        while (attempts < options.maxRetries()) {\r\n            try {\r\n                const result = await options.callback();\r\n                this.removeBlocker(styleElement);\r\n                return result;\r\n\r\n            } catch (error: any) {\r\n                lastError = error;\r\n\r\n                if (consoleCount >= 7) {\r\n                    console.clear();\r\n                    consoleCount = 0;\r\n                }\r\n\r\n                if (options.retryUnless && !options.retryUnless(error))\r\n                    throw error; // If the error should not be retried, rethrow it\r\n\r\n                // Handle connection error\r\n                styleElement = await this.onConnectionError(options, error);\r\n\r\n                if (options.onError)\r\n                    await this.invokeHook(options.onError.bind(this, error));\r\n\r\n                if (attempts >= options.maxRetries() - 1)\r\n                    throw error;\r\n\r\n                attempts++;\r\n                consoleCount++;\r\n                console.warn(`Attempt ${attempts} failed. Retrying...`, error);\r\n                await waitFor(options.delay);\r\n            }\r\n        }\r\n\r\n        console.warn(`Failed after ${options.maxRetries()} attempts`);\r\n        throw lastError;\r\n    }\r\n\r\n    async performRequestWithRetry<T = any>(\r\n        options: IApiOptions,\r\n        config: ILocalizeApiConfigs,\r\n        performRequest: <T = any>(options: IApiOptions) => Promise<T>\r\n    ): Promise<T> {\r\n\r\n        const retryUnless = config.retryOptions?.retryFunction\r\n            || this.isConnectionError;\r\n\r\n        return await this.performRetry({\r\n            connectionError: config.retryOptions?.onConnectionError,\r\n            maxRetries: () => config.retryOptions?.maxRetries ?? 1000,\r\n            delay: config.retryOptions?.delay ?? 500,\r\n            callback: () => performRequest<T>(options),\r\n            retryUnless: retryUnless,\r\n        })\r\n    }\r\n\r\n    buildUrl(baseUrl: string, path: string): string {\r\n        const normalizedUrl = `${baseUrl.trim().replace(/\\/?$/, '/')}${path.trim().replace(/^\\//, '')}`;\r\n        return normalizedUrl.endsWith('/')\r\n            ? normalizedUrl.slice(0, -1)\r\n            : normalizedUrl;\r\n    }\r\n\r\n    async invokeHook(callback?: () => Promise<any> | any): Promise<any> {\r\n        if (!callback) return;\r\n\r\n        const result = callback();\r\n        if (result instanceof Promise) {\r\n            await result;\r\n        }\r\n    }\r\n\r\n    createRequest<T = any>(\r\n        instance: Required<IRequestInstance>,\r\n        method: string,\r\n        url: string,\r\n        body: any,\r\n        options: { headers: HttpHeaders },\r\n        onInvalidResponseBody?: (body: any) => void\r\n    ) {\r\n        const request$ = instance.client.request<T>(method, url, {\r\n            ...options,\r\n            body,\r\n        }).pipe(\r\n            takeUntil(instance.destroy$()),\r\n            map((body) => extractJsonFromResponse(body, onInvalidResponseBody)),\r\n            catchError((error: HttpErrorResponse) => {\r\n                // Convert to a non-observable error to handle in the promise\r\n                return throwError(() => error);\r\n            })\r\n        );\r\n\r\n        return request$;\r\n    }\r\n\r\n    defaultRetryFunction(error: HttpErrorResponse): boolean {\r\n\r\n        // Don't retry for other errors (like 400, 401, 403, etc.)\r\n        if (!this.isConnectionError(error))\r\n            throw error;\r\n\r\n        return true;\r\n    }\r\n\r\n    isConnectionError(error: HttpErrorResponse): boolean {\r\n        const isNetworkError = error.status === 0;\r\n        const isServerError = error.status >= 1000 && error.status < 600;\r\n        return isNetworkError || isServerError;\r\n    }\r\n\r\n    async onConnectionError(\r\n        options: IRetryOptions,\r\n        error: any\r\n    ): Promise<Promise<void> | any> {\r\n\r\n        if (!options.connectionError)\r\n            return;\r\n\r\n        let styleElement: HTMLStyleElement | undefined;\r\n        if (this.isConnectionError(error)) {\r\n\r\n            styleElement = this.screenBlocker(options, error, true);\r\n\r\n            await this.invokeHook(options.connectionError.callback?.bind(this, error));\r\n            return styleElement;\r\n        } else {\r\n            this.screenBlocker(options, error, false);\r\n            styleElement?.remove();\r\n        }\r\n    }\r\n\r\n    screenBlocker(optons: IRetryOptions,\r\n        error?: HttpErrorResponse,\r\n        add: boolean = true): HTMLStyleElement | undefined {\r\n\r\n        if (!optons.connectionError?.blockScreen)\r\n            return;\r\n\r\n        const message = optons.connectionError?.message\r\n            || 'Connection error occurred. Please wait';\r\n        const errorMessage = error?.error?.message || 'An error occurred';\r\n        const suggestinMessage = optons.connectionError?.suggestionMessage\r\n            || 'Please check your internet connection or the server status.';\r\n\r\n        const zIndex = optons.connectionError?.blockScreenZIndex || 10000;\r\n\r\n        const body = document.body;\r\n        const blcokerHtml = `\r\n        <div class=\"lze-blocker\">\r\n            ${LOCALIZE_API_ASSETS.network.noConnection}\r\n            <div class=\"lze-blocker__message\">\r\n                ${message}\r\n                <span class=\"lze-blocker__dotting\">\r\n                    <span class=\"lze-blocker__dot\"></span>\r\n                    <span class=\"lze-blocker__dot\"></span>\r\n                    <span class=\"lze-blocker__dot\"></span>\r\n                </span>\r\n            </div>\r\n            <div class=\"lze-blocker__error\">${errorMessage}</div>\r\n            <div class=\"lze-blocker__error_suggestion\">${suggestinMessage}</div>\r\n        </div>\r\n        `;\r\n        const style = `\r\n        div.lze-blocker {\r\n            position: fixed;\r\n            top: 0;\r\n            left: 0;\r\n            width: 100%;\r\n            height: 100%;\r\n            background: rgba(0, 0, 0, 0.85) !important;\r\n            z-index: ${zIndex};\r\n            display: flex;\r\n            align-items: center;\r\n            justify-content: center;\r\n            flex-direction: column;\r\n            color: #fff !important;\r\n            font-family: Arial, sans-serif;\r\n            text-align: center;\r\n            padding: 20px;\r\n            box-sizing: border-box;\r\n            overflow: hidden;\r\n            user-select: none;\r\n        }\r\n\r\n        svg#lze-no-connection {\r\n            width: 75px;\r\n            height: 75px;\r\n            margin-bottom: 20px;\r\n        }\r\n        \r\n        div.lze-blocker__message {\r\n            color: #fff !important;\r\n            font-size: 18px !important;\r\n            margin-bottom: 10px;\r\n        }\r\n            \r\n        .lze-blocker__dotting {\r\n            display: inline-block;\r\n            vertical-align: middle;\r\n        }\r\n        span.lze-blocker__dot {\r\n            display: inline-block;\r\n            width: 7px;\r\n            height: 7px;\r\n            background-color: #ffffff !important;\r\n            border-radius: 50%;\r\n            margin-left: 3px;\r\n            opacity: 0.3;\r\n            animation: dotting 1s infinite;\r\n        }\r\n        .lze-blocker__dot:nth-child(1) {\r\n            animation-delay: 0s;\r\n            opacity: 1;\r\n        }\r\n        .lze-blocker__dot:nth-child(2) {\r\n            animation-delay: 0.2s;\r\n        }\r\n        .lze-blocker__dot:nth-child(3) {\r\n            animation-delay: 0.4s;\r\n        }\r\n\r\n        @keyframes dotting {\r\n            0%, 80%, 100% { opacity: 0.3; }\r\n            40% { opacity: 1; }\r\n        }\r\n\r\n        div.lze-blocker__error {\r\n            color: #f00;\r\n            font-size: 14px !important;\r\n            margin-bottom: 10px;\r\n            text-shadow: 0 0 1px #ff5f5f !important;\r\n        }\r\n\r\n        div.lze-blocker__error_suggestion {\r\n            color: #ccc !important;\r\n            font-size: 14px !important;\r\n            margin-top: 10px;\r\n        }\r\n        \r\n        @keyframes spin {\r\n            0% { transform: rotate(0deg); }\r\n            100% { transform: rotate(360deg); }\r\n        }\r\n        `;\r\n\r\n        const styleElement = document.createElement('style');\r\n\r\n        if (add) {\r\n            if (!document.querySelector('.lze-blocker')) {\r\n                styleElement.innerHTML = style;\r\n                document.head.appendChild(styleElement);\r\n                body.insertAdjacentHTML('beforeend', blcokerHtml);\r\n            }\r\n        } else {\r\n            this.removeBlocker(styleElement);\r\n        }\r\n\r\n        return styleElement;\r\n    }\r\n\r\n    private removeBlocker(styleElement: HTMLStyleElement | undefined): void {\r\n        const blocker = document.querySelector('.lze-blocker');\r\n        blocker?.remove();\r\n        styleElement?.remove();\r\n    }\r\n}\r\n\r\nfunction extractJsonFromResponse(body: any, onError?: (body: any) => void): any {\r\n    // If already an object, just return\r\n    if (typeof body === 'object' && body !== null) return body;\r\n\r\n    // If it's a string, try to extract JSON portion\r\n    if (typeof body === 'string') {\r\n        const firstBrace = body.indexOf('{');\r\n        if (firstBrace !== -1) {\r\n            const jsonString = body.substring(firstBrace);\r\n            try {\r\n                return JSON.parse(jsonString);\r\n            } catch (e) {\r\n                // If parsing fails, return original string\r\n                onError?.(body);\r\n                return body;\r\n            }\r\n        }\r\n    }\r\n\r\n    return body;\r\n}\r\n\r\nexport const ApiHelper = new LocalizeApiHelper();"]}
|
|
@@ -1,242 +0,0 @@
|
|
|
1
|
-
import { __awaiter } from "tslib";
|
|
2
|
-
import { HttpClient, HttpHeaders } from "@angular/common/http";
|
|
3
|
-
import { Injectable } from "@angular/core";
|
|
4
|
-
import { BehaviorSubject, Subject } from "rxjs";
|
|
5
|
-
import { EMethod } from "./helpers/interfaces";
|
|
6
|
-
import { ApiHelper } from "./helpers/loccalize.api.helper";
|
|
7
|
-
import { LocalizeToken, waitFor, waitUntil } from "./localize.token";
|
|
8
|
-
import { LocalizeTokenService } from "./localize.token.service";
|
|
9
|
-
import * as i0 from "@angular/core";
|
|
10
|
-
import * as i1 from "@angular/common/http";
|
|
11
|
-
import * as i2 from "./localize.token.service";
|
|
12
|
-
const SCHEMES = LocalizeToken.httpHeaders;
|
|
13
|
-
export class LocalizeApiService {
|
|
14
|
-
constructor(httpClient, localizeTokenService) {
|
|
15
|
-
this.httpClient = httpClient;
|
|
16
|
-
this.localizeTokenService = localizeTokenService;
|
|
17
|
-
this.destroy$ = new Subject();
|
|
18
|
-
this.configSubject = new BehaviorSubject({});
|
|
19
|
-
this.isRequestingSubject = new BehaviorSubject(false);
|
|
20
|
-
this.isResolvingStartupSubject = new BehaviorSubject(false);
|
|
21
|
-
this.defaultConfig = {
|
|
22
|
-
waitEachRequest: { milliseconds: 0 },
|
|
23
|
-
enableRequestCancellation: true,
|
|
24
|
-
retryOptions: {
|
|
25
|
-
maxRetries: 1000,
|
|
26
|
-
delay: 1000,
|
|
27
|
-
retryFunction: ApiHelper.defaultRetryFunction.bind(this),
|
|
28
|
-
},
|
|
29
|
-
};
|
|
30
|
-
this.apiOptions = {
|
|
31
|
-
method: EMethod.GET,
|
|
32
|
-
requestBody: null,
|
|
33
|
-
};
|
|
34
|
-
/**
|
|
35
|
-
* A higher-order function that returns a curried function for making API requests.
|
|
36
|
-
*
|
|
37
|
-
* @param baseUrl - The base URL of the API.
|
|
38
|
-
* @returns A curried function that can be used to make API requests.
|
|
39
|
-
*/
|
|
40
|
-
this.func = (baseUrl) => (path, method = EMethod.GET, reqBody = null, reqHeaders) => this.request(baseUrl, path, method, reqBody, reqHeaders);
|
|
41
|
-
}
|
|
42
|
-
get isResolvingStartup() { return this.isResolvingStartupSubject.value; }
|
|
43
|
-
get needTenant() { return this.localizeTokenService.config.tenantToken !== undefined; }
|
|
44
|
-
get isRequesting() { return this.isRequestingSubject.value; }
|
|
45
|
-
get isRevokingToken() { return this.localizeTokenService.isRevokingToken; }
|
|
46
|
-
set isRevokingToken(value) { this.localizeTokenService.isRevokingToken = value; }
|
|
47
|
-
get accessToken() { return this.localizeTokenService.accessToken; }
|
|
48
|
-
set accessToken(value) { this.localizeTokenService.accessToken = value; }
|
|
49
|
-
get refreshToken() { return this.localizeTokenService.refreshToken; }
|
|
50
|
-
get tenantToken() { return this.localizeTokenService.tenantToken; }
|
|
51
|
-
get thirdPartyConfig() { var _a; return (_a = this.localizeTokenService.config) === null || _a === void 0 ? void 0 : _a.thirdPartyConfig; }
|
|
52
|
-
get config() {
|
|
53
|
-
this.validateConfig();
|
|
54
|
-
return this.configSubject.value;
|
|
55
|
-
}
|
|
56
|
-
/**
|
|
57
|
-
* Initialize the API service.
|
|
58
|
-
* @param apiConfigs - The API configurations.
|
|
59
|
-
*/
|
|
60
|
-
init(apiConfigs) {
|
|
61
|
-
console.log('LocalizeApiService is initialized.');
|
|
62
|
-
this.configSubject.next(Object.assign(Object.assign({}, this.defaultConfig), apiConfigs));
|
|
63
|
-
}
|
|
64
|
-
cancelPendingRequests() {
|
|
65
|
-
this.config.enableRequestCancellation
|
|
66
|
-
&& this.destroy$.next();
|
|
67
|
-
}
|
|
68
|
-
ngOnDestroy() {
|
|
69
|
-
this.destroy$.next();
|
|
70
|
-
this.destroy$.complete();
|
|
71
|
-
}
|
|
72
|
-
request(baseUrl, path, method = EMethod.GET, reqBody = null, reqHeaders) {
|
|
73
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
74
|
-
yield waitUntil(() => !this.isResolvingStartup, 500);
|
|
75
|
-
yield ApiHelper.invokeHook(this.config.onPrepareRequest);
|
|
76
|
-
const apiOptions = this.buildApiOptions(baseUrl, path, method, reqBody, reqHeaders);
|
|
77
|
-
try {
|
|
78
|
-
yield this.toWaitForPreviousRequest();
|
|
79
|
-
return yield ApiHelper.performRequestWithRetry(apiOptions, this.config, this.performRequest.bind(this));
|
|
80
|
-
}
|
|
81
|
-
catch (error) {
|
|
82
|
-
return yield this.handleOnRequestError(error, apiOptions);
|
|
83
|
-
}
|
|
84
|
-
});
|
|
85
|
-
}
|
|
86
|
-
handleOnRequestError(error, options) {
|
|
87
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
88
|
-
if (error.status !== 401)
|
|
89
|
-
throw error;
|
|
90
|
-
yield waitUntil(() => !this.isResolvingStartup, 50);
|
|
91
|
-
return yield ApiHelper.performRetry({
|
|
92
|
-
maxRetries: () => 1000,
|
|
93
|
-
delay: 10,
|
|
94
|
-
retryUnless: (error) => error.status === 401 || ApiHelper.isConnectionError(error),
|
|
95
|
-
callback: () => __awaiter(this, void 0, void 0, function* () {
|
|
96
|
-
// Handle third party API unauthorized error when request
|
|
97
|
-
this.handleThirdPartyAuthError(error, options.requestUrl);
|
|
98
|
-
// Only handle 401 Unauthorized errors
|
|
99
|
-
yield this.revokeToken();
|
|
100
|
-
// Retry the request with the new access token
|
|
101
|
-
return yield this.performRequest(options);
|
|
102
|
-
})
|
|
103
|
-
});
|
|
104
|
-
});
|
|
105
|
-
}
|
|
106
|
-
handleThirdPartyAuthError(error, requestUrl) {
|
|
107
|
-
var _a, _b, _c;
|
|
108
|
-
if (!((_a = this.thirdPartyConfig) === null || _a === void 0 ? void 0 : _a.isThirdPartyMode))
|
|
109
|
-
return;
|
|
110
|
-
if (error.status === 401) {
|
|
111
|
-
if (this.thirdPartyConfig.lastUnauthorizedUrl === requestUrl) {
|
|
112
|
-
this.thirdPartyConfig.lastUnauthorizedUrl = "";
|
|
113
|
-
throw (_c = (_b = this.config).onException) === null || _c === void 0 ? void 0 : _c.call(_b, Object.assign(Object.assign({}, error), { status: 405, message: `Unauthorized request url ${requestUrl}. Please contact administrator.` }));
|
|
114
|
-
}
|
|
115
|
-
this.thirdPartyConfig.lastUnauthorizedUrl = requestUrl;
|
|
116
|
-
}
|
|
117
|
-
else {
|
|
118
|
-
this.thirdPartyConfig.lastUnauthorizedUrl = "";
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
performRequest(options) {
|
|
122
|
-
var _a;
|
|
123
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
124
|
-
// Build the request options
|
|
125
|
-
const buildOptions = { headers: this.buildHeaderOptions(options) };
|
|
126
|
-
// Create the request observable
|
|
127
|
-
const request$ = ApiHelper.createRequest({
|
|
128
|
-
client: this.httpClient,
|
|
129
|
-
destroy$: () => this.destroy$
|
|
130
|
-
}, options.method, options.requestUrl, options.requestBody, buildOptions, (_a = this.config.onResponseBodyInvalid) === null || _a === void 0 ? void 0 : _a.bind(this));
|
|
131
|
-
// Set the isRequesting state to true before making the request
|
|
132
|
-
this.isRequestingSubject.next(true);
|
|
133
|
-
const response = yield new Promise((resolve, reject) => request$.subscribe({ next: resolve, error: reject }));
|
|
134
|
-
// Reset the isRequesting state after the request completes
|
|
135
|
-
this.isRequestingSubject.next(false);
|
|
136
|
-
return response;
|
|
137
|
-
});
|
|
138
|
-
}
|
|
139
|
-
revokeToken() {
|
|
140
|
-
var _a, _b;
|
|
141
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
142
|
-
try {
|
|
143
|
-
if (yield this.interceptRevokeToken())
|
|
144
|
-
return;
|
|
145
|
-
this.isRevokingToken = true;
|
|
146
|
-
const defaultApiOptions = Object.assign(Object.assign({}, this.buildApiOptions(((_a = this.localizeTokenService.config.refreshToken) === null || _a === void 0 ? void 0 : _a.requestUrl) || '')), { refreshToken: true });
|
|
147
|
-
// If in third party mode need override the default API options with third party API options
|
|
148
|
-
const thirdParyApiOptions = this.buildThirdPartyApiOption(defaultApiOptions);
|
|
149
|
-
const apiOptions = ((_b = this.thirdPartyConfig) === null || _b === void 0 ? void 0 : _b.isThirdPartyMode) ? thirdParyApiOptions : defaultApiOptions;
|
|
150
|
-
// const revokeToken = await this.performRequest(apiOptions);
|
|
151
|
-
const revokeToken = yield ApiHelper.performRequestWithRetry(apiOptions, this.config, this.performRequest.bind(this));
|
|
152
|
-
yield this.handleOnTokenRevoked(revokeToken);
|
|
153
|
-
}
|
|
154
|
-
catch (error) {
|
|
155
|
-
// Handle the error, log it
|
|
156
|
-
yield ApiHelper.invokeHook(this.config.onAutoLogout);
|
|
157
|
-
throw error;
|
|
158
|
-
}
|
|
159
|
-
finally {
|
|
160
|
-
// Reset the revoking token state
|
|
161
|
-
this.isRevokingToken = false;
|
|
162
|
-
}
|
|
163
|
-
});
|
|
164
|
-
}
|
|
165
|
-
buildThirdPartyApiOption(apiOptions) {
|
|
166
|
-
var _a, _b, _c;
|
|
167
|
-
return Object.assign(Object.assign({}, apiOptions), { requestUrl: ((_a = this.thirdPartyConfig) === null || _a === void 0 ? void 0 : _a.refreshToken.requestUrl) || '', method: ((_b = this.thirdPartyConfig) === null || _b === void 0 ? void 0 : _b.refreshToken.method) || EMethod.GET, requestBody: ((_c = this.thirdPartyConfig) === null || _c === void 0 ? void 0 : _c.refreshToken.body) || {}, headers: Object.assign(Object.assign({}, apiOptions.headers), { [SCHEMES.X_SECRET]: `${this.localizeTokenService.clientSecret}` }) });
|
|
168
|
-
}
|
|
169
|
-
/** default http request options */
|
|
170
|
-
buildHeaderOptions(options) {
|
|
171
|
-
var _a;
|
|
172
|
-
const headers = Object.assign(Object.assign(Object.assign(Object.assign(Object.assign({}, (options.refreshToken && { [SCHEMES.X_REFRESH_TOKEN]: `${this.refreshToken}` })), (!options.isFormData && { [SCHEMES.CONTENT_TYPE]: 'application/json' })), { [SCHEMES.AUTHORIZATION]: `Bearer ${this.accessToken}` }), (this.needTenant && { [SCHEMES.X_TENANT]: `${this.tenantToken}` })), (((_a = this.thirdPartyConfig) === null || _a === void 0 ? void 0 : _a.isThirdPartyMode) && { [SCHEMES.X_CLIENT]: `${this.localizeTokenService.clientId}` }));
|
|
173
|
-
return new HttpHeaders(Object.assign(Object.assign({}, headers), options.headers));
|
|
174
|
-
}
|
|
175
|
-
buildApiOptions(baseUrl, path = '', method = EMethod.GET, requestBody = null, headers) {
|
|
176
|
-
const requestUrl = ApiHelper.buildUrl(baseUrl, path);
|
|
177
|
-
const isFormData = requestBody && requestBody instanceof FormData;
|
|
178
|
-
return Object.assign(Object.assign({}, this.apiOptions), { headers, method, requestUrl, requestBody, isFormData });
|
|
179
|
-
}
|
|
180
|
-
toWaitForPreviousRequest() {
|
|
181
|
-
var _a;
|
|
182
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
183
|
-
if (this.isRevokingToken) {
|
|
184
|
-
yield waitUntil(() => !this.isRevokingToken);
|
|
185
|
-
}
|
|
186
|
-
// to wait for each request in 50ms, even if the request is not completed
|
|
187
|
-
const waitMilliseconds = (_a = this.config.waitEachRequest) === null || _a === void 0 ? void 0 : _a.milliseconds;
|
|
188
|
-
if (waitMilliseconds && this.isRequesting) {
|
|
189
|
-
// console.warn(`Request throttling: Another request is in progress. Waiting for ${waitMilliseconds}ms.`);
|
|
190
|
-
yield waitFor(waitMilliseconds, this.isRequesting);
|
|
191
|
-
}
|
|
192
|
-
});
|
|
193
|
-
}
|
|
194
|
-
handleOnTokenRevoked(response) {
|
|
195
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
196
|
-
if ((response === null || response === void 0 ? void 0 : response.status) || (response === null || response === void 0 ? void 0 : response.token)) {
|
|
197
|
-
// If the response is successful, update the access token
|
|
198
|
-
this.accessToken = response.message || response.token; // response.token for third party revoke
|
|
199
|
-
}
|
|
200
|
-
else {
|
|
201
|
-
// If the response indicates an error, invoke the onRevokeUnauthorized hook
|
|
202
|
-
console.warn('Token revocation failed, refresh token is expired.', response.message);
|
|
203
|
-
yield ApiHelper.invokeHook(this.config.onRevokeUnauthorized);
|
|
204
|
-
}
|
|
205
|
-
});
|
|
206
|
-
}
|
|
207
|
-
interceptRevokeToken() {
|
|
208
|
-
return __awaiter(this, void 0, void 0, function* () {
|
|
209
|
-
if (this.isRevokingToken) {
|
|
210
|
-
console.warn('Token is already being revoked. Waiting for the current operation to complete...');
|
|
211
|
-
yield waitUntil(() => !this.isRevokingToken);
|
|
212
|
-
return true;
|
|
213
|
-
}
|
|
214
|
-
if (!this.refreshToken) {
|
|
215
|
-
// await ApiHelper.invokeHook(this.apiConfigs.onAutoLogout);
|
|
216
|
-
throw new Error('Refresh token is missing. Please login again.');
|
|
217
|
-
}
|
|
218
|
-
return false;
|
|
219
|
-
});
|
|
220
|
-
}
|
|
221
|
-
validateConfig() {
|
|
222
|
-
var _a, _b, _c, _d;
|
|
223
|
-
if (this.localizeTokenService.config.requiredTenant
|
|
224
|
-
&& !((_b = (_a = this.localizeTokenService.config.tenantToken) === null || _a === void 0 ? void 0 : _a.name) === null || _b === void 0 ? void 0 : _b.trim().length)) {
|
|
225
|
-
throw Error('Tenant token is required but tenantTokenName is not configured');
|
|
226
|
-
}
|
|
227
|
-
if (!((_d = (_c = this.localizeTokenService.config.refreshToken) === null || _c === void 0 ? void 0 : _c.requestUrl) === null || _d === void 0 ? void 0 : _d.trim().length)) {
|
|
228
|
-
throw Error('Revoke token URL is not configured - token refresh will not work');
|
|
229
|
-
}
|
|
230
|
-
}
|
|
231
|
-
}
|
|
232
|
-
LocalizeApiService.ɵprov = i0.ɵɵdefineInjectable({ factory: function LocalizeApiService_Factory() { return new LocalizeApiService(i0.ɵɵinject(i1.HttpClient), i0.ɵɵinject(i2.LocalizeTokenService)); }, token: LocalizeApiService, providedIn: "root" });
|
|
233
|
-
LocalizeApiService.decorators = [
|
|
234
|
-
{ type: Injectable, args: [{
|
|
235
|
-
providedIn: 'root'
|
|
236
|
-
},] }
|
|
237
|
-
];
|
|
238
|
-
LocalizeApiService.ctorParameters = () => [
|
|
239
|
-
{ type: HttpClient },
|
|
240
|
-
{ type: LocalizeTokenService }
|
|
241
|
-
];
|
|
242
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"localize.api.service.js","sourceRoot":"","sources":["../../../../../../../ANGULAR/LIBRARIES/localize.token.12.1.0/dev/src/localize-token/localize.api.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { HttpClient, HttpErrorResponse, HttpHeaders } from \"@angular/common/http\";\r\nimport { Injectable, OnDestroy } from \"@angular/core\";\r\nimport { BehaviorSubject, Subject } from \"rxjs\";\r\nimport { EMethod, IApiOptions, IHeader, ILocalizeApiConfigs } from \"./helpers/interfaces\";\r\nimport { ApiHelper } from \"./helpers/loccalize.api.helper\";\r\nimport { IThirdPartyConfig, LocalizeToken, waitFor, waitUntil } from \"./localize.token\";\r\nimport { LocalizeTokenService } from \"./localize.token.service\";\r\n\r\nconst SCHEMES = LocalizeToken.httpHeaders;\r\n\r\n@Injectable({\r\n  providedIn: 'root'\r\n})\r\nexport class LocalizeApiService implements OnDestroy {\r\n  private readonly destroy$ = new Subject<void>();\r\n  private readonly configSubject = new BehaviorSubject<Partial<ILocalizeApiConfigs>>({});\r\n  readonly isRequestingSubject = new BehaviorSubject<boolean>(false);\r\n  readonly isResolvingStartupSubject = new BehaviorSubject<boolean>(false);\r\n  get isResolvingStartup() { return this.isResolvingStartupSubject.value; }\r\n\r\n  get needTenant(): boolean { return this.localizeTokenService.config.tenantToken !== undefined; }\r\n  get isRequesting() { return this.isRequestingSubject.value; }\r\n  get isRevokingToken() { return this.localizeTokenService.isRevokingToken; }\r\n  set isRevokingToken(value: boolean) { this.localizeTokenService.isRevokingToken = value; }\r\n\r\n  get accessToken(): string | undefined { return this.localizeTokenService.accessToken; }\r\n  set accessToken(value: string | undefined) { this.localizeTokenService.accessToken = value; }\r\n\r\n  get refreshToken(): string | undefined { return this.localizeTokenService.refreshToken; }\r\n  get tenantToken(): string | undefined { return this.localizeTokenService.tenantToken; }\r\n  get thirdPartyConfig(): IThirdPartyConfig | undefined { return this.localizeTokenService.config?.thirdPartyConfig }\r\n  private defaultConfig: Partial<ILocalizeApiConfigs> = {\r\n    waitEachRequest: { milliseconds: 0 },\r\n    enableRequestCancellation: true,\r\n\r\n    retryOptions: {\r\n      maxRetries: 1000,\r\n      delay: 1000,\r\n      retryFunction: ApiHelper.defaultRetryFunction.bind(this),\r\n    },\r\n\r\n  };\r\n\r\n  private get config(): Partial<ILocalizeApiConfigs> {\r\n    this.validateConfig();\r\n    return this.configSubject.value;\r\n  }\r\n\r\n  private apiOptions: Partial<IApiOptions> = {\r\n    method: EMethod.GET,\r\n    requestBody: null,\r\n  };\r\n\r\n  constructor(readonly httpClient: HttpClient,\r\n    private readonly localizeTokenService: LocalizeTokenService\r\n  ) { }\r\n\r\n  /**\r\n   * Initialize the API service.\r\n   * @param apiConfigs - The API configurations.\r\n   */\r\n  init(apiConfigs: ILocalizeApiConfigs) {\r\n    console.log('LocalizeApiService is initialized.');\r\n    this.configSubject.next({ ...this.defaultConfig, ...apiConfigs });\r\n  }\r\n\r\n  cancelPendingRequests(): void {\r\n    this.config.enableRequestCancellation\r\n      && this.destroy$.next();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.destroy$.next();\r\n    this.destroy$.complete();\r\n  }\r\n\r\n  /**\r\n   * A higher-order function that returns a curried function for making API requests.\r\n   *\r\n   * @param baseUrl - The base URL of the API.\r\n   * @returns A curried function that can be used to make API requests.\r\n   */\r\n  func = (baseUrl: string) =>\r\n    <T = any>(path: string, method: EMethod = EMethod.GET, reqBody: any = null, reqHeaders?: IHeader) =>\r\n      this.request<T>(baseUrl, path, method, reqBody, reqHeaders)\r\n\r\n  public async request<T = any>(baseUrl: string,\r\n    path: string,\r\n    method: EMethod = EMethod.GET,\r\n    reqBody: any = null,\r\n    reqHeaders?: IHeader): Promise<T> {\r\n\r\n    await waitUntil(() => !this.isResolvingStartup, 500);\r\n\r\n    await ApiHelper.invokeHook(this.config.onPrepareRequest);\r\n\r\n    const apiOptions = this.buildApiOptions(baseUrl, path, method, reqBody, reqHeaders);\r\n\r\n    try {\r\n      await this.toWaitForPreviousRequest();\r\n\r\n      return await ApiHelper.performRequestWithRetry<T>(\r\n        apiOptions,\r\n        this.config,\r\n        this.performRequest.bind(this)\r\n      );\r\n\r\n    }\r\n    catch (error: any) {\r\n      return await this.handleOnRequestError(error, apiOptions);\r\n    }\r\n  }\r\n\r\n  private async handleOnRequestError(error: HttpErrorResponse, options: IApiOptions): Promise<any> {\r\n    if (error.status !== 401)\r\n      throw error;\r\n\r\n    await waitUntil(() => !this.isResolvingStartup, 50);\r\n\r\n    return await ApiHelper.performRetry({\r\n      maxRetries: () => 1000,\r\n      delay: 10,\r\n      retryUnless: (error) => error.status === 401 || ApiHelper.isConnectionError(error),\r\n      callback: async () => {\r\n        // Handle third party API unauthorized error when request \r\n        this.handleThirdPartyAuthError(error, options.requestUrl);\r\n\r\n        // Only handle 401 Unauthorized errors\r\n        await this.revokeToken();\r\n\r\n        // Retry the request with the new access token\r\n        return await this.performRequest(options);\r\n      }\r\n    })\r\n  }\r\n\r\n  private handleThirdPartyAuthError(error: HttpErrorResponse, requestUrl: string) {\r\n    if (!this.thirdPartyConfig?.isThirdPartyMode) return;\r\n    if (error.status === 401) {\r\n      if (this.thirdPartyConfig.lastUnauthorizedUrl === requestUrl) {\r\n        this.thirdPartyConfig.lastUnauthorizedUrl = \"\";\r\n        throw this.config.onException?.({ ...error, status: 405, message: `Unauthorized request url ${requestUrl}. Please contact administrator.` });\r\n      }\r\n      this.thirdPartyConfig.lastUnauthorizedUrl = requestUrl;\r\n    } else {\r\n      this.thirdPartyConfig.lastUnauthorizedUrl = \"\";\r\n    }\r\n  }\r\n\r\n  private async performRequest<T = any>(options: IApiOptions): Promise<T> {\r\n\r\n    // Build the request options\r\n    const buildOptions = { headers: this.buildHeaderOptions(options) };\r\n\r\n    // Create the request observable\r\n    const request$ = ApiHelper.createRequest<T>({\r\n      client: this.httpClient,\r\n      destroy$: () => this.destroy$\r\n    }, options.method, options.requestUrl, options.requestBody, buildOptions,\r\n      this.config.onResponseBodyInvalid?.bind(this));\r\n\r\n    // Set the isRequesting state to true before making the request\r\n    this.isRequestingSubject.next(true);\r\n\r\n    const response = await new Promise<any>((resolve, reject) =>\r\n      request$.subscribe({ next: resolve, error: reject }));\r\n\r\n    // Reset the isRequesting state after the request completes\r\n    this.isRequestingSubject.next(false);\r\n\r\n    return response;\r\n  }\r\n\r\n  private async revokeToken(): Promise<void> {\r\n\r\n    try {\r\n      if (await this.interceptRevokeToken())\r\n        return;\r\n\r\n      this.isRevokingToken = true;\r\n      const defaultApiOptions: IApiOptions = {\r\n        ...this.buildApiOptions(this.localizeTokenService.config.refreshToken?.requestUrl || ''),\r\n        refreshToken: true\r\n      };\r\n\r\n      // If in third party mode need override the default API options with third party API options\r\n      const thirdParyApiOptions = this.buildThirdPartyApiOption(defaultApiOptions);\r\n      const apiOptions: IApiOptions = this.thirdPartyConfig?.isThirdPartyMode ? thirdParyApiOptions : defaultApiOptions;\r\n\r\n      // const revokeToken = await this.performRequest(apiOptions);\r\n      const revokeToken = await ApiHelper.performRequestWithRetry(\r\n        apiOptions,\r\n        this.config,\r\n        this.performRequest.bind(this)\r\n      );\r\n\r\n      await this.handleOnTokenRevoked(revokeToken);\r\n\r\n    } catch (error: any) {\r\n      // Handle the error, log it\r\n      await ApiHelper.invokeHook(this.config.onAutoLogout);\r\n      throw error;\r\n\r\n    } finally {\r\n      // Reset the revoking token state\r\n      this.isRevokingToken = false;\r\n    }\r\n  }\r\n\r\n  private buildThirdPartyApiOption(apiOptions: IApiOptions): IApiOptions {\r\n    return {\r\n      ...apiOptions,\r\n      requestUrl: this.thirdPartyConfig?.refreshToken.requestUrl || '',\r\n      method: this.thirdPartyConfig?.refreshToken.method || EMethod.GET,\r\n      requestBody: this.thirdPartyConfig?.refreshToken.body || {},\r\n      headers: {\r\n        ...apiOptions.headers,\r\n        [SCHEMES.X_SECRET]: `${this.localizeTokenService.clientSecret}`\r\n      }\r\n    };\r\n  }\r\n\r\n  /** default http request options */\r\n  private buildHeaderOptions(options: IApiOptions): HttpHeaders {\r\n\r\n    const headers: { [key: string]: string } = {\r\n      ...(options.refreshToken && { [SCHEMES.X_REFRESH_TOKEN]: `${this.refreshToken}` }),\r\n      ...(!options.isFormData && { [SCHEMES.CONTENT_TYPE]: 'application/json' }),\r\n      [SCHEMES.AUTHORIZATION]: `Bearer ${this.accessToken}`,\r\n      ...(this.needTenant && { [SCHEMES.X_TENANT]: `${this.tenantToken}` }),\r\n      // Add Client Secret header for third party revoke token request\r\n      ...(this.thirdPartyConfig?.isThirdPartyMode && { [SCHEMES.X_CLIENT]: `${this.localizeTokenService.clientId}` }),\r\n    };\r\n\r\n    return new HttpHeaders({ ...headers, ...options.headers });\r\n  }\r\n\r\n  private buildApiOptions(\r\n    baseUrl: string,\r\n    path: string = '',\r\n    method: EMethod = EMethod.GET,\r\n    requestBody: any = null,\r\n    headers?: { [key: string]: string }\r\n  ): IApiOptions {\r\n\r\n    const requestUrl = ApiHelper.buildUrl(baseUrl, path);\r\n    const isFormData = requestBody && requestBody instanceof FormData;\r\n    return {\r\n      ...this.apiOptions,\r\n      ...{ headers, method, requestUrl, requestBody, isFormData }\r\n    };\r\n  }\r\n\r\n\r\n  private async toWaitForPreviousRequest() {\r\n    if (this.isRevokingToken) {\r\n      await waitUntil(() => !this.isRevokingToken);\r\n    }\r\n\r\n    // to wait for each request in 50ms, even if the request is not completed\r\n    const waitMilliseconds = this.config.waitEachRequest?.milliseconds;\r\n    if (waitMilliseconds && this.isRequesting) {\r\n      // console.warn(`Request throttling: Another request is in progress. Waiting for ${waitMilliseconds}ms.`);\r\n      await waitFor(waitMilliseconds, this.isRequesting);\r\n    }\r\n  }\r\n\r\n  private async handleOnTokenRevoked(response: any): Promise<void> {\r\n    if (response?.status || response?.token) {\r\n      // If the response is successful, update the access token\r\n      this.accessToken = response.message || response.token; // response.token for third party revoke\r\n    } else {\r\n      // If the response indicates an error, invoke the onRevokeUnauthorized hook\r\n      console.warn('Token revocation failed, refresh token is expired.', response.message);\r\n      await ApiHelper.invokeHook(this.config.onRevokeUnauthorized);\r\n    }\r\n  }\r\n\r\n  private async interceptRevokeToken() {\r\n\r\n    if (this.isRevokingToken) {\r\n      console.warn('Token is already being revoked. Waiting for the current operation to complete...');\r\n      await waitUntil(() => !this.isRevokingToken);\r\n      return true;\r\n    }\r\n\r\n    if (!this.refreshToken) {\r\n      // await ApiHelper.invokeHook(this.apiConfigs.onAutoLogout);\r\n      throw new Error('Refresh token is missing. Please login again.');\r\n    }\r\n    return false;\r\n  }\r\n\r\n  validateConfig(): void {\r\n    if (this.localizeTokenService.config.requiredTenant\r\n      && !this.localizeTokenService.config.tenantToken?.name?.trim().length) {\r\n      throw Error('Tenant token is required but tenantTokenName is not configured');\r\n    }\r\n\r\n    if (!this.localizeTokenService.config.refreshToken?.requestUrl?.trim().length) {\r\n      throw Error('Revoke token URL is not configured - token refresh will not work');\r\n    }\r\n  }\r\n\r\n}\r\n"]}
|