@qbs-origin/origin-form 0.5.0
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/README.md +24 -0
- package/esm2022/lib/action-step-handler.mjs +163 -0
- package/esm2022/lib/auth-client.service.mjs +69 -0
- package/esm2022/lib/enums/label.keys.mjs +721 -0
- package/esm2022/lib/form-css.helper.mjs +367 -0
- package/esm2022/lib/formly/baseFormlyControlComponent.mjs +52 -0
- package/esm2022/lib/formly/baseFormlyStepComponent.mjs +59 -0
- package/esm2022/lib/formly/custom-section-separator.component.mjs +32 -0
- package/esm2022/lib/formly/form-section-separator.component.mjs +36 -0
- package/esm2022/lib/formly/formly-action.mjs +56 -0
- package/esm2022/lib/formly/formly-checkbox/formly-checkbox.component.mjs +52 -0
- package/esm2022/lib/formly/formly-dictionary-dropdown-tree/formly-dictionary-dropdown-tree.component.mjs +261 -0
- package/esm2022/lib/formly/formly-download-documents/formly-download-documents.component.mjs +126 -0
- package/esm2022/lib/formly/formly-enrol-card/formly-enrol-card.component.mjs +120 -0
- package/esm2022/lib/formly/formly-field-stepper/formly-field-stepper.component.mjs +762 -0
- package/esm2022/lib/formly/formly-generate-documents/formly-generate-documents.component.mjs +57 -0
- package/esm2022/lib/formly/formly-identification.component.mjs +84 -0
- package/esm2022/lib/formly/formly-open-banking/formly-open-banking.component.mjs +590 -0
- package/esm2022/lib/formly/formly-paragraph/formly-paragraph.component.mjs +35 -0
- package/esm2022/lib/formly/formly-radio/formly-radio-component.mjs +49 -0
- package/esm2022/lib/formly/formly-row-fille.mjs +12 -0
- package/esm2022/lib/formly/formly-scan-id/formly-scan-id.component.mjs +284 -0
- package/esm2022/lib/formly/formly-sign/formly-sign.component.mjs +173 -0
- package/esm2022/lib/formly/formly-upload-documents/formly-upload-documents.component.mjs +198 -0
- package/esm2022/lib/formly/formly-validate-contact-info/formly-validate-contact-info.component.mjs +124 -0
- package/esm2022/lib/formly/formly-view-documents/formly-view-documents.component.mjs +245 -0
- package/esm2022/lib/formly/formly-view-offers/formly-view-offers.component.mjs +160 -0
- package/esm2022/lib/model-population.helper.mjs +265 -0
- package/esm2022/lib/models/application-type.model.mjs +12 -0
- package/esm2022/lib/models/application.model.mjs +30 -0
- package/esm2022/lib/models/auth/users.model.mjs +2 -0
- package/esm2022/lib/models/dictionary.model.mjs +20 -0
- package/esm2022/lib/models/flux.model.mjs +105 -0
- package/esm2022/lib/models/forms.model.mjs +572 -0
- package/esm2022/lib/models/label-info.model.mjs +2 -0
- package/esm2022/lib/models/label.model.mjs +2 -0
- package/esm2022/lib/models/language.model.mjs +3 -0
- package/esm2022/lib/models/list.model.mjs +2 -0
- package/esm2022/lib/models/partner.model.mjs +3 -0
- package/esm2022/lib/models/treeview.model.mjs +15 -0
- package/esm2022/lib/origin-form-auth.service.mjs +40 -0
- package/esm2022/lib/origin-form-config.model.mjs +2 -0
- package/esm2022/lib/origin-form-token.interceptor.mjs +35 -0
- package/esm2022/lib/origin-form.component.mjs +2391 -0
- package/esm2022/lib/origin-form.module.mjs +479 -0
- package/esm2022/lib/origin-form.service.mjs +14 -0
- package/esm2022/lib/others/check-list.database.mjs +55 -0
- package/esm2022/lib/others/config-service.mjs +42 -0
- package/esm2022/lib/others/dictionary-label-info.mjs +3 -0
- package/esm2022/lib/others/environment-type.mjs +21 -0
- package/esm2022/lib/others/external-link.directive.mjs +49 -0
- package/esm2022/lib/others/flux-helper.mjs +1397 -0
- package/esm2022/lib/others/picker.component.mjs +119 -0
- package/esm2022/lib/others/translation.pipe.mjs +21 -0
- package/esm2022/lib/others/translations-helper.mjs +258 -0
- package/esm2022/lib/others/utils.mjs +272 -0
- package/esm2022/lib/services/applicationData.service.mjs +145 -0
- package/esm2022/lib/services/auth-http.service.mjs +80 -0
- package/esm2022/lib/services/dialog.service.mjs +56 -0
- package/esm2022/lib/services/dictionary.service.mjs +198 -0
- package/esm2022/lib/services/forms.service.mjs +47 -0
- package/esm2022/lib/services/labels.service.mjs +29 -0
- package/esm2022/lib/services/language.service.mjs +24 -0
- package/esm2022/lib/services/open-banking.service.mjs +194 -0
- package/esm2022/lib/services/origin-form-signalr-handler.service.mjs +107 -0
- package/esm2022/lib/services/origin-form-signalr.service.mjs +105 -0
- package/esm2022/lib/services/otp.service.mjs +28 -0
- package/esm2022/lib/services/proxy.service.mjs +79 -0
- package/esm2022/lib/services/scroll-to-error.service.mjs +369 -0
- package/esm2022/lib/services/translation.service.mjs +27 -0
- package/esm2022/lib/shared-components/confirmation.component.mjs +34 -0
- package/esm2022/lib/shared-components/dictionaries-tree.component.mjs +301 -0
- package/esm2022/lib/shared-components/grid.component.mjs +241 -0
- package/esm2022/lib/shared-components/treeview/treeview.component.mjs +224 -0
- package/esm2022/lib/theme-css.mjs +2254 -0
- package/esm2022/lib/theme-injector.service.mjs +26 -0
- package/esm2022/public-api.mjs +4 -0
- package/esm2022/qbs-origin-origin-form.mjs +5 -0
- package/fesm2022/qbs-origin-origin-form.mjs +15215 -0
- package/fesm2022/qbs-origin-origin-form.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/action-step-handler.d.ts +49 -0
- package/lib/auth-client.service.d.ts +17 -0
- package/lib/enums/label.keys.d.ts +720 -0
- package/lib/form-css.helper.d.ts +28 -0
- package/lib/formly/baseFormlyControlComponent.d.ts +25 -0
- package/lib/formly/baseFormlyStepComponent.d.ts +29 -0
- package/lib/formly/custom-section-separator.component.d.ts +6 -0
- package/lib/formly/form-section-separator.component.d.ts +10 -0
- package/lib/formly/formly-action.d.ts +13 -0
- package/lib/formly/formly-checkbox/formly-checkbox.component.d.ts +15 -0
- package/lib/formly/formly-dictionary-dropdown-tree/formly-dictionary-dropdown-tree.component.d.ts +45 -0
- package/lib/formly/formly-download-documents/formly-download-documents.component.d.ts +22 -0
- package/lib/formly/formly-enrol-card/formly-enrol-card.component.d.ts +114 -0
- package/lib/formly/formly-field-stepper/formly-field-stepper.component.d.ts +79 -0
- package/lib/formly/formly-generate-documents/formly-generate-documents.component.d.ts +17 -0
- package/lib/formly/formly-identification.component.d.ts +19 -0
- package/lib/formly/formly-open-banking/formly-open-banking.component.d.ts +119 -0
- package/lib/formly/formly-paragraph/formly-paragraph.component.d.ts +10 -0
- package/lib/formly/formly-radio/formly-radio-component.d.ts +15 -0
- package/lib/formly/formly-row-fille.d.ts +6 -0
- package/lib/formly/formly-scan-id/formly-scan-id.component.d.ts +41 -0
- package/lib/formly/formly-sign/formly-sign.component.d.ts +36 -0
- package/lib/formly/formly-upload-documents/formly-upload-documents.component.d.ts +25 -0
- package/lib/formly/formly-validate-contact-info/formly-validate-contact-info.component.d.ts +79 -0
- package/lib/formly/formly-view-documents/formly-view-documents.component.d.ts +33 -0
- package/lib/formly/formly-view-offers/formly-view-offers.component.d.ts +23 -0
- package/lib/model-population.helper.d.ts +8 -0
- package/lib/models/application-type.model.d.ts +27 -0
- package/lib/models/application.model.d.ts +107 -0
- package/lib/models/auth/users.model.d.ts +20 -0
- package/lib/models/dictionary.model.d.ts +77 -0
- package/lib/models/flux.model.d.ts +101 -0
- package/lib/models/forms.model.d.ts +504 -0
- package/lib/models/label-info.model.d.ts +10 -0
- package/lib/models/label.model.d.ts +4 -0
- package/lib/models/language.model.d.ts +5 -0
- package/lib/models/list.model.d.ts +8 -0
- package/lib/models/partner.model.d.ts +12 -0
- package/lib/models/treeview.model.d.ts +17 -0
- package/lib/origin-form-auth.service.d.ts +15 -0
- package/lib/origin-form-config.model.d.ts +12 -0
- package/lib/origin-form-token.interceptor.d.ts +12 -0
- package/lib/origin-form.component.d.ts +231 -0
- package/lib/origin-form.module.d.ts +84 -0
- package/lib/origin-form.service.d.ts +6 -0
- package/lib/others/check-list.database.d.ts +16 -0
- package/lib/others/config-service.d.ts +22 -0
- package/lib/others/dictionary-label-info.d.ts +6 -0
- package/lib/others/environment-type.d.ts +8 -0
- package/lib/others/external-link.directive.d.ts +12 -0
- package/lib/others/flux-helper.d.ts +115 -0
- package/lib/others/picker.component.d.ts +36 -0
- package/lib/others/translation.pipe.d.ts +10 -0
- package/lib/others/translations-helper.d.ts +31 -0
- package/lib/others/utils.d.ts +37 -0
- package/lib/services/applicationData.service.d.ts +35 -0
- package/lib/services/auth-http.service.d.ts +21 -0
- package/lib/services/dialog.service.d.ts +20 -0
- package/lib/services/dictionary.service.d.ts +89 -0
- package/lib/services/forms.service.d.ts +17 -0
- package/lib/services/labels.service.d.ts +13 -0
- package/lib/services/language.service.d.ts +14 -0
- package/lib/services/open-banking.service.d.ts +137 -0
- package/lib/services/origin-form-signalr-handler.service.d.ts +29 -0
- package/lib/services/origin-form-signalr.service.d.ts +24 -0
- package/lib/services/otp.service.d.ts +22 -0
- package/lib/services/proxy.service.d.ts +29 -0
- package/lib/services/scroll-to-error.service.d.ts +54 -0
- package/lib/services/translation.service.d.ts +10 -0
- package/lib/shared-components/confirmation.component.d.ts +77 -0
- package/lib/shared-components/dictionaries-tree.component.d.ts +51 -0
- package/lib/shared-components/grid.component.d.ts +138 -0
- package/lib/shared-components/treeview/treeview.component.d.ts +121 -0
- package/lib/theme-css.d.ts +2 -0
- package/lib/theme-injector.service.d.ts +8 -0
- package/package.json +42 -0
- package/public-api.d.ts +3 -0
- package/schematics-compiled/collection.json +10 -0
- package/schematics-compiled/ng-add/index.d.ts +2 -0
- package/schematics-compiled/ng-add/index.js +67 -0
- package/schematics-compiled/ng-add/index.js.map +1 -0
- package/schematics-compiled/ng-add/schema.json +8 -0
- package/src/lib/assets/fonts/Figtree-Bold.ttf +0 -0
- package/src/lib/assets/fonts/Figtree-Light.ttf +0 -0
- package/src/lib/assets/fonts/Figtree-Regular.ttf +0 -0
- package/src/lib/assets/fonts/Sora-ExtraBold.ttf +0 -0
- package/src/lib/assets/fonts/Sora-Light.ttf +0 -0
- package/src/lib/assets/fonts/Sora-Regular.ttf +0 -0
- package/src/lib/assets/fonts/ttrounds-bold-webfont.woff +0 -0
- package/src/lib/assets/fonts/ttrounds-bold-webfont.woff2 +0 -0
- package/src/lib/assets/fonts/ttrounds-regular-webfont.woff +0 -0
- package/src/lib/assets/fonts/ttrounds-regular-webfont.woff2 +0 -0
- package/src/lib/assets/fonts/ttrounds-thin-webfont.woff +0 -0
- package/src/lib/assets/fonts/ttrounds-thin-webfont.woff2 +0 -0
- package/src/lib/assets/images/flag/icon-flag-de.svg +10 -0
- package/src/lib/assets/images/flag/icon-flag-en.svg +1 -0
- package/src/lib/assets/images/flag/icon-flag-es.svg +11 -0
- package/src/lib/assets/images/flag/icon-flag-fr.svg +1 -0
- package/src/lib/assets/images/flag/icon-flag-ro.svg +11 -0
- package/src/lib/assets/images/flag/origin-form/new-id-card.png +0 -0
- package/src/lib/assets/images/flag/origin-form/old-id-card.png +0 -0
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { HttpParams } from '@angular/common/http';
|
|
3
|
+
import { of } from 'rxjs';
|
|
4
|
+
import { map, catchError } from 'rxjs/operators';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@angular/common/http";
|
|
7
|
+
import * as i2 from "../others/config-service";
|
|
8
|
+
// Status code mapping for API responses
|
|
9
|
+
const STATUS_CODE_MAP = {
|
|
10
|
+
0: 'received',
|
|
11
|
+
1: 'valid',
|
|
12
|
+
2: 'rejected',
|
|
13
|
+
3: 'expired',
|
|
14
|
+
4: 'terminated',
|
|
15
|
+
5: 'revokedByPsu'
|
|
16
|
+
};
|
|
17
|
+
// Bank-specific transaction period limits (in days)
|
|
18
|
+
const BANK_TRANSACTION_LIMITS = {
|
|
19
|
+
'BCR': 90,
|
|
20
|
+
'BTRL': 120, // BT - Banca Transilvania
|
|
21
|
+
'BT': 90,
|
|
22
|
+
'BRD': 730, // 2 years
|
|
23
|
+
'RZBA': -1, // RZB - unlimited
|
|
24
|
+
'PRVA': 180, // Patria Bank
|
|
25
|
+
'CEC': 180,
|
|
26
|
+
'INGB': 90,
|
|
27
|
+
'ING': 90
|
|
28
|
+
};
|
|
29
|
+
export class OpenBankingService {
|
|
30
|
+
constructor(http, configService) {
|
|
31
|
+
this.http = http;
|
|
32
|
+
this.configService = configService;
|
|
33
|
+
this.apiUrl = '';
|
|
34
|
+
// Get API URL from app config
|
|
35
|
+
const config = this.configService.readConfig();
|
|
36
|
+
this.apiUrl = config.baseUrlSmartAccounts;
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Fix data URL format for logos
|
|
40
|
+
* Converts "data:image/svg;base64," to "data:image/svg+xml;base64,"
|
|
41
|
+
*/
|
|
42
|
+
fixDataUrlFormat(logoUrl) {
|
|
43
|
+
if (!logoUrl)
|
|
44
|
+
return undefined;
|
|
45
|
+
// If it's already a proper data URL, return as is
|
|
46
|
+
if (logoUrl.startsWith('data:image/')) {
|
|
47
|
+
// Fix common format issues
|
|
48
|
+
return logoUrl
|
|
49
|
+
.replace('data:image/svg;base64,', 'data:image/svg+xml;base64,')
|
|
50
|
+
.replace('data:image/png;base64,', 'data:image/png;base64,')
|
|
51
|
+
.replace('data:image/jpeg;base64,', 'data:image/jpeg;base64,')
|
|
52
|
+
.replace('data:image/jpg;base64,', 'data:image/jpeg;base64,');
|
|
53
|
+
}
|
|
54
|
+
return logoUrl;
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Get list of supported banks
|
|
58
|
+
* @param countryCode Optional country code filter
|
|
59
|
+
*/
|
|
60
|
+
getBanks(countryCode) {
|
|
61
|
+
let params = new HttpParams();
|
|
62
|
+
if (countryCode) {
|
|
63
|
+
params = params.set('countryCode', countryCode);
|
|
64
|
+
}
|
|
65
|
+
return this.http.get(`${this.apiUrl}/banks`, { params }).pipe(map(banks => {
|
|
66
|
+
return banks.map(bank => ({
|
|
67
|
+
...bank,
|
|
68
|
+
// Fix data URL format if needed
|
|
69
|
+
logoUrl: this.fixDataUrlFormat(bank.logoUrl)
|
|
70
|
+
}));
|
|
71
|
+
}), catchError((error) => {
|
|
72
|
+
console.error('Error fetching banks:', error);
|
|
73
|
+
return of([]);
|
|
74
|
+
}));
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Initialize consent flow
|
|
78
|
+
*/
|
|
79
|
+
initializeConsent(request) {
|
|
80
|
+
return this.http.post(`${this.apiUrl}/consents/initialize`, request).pipe(catchError((error) => {
|
|
81
|
+
console.error('Error initializing consent:', error);
|
|
82
|
+
return of();
|
|
83
|
+
}));
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Get consent status
|
|
87
|
+
*/
|
|
88
|
+
getConsentStatus(bankCode, consentId) {
|
|
89
|
+
return this.http.get(`${this.apiUrl}/consents/${bankCode}/${consentId}`).pipe(map(response => {
|
|
90
|
+
// Map numeric status to string if needed
|
|
91
|
+
if (typeof response.status === 'number') {
|
|
92
|
+
return {
|
|
93
|
+
...response,
|
|
94
|
+
status: STATUS_CODE_MAP[response.status] || 'received'
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
return response;
|
|
98
|
+
}), catchError((error) => {
|
|
99
|
+
console.error('Error checking consent status:', error);
|
|
100
|
+
return of();
|
|
101
|
+
}));
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Process consent callback
|
|
105
|
+
*/
|
|
106
|
+
processConsentCallback(params) {
|
|
107
|
+
const queryParams = new HttpParams()
|
|
108
|
+
.set('message', params.message)
|
|
109
|
+
.set('type', params.type)
|
|
110
|
+
.set('internalConsentId', params.internalConsentId)
|
|
111
|
+
.set('code', params.code || '')
|
|
112
|
+
.set('state', params.state || '');
|
|
113
|
+
return this.http.get(`${this.apiUrl}/consents/callback`, { params: queryParams }).pipe(catchError((error) => {
|
|
114
|
+
console.error('Error processing consent callback:', error);
|
|
115
|
+
return of({
|
|
116
|
+
message: 'Consent processed (mock)',
|
|
117
|
+
internalConsentId: params.internalConsentId,
|
|
118
|
+
status: 'approved'
|
|
119
|
+
});
|
|
120
|
+
}));
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Get accounts for a consent
|
|
124
|
+
*/
|
|
125
|
+
getAccounts(consentId, bankCode) {
|
|
126
|
+
const params = new HttpParams()
|
|
127
|
+
.set('consentId', consentId)
|
|
128
|
+
.set('bankCode', bankCode);
|
|
129
|
+
return this.http.get(`${this.apiUrl}/accounts`, { params }).pipe(map(accounts => {
|
|
130
|
+
return accounts.map(account => ({
|
|
131
|
+
...account,
|
|
132
|
+
}));
|
|
133
|
+
}), catchError((error) => {
|
|
134
|
+
console.error('Error fetching accounts:', error);
|
|
135
|
+
return of();
|
|
136
|
+
}));
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Get maximum transaction period for a bank
|
|
140
|
+
* @param bankCode The bank code
|
|
141
|
+
* @returns Maximum number of days or -1 for unlimited
|
|
142
|
+
*/
|
|
143
|
+
getMaxTransactionPeriod(bankCode) {
|
|
144
|
+
const upperBankCode = bankCode.toUpperCase();
|
|
145
|
+
// Check if we have a specific limit for this bank
|
|
146
|
+
if (BANK_TRANSACTION_LIMITS.hasOwnProperty(upperBankCode)) {
|
|
147
|
+
return BANK_TRANSACTION_LIMITS[upperBankCode];
|
|
148
|
+
}
|
|
149
|
+
// Default to 90 days if bank not specified
|
|
150
|
+
return 90;
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Get transactions for an account
|
|
154
|
+
*/
|
|
155
|
+
getTransactions(request) {
|
|
156
|
+
let params = new HttpParams()
|
|
157
|
+
.set('consentId', request.consentId)
|
|
158
|
+
.set('bankCode', request.bankCode)
|
|
159
|
+
.set('accountId', request.accountId);
|
|
160
|
+
if (request.fromDate) {
|
|
161
|
+
params = params.set('fromDate', request.fromDate);
|
|
162
|
+
}
|
|
163
|
+
if (request.toDate) {
|
|
164
|
+
params = params.set('toDate', request.toDate);
|
|
165
|
+
}
|
|
166
|
+
if (request.page) {
|
|
167
|
+
params = params.set('page', request.page.toString());
|
|
168
|
+
}
|
|
169
|
+
if (request.pageSize) {
|
|
170
|
+
params = params.set('pageSize', request.pageSize.toString());
|
|
171
|
+
}
|
|
172
|
+
return this.http.get(`${this.apiUrl}/transactions`, { params }).pipe(map(page => {
|
|
173
|
+
return {
|
|
174
|
+
...page,
|
|
175
|
+
pageNumber: page.page,
|
|
176
|
+
items: page.items.map(txn => ({
|
|
177
|
+
...txn
|
|
178
|
+
}))
|
|
179
|
+
};
|
|
180
|
+
}), catchError((error) => {
|
|
181
|
+
console.error('Error fetching transactions:', error);
|
|
182
|
+
return of();
|
|
183
|
+
}));
|
|
184
|
+
}
|
|
185
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OpenBankingService, deps: [{ token: i1.HttpClient }, { token: i2.ConfigService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
186
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OpenBankingService, providedIn: 'root' }); }
|
|
187
|
+
}
|
|
188
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OpenBankingService, decorators: [{
|
|
189
|
+
type: Injectable,
|
|
190
|
+
args: [{
|
|
191
|
+
providedIn: 'root',
|
|
192
|
+
}]
|
|
193
|
+
}], ctorParameters: () => [{ type: i1.HttpClient }, { type: i2.ConfigService }] });
|
|
194
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"open-banking.service.js","sourceRoot":"","sources":["../../../../../projects/origin-form/src/lib/services/open-banking.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { HttpClient, HttpParams } from '@angular/common/http';\nimport { Observable, of } from 'rxjs';\nimport { map, catchError } from 'rxjs/operators';\nimport { ConfigService } from '../others/config-service';\n\n// Bank interfaces\nexport interface Bank {\n  id: string;\n  bankCode: string;\n  displayName: string;\n  logoUrl?: string;\n  countryCode: string;\n  requiresRedirect: boolean;\n  supportedFeatures: string;\n  isActive: boolean;\n}\n\n// Consent interfaces\nexport interface ConsentInitRequest {\n  bankCode: string;\n  psuEmail: string;\n  psuIntermediaryId: string;\n  periodOfValidity?: number; // Days, default 90\n  tcAccepted: boolean;\n}\n\nexport interface ConsentInitResponse {\n  authorizationUrl: string;\n  consentId: string;\n}\n\nexport interface ConsentStatus {\n  consentId: string;\n  status: 'received' | 'valid' | 'rejected' | 'expired' | 'terminated' | 'revokedByPsu';\n  expiresAt?: string;\n}\n\n// Status code mapping for API responses\nconst STATUS_CODE_MAP: { [key: number]: ConsentStatus['status'] } = {\n  0: 'received',\n  1: 'valid',\n  2: 'rejected',\n  3: 'expired',\n  4: 'terminated',\n  5: 'revokedByPsu'\n};\n\n// Bank-specific transaction period limits (in days)\nconst BANK_TRANSACTION_LIMITS: { [key: string]: number } = {\n  'BCR': 90,\n  'BTRL': 120,  // BT - Banca Transilvania\n  'BT': 90,\n  'BRD': 730,  // 2 years\n  'RZBA': -1,  // RZB - unlimited\n  'PRVA': 180,  // Patria Bank\n  'CEC': 180,\n  'INGB': 90,\n  'ING': 90\n};\n\nexport interface ConsentCallbackParams {\n  message: string;\n  type: string;\n  internalConsentId: string;\n  code?: string;\n  state?: string;\n}\n\nexport interface ConsentCallbackResponse {\n  message: string;\n  internalConsentId: string;\n  status: 'approved' | 'rejected';\n}\n\n// Account interfaces\nexport interface Account {\n  id: string;\n  accountId: string;\n  iban: string;\n  currency: string;\n  name: string;\n  accountHolderName: string;\n  accountHolderConfidence: string;\n  balance: number;\n  lastSyncedAt: string;\n}\n\n// Transaction interfaces\nexport interface Transaction {\n  transactionId: string;\n  accountId: string;\n  amount: number;\n  currency: string;\n  bookingDate: string;\n  valueDate?: string;\n  creditorName?: string;\n  creditorIBAN?: string;\n  debtorName?: string;\n  debtorIBAN?: string;\n  remittanceInformation?: string;\n  status: 'Booked' | 'Pending';\n  type: 'Debit' | 'Credit';\n  // Additional fields for compatibility\n  description?: string;\n  transactionDate?: string;\n  category?: string;\n  merchantName?: string;\n  reference?: string;\n}\n\nexport interface TransactionPage {\n  items: Transaction[];\n  page: number;\n  pageNumber?: number; // Alias for page\n  pageSize: number;\n  totalCount: number;\n  totalPages: number;\n  hasNextPage: boolean;\n  hasPreviousPage: boolean;\n  nextPageLink?: string;\n}\n\nexport interface TransactionRequest {\n  consentId: string;\n  bankCode: string;\n  accountId: string;\n  fromDate?: string;\n  toDate?: string;\n  page?: number;\n  pageSize?: number;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class OpenBankingService {\n  private apiUrl: string = '';\n\n  constructor(\n    private http: HttpClient,\n    private configService: ConfigService\n  ) {\n    // Get API URL from app config\n    const config = this.configService.readConfig();\n    this.apiUrl = config.baseUrlSmartAccounts;\n  }\n\n  /**\n   * Fix data URL format for logos\n   * Converts \"data:image/svg;base64,\" to \"data:image/svg+xml;base64,\"\n   */\n  private fixDataUrlFormat(logoUrl?: string): string | undefined {\n    if (!logoUrl) return undefined;\n\n    // If it's already a proper data URL, return as is\n    if (logoUrl.startsWith('data:image/')) {\n      // Fix common format issues\n      return logoUrl\n        .replace('data:image/svg;base64,', 'data:image/svg+xml;base64,')\n        .replace('data:image/png;base64,', 'data:image/png;base64,')\n        .replace('data:image/jpeg;base64,', 'data:image/jpeg;base64,')\n        .replace('data:image/jpg;base64,', 'data:image/jpeg;base64,');\n    }\n\n    return logoUrl;\n  }\n\n  /**\n   * Get list of supported banks\n   * @param countryCode Optional country code filter\n   */\n  getBanks(countryCode?: string): Observable<Bank[]> {\n    let params = new HttpParams();\n    if (countryCode) {\n      params = params.set('countryCode', countryCode);\n    }\n\n    return this.http.get<Bank[]>(`${this.apiUrl}/banks`, { params }).pipe(\n      map(banks => {\n        return banks.map(bank => ({\n          ...bank,\n          // Fix data URL format if needed\n          logoUrl: this.fixDataUrlFormat(bank.logoUrl)\n        }));\n      }),\n      catchError((error) => {\n        console.error('Error fetching banks:', error);\n        return of([]);\n      })\n    );\n  }\n\n  /**\n   * Initialize consent flow\n   */\n  initializeConsent(request: ConsentInitRequest): Observable<ConsentInitResponse> {\n    return this.http.post<ConsentInitResponse>(`${this.apiUrl}/consents/initialize`, request).pipe(\n      catchError((error) => {\n        console.error('Error initializing consent:', error);\n        return of();\n      })\n    );\n  }\n\n  /**\n   * Get consent status\n   */\n  getConsentStatus(bankCode: string, consentId: string): Observable<ConsentStatus> {\n    return this.http.get<any>(`${this.apiUrl}/consents/${bankCode}/${consentId}`).pipe(\n      map(response => {\n        // Map numeric status to string if needed\n        if (typeof response.status === 'number') {\n          return {\n            ...response,\n            status: STATUS_CODE_MAP[response.status] || 'received'\n          };\n        }\n        return response;\n      }),\n      catchError((error) => {\n        console.error('Error checking consent status:', error);\n        return of();\n      })\n    );\n  }\n\n  /**\n   * Process consent callback\n   */\n  processConsentCallback(params: ConsentCallbackParams): Observable<ConsentCallbackResponse> {\n    const queryParams = new HttpParams()\n      .set('message', params.message)\n      .set('type', params.type)\n      .set('internalConsentId', params.internalConsentId)\n      .set('code', params.code || '')\n      .set('state', params.state || '');\n\n    return this.http.get<ConsentCallbackResponse>(`${this.apiUrl}/consents/callback`, { params: queryParams }).pipe(\n      catchError((error) => {\n        console.error('Error processing consent callback:', error);\n        return of({\n          message: 'Consent processed (mock)',\n          internalConsentId: params.internalConsentId,\n          status: 'approved' as const\n        });\n      })\n    );\n  }\n\n  /**\n   * Get accounts for a consent\n   */\n  getAccounts(consentId: string, bankCode: string): Observable<Account[]> {\n    const params = new HttpParams()\n      .set('consentId', consentId)\n      .set('bankCode', bankCode);\n\n    return this.http.get<Account[]>(`${this.apiUrl}/accounts`, { params }).pipe(\n      map(accounts => {\n        return accounts.map(account => ({\n          ...account,\n        }));\n      }),\n      catchError((error) => {\n        console.error('Error fetching accounts:', error);\n        return of();\n      })\n    );\n  }\n\n  /**\n   * Get maximum transaction period for a bank\n   * @param bankCode The bank code\n   * @returns Maximum number of days or -1 for unlimited\n   */\n  getMaxTransactionPeriod(bankCode: string): number {\n    const upperBankCode = bankCode.toUpperCase();\n    // Check if we have a specific limit for this bank\n    if (BANK_TRANSACTION_LIMITS.hasOwnProperty(upperBankCode)) {\n      return BANK_TRANSACTION_LIMITS[upperBankCode];\n    }\n    // Default to 90 days if bank not specified\n    return 90;\n  }\n\n  /**\n   * Get transactions for an account\n   */\n  getTransactions(request: TransactionRequest): Observable<TransactionPage> {\n    let params = new HttpParams()\n      .set('consentId', request.consentId)\n      .set('bankCode', request.bankCode)\n      .set('accountId', request.accountId);\n\n    if (request.fromDate) {\n      params = params.set('fromDate', request.fromDate);\n    }\n    if (request.toDate) {\n      params = params.set('toDate', request.toDate);\n    }\n    if (request.page) {\n      params = params.set('page', request.page.toString());\n    }\n    if (request.pageSize) {\n      params = params.set('pageSize', request.pageSize.toString());\n    }\n\n    return this.http.get<TransactionPage>(`${this.apiUrl}/transactions`, { params }).pipe(\n      map(page => {\n        return {\n          ...page,\n          pageNumber: page.page,\n          items: page.items.map(txn => ({\n            ...txn\n          }))\n        };\n      }),\n      catchError((error) => {\n        console.error('Error fetching transactions:', error);\n        return of();\n      })\n    );\n  }\n}"]}
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import { Subject, Subscription, BehaviorSubject, firstValueFrom, interval, } from 'rxjs';
|
|
3
|
+
import { ApplicationDataService } from './applicationData.service';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "./origin-form-signalr.service";
|
|
6
|
+
export class OriginFormSignalrHandlerService {
|
|
7
|
+
constructor(originFormSignalrService) {
|
|
8
|
+
this.originFormSignalrService = originFormSignalrService;
|
|
9
|
+
this.subscriptions = new Subscription();
|
|
10
|
+
this.applicationDataService = inject(ApplicationDataService);
|
|
11
|
+
this.currentAppDataUuid = null;
|
|
12
|
+
this._isLoading = new BehaviorSubject(false);
|
|
13
|
+
this.isLoading$ = this._isLoading.asObservable();
|
|
14
|
+
this._operationCompleted = new Subject();
|
|
15
|
+
this.operationCompleted$ = this._operationCompleted.asObservable();
|
|
16
|
+
this.originFormSignalrService
|
|
17
|
+
.initiateSignalrConnection()
|
|
18
|
+
.then(() => this.subscribeToCoreEvents())
|
|
19
|
+
.catch((error) => console.error('CollectorFormSignalrHandlerService: Error initiating base SignalR connection', error));
|
|
20
|
+
}
|
|
21
|
+
subscribeToCoreEvents() {
|
|
22
|
+
const kycSub = this.originFormSignalrService.kycProcessingComplete.subscribe((appDataId) => {
|
|
23
|
+
if (appDataId === this.currentAppDataUuid) {
|
|
24
|
+
this.setKycOperationCompleted();
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
console.warn(`Received KycProcessingComplete event for unexpected appDataId: ${appDataId}. Expected: ${this.currentAppDataUuid}`);
|
|
28
|
+
}
|
|
29
|
+
});
|
|
30
|
+
this.subscriptions.add(kycSub);
|
|
31
|
+
}
|
|
32
|
+
async setKycOperationCompleted() {
|
|
33
|
+
if (this.currentAppDataUuid) {
|
|
34
|
+
let appData = null;
|
|
35
|
+
try {
|
|
36
|
+
appData = await this.applicationDataService.getSteps(this.currentAppDataUuid);
|
|
37
|
+
}
|
|
38
|
+
catch (error) {
|
|
39
|
+
console.error('Error fetching steps after KYC completion:', error);
|
|
40
|
+
}
|
|
41
|
+
this._isLoading.next(false);
|
|
42
|
+
this._operationCompleted.next({
|
|
43
|
+
operationIdentifier: 'KYC_PROCESSING',
|
|
44
|
+
success: true,
|
|
45
|
+
data: appData,
|
|
46
|
+
});
|
|
47
|
+
this.currentAppDataUuid = null;
|
|
48
|
+
}
|
|
49
|
+
if (this.poolingInterval) {
|
|
50
|
+
this.poolingInterval.unsubscribe();
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
async startBackupPooling() {
|
|
54
|
+
let isBusy = false;
|
|
55
|
+
this.poolingInterval = interval(5000)
|
|
56
|
+
.subscribe(() => {
|
|
57
|
+
if (this.currentAppDataUuid && !isBusy) {
|
|
58
|
+
isBusy = true;
|
|
59
|
+
this.applicationDataService.getProcessStatus(this.currentAppDataUuid).subscribe(status => {
|
|
60
|
+
isBusy = false;
|
|
61
|
+
if (status === 2) {
|
|
62
|
+
this.setKycOperationCompleted();
|
|
63
|
+
}
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
async initiateKycProcessing(appDataUuid) {
|
|
69
|
+
if (this._isLoading.value) {
|
|
70
|
+
console.warn('Another operation is already in progress.');
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
this._isLoading.next(true);
|
|
74
|
+
this.currentAppDataUuid = appDataUuid;
|
|
75
|
+
try {
|
|
76
|
+
await firstValueFrom(this.applicationDataService.makeProcessApiCall(appDataUuid));
|
|
77
|
+
this.startBackupPooling(); //fire and forget
|
|
78
|
+
}
|
|
79
|
+
catch (error) {
|
|
80
|
+
console.error('Error initiating KYC processing (API call failed):', error);
|
|
81
|
+
this._isLoading.next(false);
|
|
82
|
+
this._operationCompleted.next({
|
|
83
|
+
operationIdentifier: 'KYC_PROCESSING',
|
|
84
|
+
success: false,
|
|
85
|
+
error,
|
|
86
|
+
});
|
|
87
|
+
this.currentAppDataUuid = null;
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
ngOnDestroy() {
|
|
91
|
+
if (this.subscriptions) {
|
|
92
|
+
this.subscriptions.unsubscribe();
|
|
93
|
+
}
|
|
94
|
+
if (this.poolingInterval) {
|
|
95
|
+
this.poolingInterval.unsubscribe();
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OriginFormSignalrHandlerService, deps: [{ token: i1.OriginFormSignalrService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
99
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OriginFormSignalrHandlerService, providedIn: 'root' }); }
|
|
100
|
+
}
|
|
101
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OriginFormSignalrHandlerService, decorators: [{
|
|
102
|
+
type: Injectable,
|
|
103
|
+
args: [{
|
|
104
|
+
providedIn: 'root',
|
|
105
|
+
}]
|
|
106
|
+
}], ctorParameters: () => [{ type: i1.OriginFormSignalrService }] });
|
|
107
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"origin-form-signalr-handler.service.js","sourceRoot":"","sources":["../../../../../projects/origin-form/src/lib/services/origin-form-signalr-handler.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, OnDestroy, inject } from '@angular/core';\nimport {\n  Observable,\n  Subject,\n  Subscription,\n  BehaviorSubject,\n  firstValueFrom,\n  interval,\n  startWith,\n  takeWhile,\n} from 'rxjs';\nimport { OriginFormSignalrService } from './origin-form-signalr.service';\nimport { ApplicationDataService } from './applicationData.service';\n\nexport interface OperationResult {\n  operationIdentifier: string;\n  success: boolean;\n  data?: any;\n  error?: any;\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class OriginFormSignalrHandlerService implements OnDestroy {\n  private subscriptions: Subscription = new Subscription();\n  private applicationDataService = inject(ApplicationDataService);\n  private currentAppDataUuid: string | null = null;\n\n  private _isLoading = new BehaviorSubject<boolean>(false);\n  public readonly isLoading$: Observable<boolean> = this._isLoading.asObservable();\n\n  private _operationCompleted = new Subject<OperationResult>();\n  public readonly operationCompleted$: Observable<OperationResult> =\n    this._operationCompleted.asObservable();\n\n  constructor(private originFormSignalrService: OriginFormSignalrService) {\n    this.originFormSignalrService\n      .initiateSignalrConnection()\n      .then(() => this.subscribeToCoreEvents())\n      .catch((error) =>\n        console.error(\n          'CollectorFormSignalrHandlerService: Error initiating base SignalR connection',\n          error\n        )\n      );\n  }\n\n  private subscribeToCoreEvents(): void {\n    const kycSub =\n      this.originFormSignalrService.kycProcessingComplete.subscribe(\n        (appDataId: string) => {\n          if (appDataId === this.currentAppDataUuid) {\n            this.setKycOperationCompleted();\n          } else {\n            console.warn(\n              `Received KycProcessingComplete event for unexpected appDataId: ${appDataId}. Expected: ${this.currentAppDataUuid}`\n            );\n          }\n        }\n      );\n    this.subscriptions.add(kycSub);\n  }\n\n  private async setKycOperationCompleted() {\n    if (this.currentAppDataUuid) {\n      let appData = null;\n\n      try {\n        appData = await this.applicationDataService.getSteps(this.currentAppDataUuid);\n      } catch (error) {\n        console.error('Error fetching steps after KYC completion:', error);\n      }\n\n      this._isLoading.next(false);\n\n      this._operationCompleted.next({\n        operationIdentifier: 'KYC_PROCESSING',\n        success: true,\n        data: appData,\n      });\n      this.currentAppDataUuid = null;\n    }\n    if (this.poolingInterval) {\n      this.poolingInterval.unsubscribe();\n    }\n  }\n\n  private poolingInterval: Subscription;\n\n  private async startBackupPooling() {\n    let isBusy = false;\n    this.poolingInterval = interval(5000)\n      .subscribe(() => {\n        if (this.currentAppDataUuid && !isBusy) {\n          isBusy = true;\n          this.applicationDataService.getProcessStatus(this.currentAppDataUuid).subscribe(status => {\n            isBusy = false;\n            if (status === 2) {\n              this.setKycOperationCompleted();\n            }\n          });\n        }\n      });\n  }\n\n  public async initiateKycProcessing(appDataUuid: string): Promise<void> {\n    if (this._isLoading.value) {\n      console.warn('Another operation is already in progress.');\n      return;\n    }\n    this._isLoading.next(true);\n    this.currentAppDataUuid = appDataUuid;\n    try {\n      await firstValueFrom(\n        this.applicationDataService.makeProcessApiCall(appDataUuid)\n      );\n      this.startBackupPooling();//fire and forget\n    } catch (error) {\n      console.error(\n        'Error initiating KYC processing (API call failed):',\n        error\n      );\n      this._isLoading.next(false);\n      this._operationCompleted.next({\n        operationIdentifier: 'KYC_PROCESSING',\n        success: false,\n        error,\n      });\n      this.currentAppDataUuid = null;\n    }\n  }\n\n  ngOnDestroy(): void {\n    if (this.subscriptions) {\n      this.subscriptions.unsubscribe();\n    }\n    if (this.poolingInterval) {\n      this.poolingInterval.unsubscribe();\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import * as signalR from '@microsoft/signalr';
|
|
3
|
+
import { Subject } from 'rxjs';
|
|
4
|
+
import { ConfigService } from '../others/config-service';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
export class OriginFormSignalrService {
|
|
7
|
+
constructor() {
|
|
8
|
+
this.connection = null;
|
|
9
|
+
this.connectionPromise = null;
|
|
10
|
+
// Generic subjects for different events
|
|
11
|
+
this.eventSubjects = new Map();
|
|
12
|
+
// Specific event for KYC completion as an example
|
|
13
|
+
this.kycProcessingComplete = new Subject();
|
|
14
|
+
this.configService = inject(ConfigService);
|
|
15
|
+
this.hubUrl =
|
|
16
|
+
this.configService.readConfig().baseUrlBusiness + '/collectorhub';
|
|
17
|
+
this.registerSpecificEvents();
|
|
18
|
+
}
|
|
19
|
+
registerSpecificEvents() {
|
|
20
|
+
this.eventSubjects.set('KycProcessingComplete', this.kycProcessingComplete);
|
|
21
|
+
}
|
|
22
|
+
async initiateSignalrConnection() {
|
|
23
|
+
if (this.connection &&
|
|
24
|
+
this.connection.state === signalR.HubConnectionState.Connected) {
|
|
25
|
+
console.log('SignalR connection already established.');
|
|
26
|
+
return Promise.resolve();
|
|
27
|
+
}
|
|
28
|
+
if (this.connectionPromise) {
|
|
29
|
+
return this.connectionPromise;
|
|
30
|
+
}
|
|
31
|
+
this.connection = new signalR.HubConnectionBuilder()
|
|
32
|
+
.withUrl(this.hubUrl, {
|
|
33
|
+
skipNegotiation: true,
|
|
34
|
+
transport: signalR.HttpTransportType.WebSockets,
|
|
35
|
+
})
|
|
36
|
+
.withAutomaticReconnect()
|
|
37
|
+
.withHubProtocol(new signalR.JsonHubProtocol())
|
|
38
|
+
.build();
|
|
39
|
+
this.setSignalrClientMethods();
|
|
40
|
+
this.connectionPromise = this.connection
|
|
41
|
+
.start()
|
|
42
|
+
.then(() => {
|
|
43
|
+
if (this.connection) {
|
|
44
|
+
console.log(`Collector SignalR connection success! connectionId: ${this.connection.connectionId}`);
|
|
45
|
+
}
|
|
46
|
+
this.connectionPromise = null;
|
|
47
|
+
})
|
|
48
|
+
.catch((error) => {
|
|
49
|
+
console.error(`Collector SignalR connection error: ${error}`);
|
|
50
|
+
this.connectionPromise = null;
|
|
51
|
+
throw error;
|
|
52
|
+
});
|
|
53
|
+
return this.connectionPromise;
|
|
54
|
+
}
|
|
55
|
+
setSignalrClientMethods() {
|
|
56
|
+
if (!this.connection)
|
|
57
|
+
return;
|
|
58
|
+
// Generic handler: Iterate over pre-registered event subjects
|
|
59
|
+
this.eventSubjects.forEach((subject, eventName) => {
|
|
60
|
+
this.connection?.on(eventName, (data) => {
|
|
61
|
+
subject.next(data);
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
// Example for a specific event if not using the generic map for some reason
|
|
65
|
+
// this.connection.on('KycProcessingComplete', (data: any) => {
|
|
66
|
+
// this.kycProcessingComplete.next(data);
|
|
67
|
+
// });
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Allows dynamic registration of an event listener if the event name isn't known at design time
|
|
71
|
+
* or if a component wants to manage its own subject.
|
|
72
|
+
* @param eventName The name of the event from the server.
|
|
73
|
+
* @returns An RxJS Subject that will emit data when the event is received.
|
|
74
|
+
*/
|
|
75
|
+
listenForEvent(eventName) {
|
|
76
|
+
if (!this.eventSubjects.has(eventName)) {
|
|
77
|
+
const newSubject = new Subject();
|
|
78
|
+
this.eventSubjects.set(eventName, newSubject);
|
|
79
|
+
// If connection is already live, register the handler immediately
|
|
80
|
+
if (this.connection &&
|
|
81
|
+
this.connection.state === signalR.HubConnectionState.Connected) {
|
|
82
|
+
this.connection.on(eventName, (data) => {
|
|
83
|
+
newSubject.next(data);
|
|
84
|
+
});
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
return this.eventSubjects.get(eventName);
|
|
88
|
+
}
|
|
89
|
+
async stopConnection() {
|
|
90
|
+
if (this.connection &&
|
|
91
|
+
this.connection.state === signalR.HubConnectionState.Connected) {
|
|
92
|
+
await this.connection.stop();
|
|
93
|
+
console.log('Collector SignalR connection stopped.');
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OriginFormSignalrService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
97
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OriginFormSignalrService, providedIn: 'root' }); }
|
|
98
|
+
}
|
|
99
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OriginFormSignalrService, decorators: [{
|
|
100
|
+
type: Injectable,
|
|
101
|
+
args: [{
|
|
102
|
+
providedIn: 'root',
|
|
103
|
+
}]
|
|
104
|
+
}], ctorParameters: () => [] });
|
|
105
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"origin-form-signalr.service.js","sourceRoot":"","sources":["../../../../../projects/origin-form/src/lib/services/origin-form-signalr.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport * as signalR from '@microsoft/signalr';\nimport { Subject } from 'rxjs';\nimport { ConfigService } from '../others/config-service';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class OriginFormSignalrService {\n  private connection: signalR.HubConnection | null = null;\n  private connectionPromise: Promise<void> | null = null;\n  private hubUrl: string;\n\n  // Generic subjects for different events\n  private eventSubjects: Map<string, Subject<any>> = new Map();\n\n  // Specific event for KYC completion as an example\n  public kycProcessingComplete: Subject<any> = new Subject<any>();\n\n  private configService = inject(ConfigService);\n\n  constructor() {\n    this.hubUrl =\n      this.configService.readConfig().baseUrlBusiness + '/collectorhub';\n    this.registerSpecificEvents();\n  }\n\n  private registerSpecificEvents() {\n    this.eventSubjects.set('KycProcessingComplete', this.kycProcessingComplete);\n  }\n\n  public async initiateSignalrConnection(): Promise<void> {\n    if (\n      this.connection &&\n      this.connection.state === signalR.HubConnectionState.Connected\n    ) {\n      console.log('SignalR connection already established.');\n      return Promise.resolve();\n    }\n\n    if (this.connectionPromise) {\n      return this.connectionPromise;\n    }\n\n    this.connection = new signalR.HubConnectionBuilder()\n      .withUrl(this.hubUrl, {\n        skipNegotiation: true,\n        transport: signalR.HttpTransportType.WebSockets,\n      })\n      .withAutomaticReconnect()\n      .withHubProtocol(new signalR.JsonHubProtocol())\n      .build();\n\n    this.setSignalrClientMethods();\n\n    this.connectionPromise = this.connection\n      .start()\n      .then(() => {\n        if (this.connection) {\n          console.log(\n            `Collector SignalR connection success! connectionId: ${this.connection.connectionId}`\n          );\n        }\n        this.connectionPromise = null;\n      })\n      .catch((error: any) => {\n        console.error(`Collector SignalR connection error: ${error}`);\n        this.connectionPromise = null;\n        throw error;\n      });\n\n    return this.connectionPromise;\n  }\n\n  private setSignalrClientMethods(): void {\n    if (!this.connection) return;\n\n    // Generic handler: Iterate over pre-registered event subjects\n    this.eventSubjects.forEach((subject, eventName) => {\n      this.connection?.on(eventName, (data: any) => {\n        subject.next(data);\n      });\n    });\n\n    // Example for a specific event if not using the generic map for some reason\n    // this.connection.on('KycProcessingComplete', (data: any) => {\n    //   this.kycProcessingComplete.next(data);\n    // });\n  }\n\n  /**\n   * Allows dynamic registration of an event listener if the event name isn't known at design time\n   * or if a component wants to manage its own subject.\n   * @param eventName The name of the event from the server.\n   * @returns An RxJS Subject that will emit data when the event is received.\n   */\n  public listenForEvent<T>(eventName: string): Subject<T> {\n    if (!this.eventSubjects.has(eventName)) {\n      const newSubject = new Subject<T>();\n      this.eventSubjects.set(eventName, newSubject);\n      // If connection is already live, register the handler immediately\n      if (\n        this.connection &&\n        this.connection.state === signalR.HubConnectionState.Connected\n      ) {\n        this.connection.on(eventName, (data: T) => {\n          newSubject.next(data);\n        });\n      }\n    }\n    return this.eventSubjects.get(eventName) as Subject<T>;\n  }\n\n  public async stopConnection(): Promise<void> {\n    if (\n      this.connection &&\n      this.connection.state === signalR.HubConnectionState.Connected\n    ) {\n      await this.connection.stop();\n      console.log('Collector SignalR connection stopped.');\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { Injectable, inject } from '@angular/core';
|
|
2
|
+
import { catchError, map, of } from 'rxjs';
|
|
3
|
+
import { ConfigService } from '../others/config-service';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
import * as i1 from "./auth-http.service";
|
|
6
|
+
export class OtpService {
|
|
7
|
+
constructor(http) {
|
|
8
|
+
this.http = http;
|
|
9
|
+
this.configService = inject(ConfigService);
|
|
10
|
+
}
|
|
11
|
+
generate(params) {
|
|
12
|
+
const url = this.configService.readConfig().baseUrlInfrastructure +
|
|
13
|
+
`/otp/generate-app-otp`;
|
|
14
|
+
const data = { ...params, templateName: 'otpCode' };
|
|
15
|
+
return this.http.postExtended(url, data, { observe: 'response' }).pipe(map((response) => response), catchError((error) => of({ ...error, body: false })));
|
|
16
|
+
}
|
|
17
|
+
validate(data) {
|
|
18
|
+
const url = this.configService.readConfig().baseUrlInfrastructure +
|
|
19
|
+
`/otp/validate-app-otp`;
|
|
20
|
+
return this.http.postExtended(url, data, { observe: 'response' }).pipe(map((response) => response), catchError((error) => of({ ...error, body: false })));
|
|
21
|
+
}
|
|
22
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OtpService, deps: [{ token: i1.AuthHttpService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
23
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OtpService }); }
|
|
24
|
+
}
|
|
25
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: OtpService, decorators: [{
|
|
26
|
+
type: Injectable
|
|
27
|
+
}], ctorParameters: () => [{ type: i1.AuthHttpService }] });
|
|
28
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3RwLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9vcmlnaW4tZm9ybS9zcmMvbGliL3NlcnZpY2VzL290cC5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ25ELE9BQU8sRUFBRSxVQUFVLEVBQUUsR0FBRyxFQUFjLEVBQUUsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUN2RCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sMEJBQTBCLENBQUM7OztBQUl6RCxNQUFNLE9BQU8sVUFBVTtJQUNyQixZQUFvQixJQUFxQjtRQUFyQixTQUFJLEdBQUosSUFBSSxDQUFpQjtRQUV6QyxrQkFBYSxHQUFHLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUZNLENBQUM7SUFJN0MsUUFBUSxDQUFDLE1BS1I7UUFDQyxNQUFNLEdBQUcsR0FDUCxJQUFJLENBQUMsYUFBYSxDQUFDLFVBQVUsRUFBRSxDQUFDLHFCQUFxQjtZQUNyRCx1QkFBdUIsQ0FBQztRQUMxQixNQUFNLElBQUksR0FBRyxFQUFFLEdBQUcsTUFBTSxFQUFFLFlBQVksRUFBRSxTQUFTLEVBQUUsQ0FBQztRQUNwRCxPQUFPLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEdBQUcsRUFBRSxJQUFJLEVBQUUsRUFBRSxPQUFPLEVBQUUsVUFBVSxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQ3BFLEdBQUcsQ0FBQyxDQUFDLFFBQVEsRUFBRSxFQUFFLENBQUMsUUFBUSxDQUFDLEVBQzNCLFVBQVUsQ0FBQyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsR0FBRyxLQUFLLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxDQUFDLENBQUMsQ0FDckQsQ0FBQztJQUNKLENBQUM7SUFFRCxRQUFRLENBQUMsSUFJUjtRQUNDLE1BQU0sR0FBRyxHQUNQLElBQUksQ0FBQyxhQUFhLENBQUMsVUFBVSxFQUFFLENBQUMscUJBQXFCO1lBQ3JELHVCQUF1QixDQUFDO1FBQzFCLE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQyxZQUFZLENBQUMsR0FBRyxFQUFFLElBQUksRUFBRSxFQUFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FDcEUsR0FBRyxDQUFDLENBQUMsUUFBUSxFQUFFLEVBQUUsQ0FBQyxRQUFRLENBQUMsRUFDM0IsVUFBVSxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsRUFBRSxHQUFHLEtBQUssRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLENBQUMsQ0FBQyxDQUNyRCxDQUFDO0lBQ0osQ0FBQzsrR0FqQ1UsVUFBVTttSEFBVixVQUFVOzs0RkFBVixVQUFVO2tCQUR0QixVQUFVIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSwgaW5qZWN0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBjYXRjaEVycm9yLCBtYXAsIE9ic2VydmFibGUsIG9mIH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBDb25maWdTZXJ2aWNlIH0gZnJvbSAnLi4vb3RoZXJzL2NvbmZpZy1zZXJ2aWNlJztcbmltcG9ydCB7IEF1dGhIdHRwU2VydmljZSB9IGZyb20gJy4vYXV0aC1odHRwLnNlcnZpY2UnO1xuXG5ASW5qZWN0YWJsZSgpXG5leHBvcnQgY2xhc3MgT3RwU2VydmljZSB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgaHR0cDogQXV0aEh0dHBTZXJ2aWNlKSB7fVxuXG4gIGNvbmZpZ1NlcnZpY2UgPSBpbmplY3QoQ29uZmlnU2VydmljZSk7XG5cbiAgZ2VuZXJhdGUocGFyYW1zOiB7XG4gICAgcGhvbmVOdW1iZXI6IHN0cmluZyB8IG51bGw7XG4gICAgZW1haWxBZGRyZXNzOiBzdHJpbmcgfCBudWxsO1xuICAgIGFwcERhdGFJZDogc3RyaW5nO1xuICAgIHN0ZXBJZDogc3RyaW5nO1xuICB9KTogT2JzZXJ2YWJsZTxhbnk+IHtcbiAgICBjb25zdCB1cmwgPVxuICAgICAgdGhpcy5jb25maWdTZXJ2aWNlLnJlYWRDb25maWcoKS5iYXNlVXJsSW5mcmFzdHJ1Y3R1cmUgK1xuICAgICAgYC9vdHAvZ2VuZXJhdGUtYXBwLW90cGA7XG4gICAgY29uc3QgZGF0YSA9IHsgLi4ucGFyYW1zLCB0ZW1wbGF0ZU5hbWU6ICdvdHBDb2RlJyB9O1xuICAgIHJldHVybiB0aGlzLmh0dHAucG9zdEV4dGVuZGVkKHVybCwgZGF0YSwgeyBvYnNlcnZlOiAncmVzcG9uc2UnIH0pLnBpcGUoXG4gICAgICBtYXAoKHJlc3BvbnNlKSA9PiByZXNwb25zZSksXG4gICAgICBjYXRjaEVycm9yKChlcnJvcikgPT4gb2YoeyAuLi5lcnJvciwgYm9keTogZmFsc2UgfSkpXG4gICAgKTtcbiAgfVxuXG4gIHZhbGlkYXRlKGRhdGE6IHtcbiAgICBvdHBDb2RlOiBzdHJpbmc7XG4gICAgYXBwRGF0YUlkOiBzdHJpbmc7XG4gICAgc3RlcElkOiBzdHJpbmc7XG4gIH0pOiBPYnNlcnZhYmxlPGFueT4ge1xuICAgIGNvbnN0IHVybCA9XG4gICAgICB0aGlzLmNvbmZpZ1NlcnZpY2UucmVhZENvbmZpZygpLmJhc2VVcmxJbmZyYXN0cnVjdHVyZSArXG4gICAgICBgL290cC92YWxpZGF0ZS1hcHAtb3RwYDtcbiAgICByZXR1cm4gdGhpcy5odHRwLnBvc3RFeHRlbmRlZCh1cmwsIGRhdGEsIHsgb2JzZXJ2ZTogJ3Jlc3BvbnNlJyB9KS5waXBlKFxuICAgICAgbWFwKChyZXNwb25zZSkgPT4gcmVzcG9uc2UpLFxuICAgICAgY2F0Y2hFcnJvcigoZXJyb3IpID0+IG9mKHsgLi4uZXJyb3IsIGJvZHk6IGZhbHNlIH0pKVxuICAgICk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "./auth-http.service";
|
|
4
|
+
import * as i2 from "@angular/router";
|
|
5
|
+
export class ProxyService {
|
|
6
|
+
constructor(http, route) {
|
|
7
|
+
this.http = http;
|
|
8
|
+
this.route = route;
|
|
9
|
+
this.apiURL = 'https://apiro.id-kyc.com';
|
|
10
|
+
this.env = '';
|
|
11
|
+
this.tokenUrl = '';
|
|
12
|
+
this.debug = false;
|
|
13
|
+
this.langIso = '';
|
|
14
|
+
this.token = '';
|
|
15
|
+
this.debug = this.env == 'QA' || this.env == 'DEMO';
|
|
16
|
+
this.tokenUrl =
|
|
17
|
+
this.apiURL +
|
|
18
|
+
(this.env == 'PROD'
|
|
19
|
+
? '/public/authorization/token'
|
|
20
|
+
: '/dev_public/authorization/token');
|
|
21
|
+
}
|
|
22
|
+
getValuesForLog() {
|
|
23
|
+
return {
|
|
24
|
+
apiURL: this.apiURL,
|
|
25
|
+
orderID: this.orderID,
|
|
26
|
+
env: this.env,
|
|
27
|
+
tokenUrl: this.tokenUrl,
|
|
28
|
+
token: this.token,
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
getToken() {
|
|
32
|
+
return new Promise((resolve, reject) => {
|
|
33
|
+
var tokenUrlFull = this.tokenUrl + 'new?flow=origin';
|
|
34
|
+
if (this.debug)
|
|
35
|
+
console.log('GetToken called at' + tokenUrlFull);
|
|
36
|
+
this.http.get(tokenUrlFull).subscribe({
|
|
37
|
+
next: (resp) => {
|
|
38
|
+
this.token = resp.token;
|
|
39
|
+
sessionStorage.setItem('token-kyc', this.token);
|
|
40
|
+
if (this.debug)
|
|
41
|
+
console.log('GetToken responded: ' + JSON.stringify(this.getValuesForLog()));
|
|
42
|
+
resolve({ success: true, error: '' });
|
|
43
|
+
},
|
|
44
|
+
error: (error) => {
|
|
45
|
+
console.log('GetToken responded: ' + JSON.stringify(error));
|
|
46
|
+
resolve({ success: false, error: error });
|
|
47
|
+
},
|
|
48
|
+
complete: () => {
|
|
49
|
+
console.log('GetToken complete');
|
|
50
|
+
resolve({ success: true, error: '' });
|
|
51
|
+
},
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
async getTokenAndSave() {
|
|
56
|
+
try {
|
|
57
|
+
const tokenResp = (await this.getToken());
|
|
58
|
+
if (tokenResp.success) {
|
|
59
|
+
return this.token;
|
|
60
|
+
}
|
|
61
|
+
else {
|
|
62
|
+
throw new Error('Token retrieval failed');
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
catch (error) {
|
|
66
|
+
console.error('Error retrieving token:', error);
|
|
67
|
+
throw error;
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProxyService, deps: [{ token: i1.AuthHttpService }, { token: i2.ActivatedRoute }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
71
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProxyService, providedIn: 'root' }); }
|
|
72
|
+
}
|
|
73
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: ProxyService, decorators: [{
|
|
74
|
+
type: Injectable,
|
|
75
|
+
args: [{
|
|
76
|
+
providedIn: 'root',
|
|
77
|
+
}]
|
|
78
|
+
}], ctorParameters: () => [{ type: i1.AuthHttpService }, { type: i2.ActivatedRoute }] });
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJveHkuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL29yaWdpbi1mb3JtL3NyYy9saWIvc2VydmljZXMvcHJveHkuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7O0FBVzNDLE1BQU0sT0FBTyxZQUFZO0lBU3ZCLFlBQW9CLElBQXFCLEVBQVUsS0FBcUI7UUFBcEQsU0FBSSxHQUFKLElBQUksQ0FBaUI7UUFBVSxVQUFLLEdBQUwsS0FBSyxDQUFnQjtRQVBqRSxXQUFNLEdBQVcsMEJBQTBCLENBQUM7UUFFNUMsUUFBRyxHQUFXLEVBQUUsQ0FBQztRQUNoQixhQUFRLEdBQVcsRUFBRSxDQUFDO1FBQ3ZCLFVBQUssR0FBWSxLQUFLLENBQUM7UUFDdkIsWUFBTyxHQUFXLEVBQUUsQ0FBQztRQUcxQixJQUFJLENBQUMsS0FBSyxHQUFHLEVBQUUsQ0FBQztRQUNoQixJQUFJLENBQUMsS0FBSyxHQUFHLElBQUksQ0FBQyxHQUFHLElBQUksSUFBSSxJQUFJLElBQUksQ0FBQyxHQUFHLElBQUksTUFBTSxDQUFDO1FBQ3BELElBQUksQ0FBQyxRQUFRO1lBQ1gsSUFBSSxDQUFDLE1BQU07Z0JBQ1gsQ0FBQyxJQUFJLENBQUMsR0FBRyxJQUFJLE1BQU07b0JBQ2pCLENBQUMsQ0FBQyw2QkFBNkI7b0JBQy9CLENBQUMsQ0FBQyxpQ0FBaUMsQ0FBQyxDQUFDO0lBQzNDLENBQUM7SUFFRCxlQUFlO1FBQ2IsT0FBTztZQUNMLE1BQU0sRUFBRSxJQUFJLENBQUMsTUFBTTtZQUNuQixPQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU87WUFDckIsR0FBRyxFQUFFLElBQUksQ0FBQyxHQUFHO1lBQ2IsUUFBUSxFQUFFLElBQUksQ0FBQyxRQUFRO1lBQ3ZCLEtBQUssRUFBRSxJQUFJLENBQUMsS0FBSztTQUNsQixDQUFDO0lBQ0osQ0FBQztJQUVELFFBQVE7UUFDTixPQUFPLElBQUksT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxFQUFFO1lBQ3JDLElBQUksWUFBWSxHQUFHLElBQUksQ0FBQyxRQUFRLEdBQUcsaUJBQWlCLENBQUM7WUFDckQsSUFBSSxJQUFJLENBQUMsS0FBSztnQkFBRSxPQUFPLENBQUMsR0FBRyxDQUFDLG9CQUFvQixHQUFHLFlBQVksQ0FBQyxDQUFDO1lBQ2pFLElBQUksQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFnQixZQUFZLENBQUMsQ0FBQyxTQUFTLENBQUM7Z0JBQ25ELElBQUksRUFBRSxDQUFDLElBQUksRUFBRSxFQUFFO29CQUNiLElBQUksQ0FBQyxLQUFLLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQztvQkFDeEIsY0FBYyxDQUFDLE9BQU8sQ0FBQyxXQUFXLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO29CQUNoRCxJQUFJLElBQUksQ0FBQyxLQUFLO3dCQUNaLE9BQU8sQ0FBQyxHQUFHLENBQ1Qsc0JBQXNCLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUMsQ0FDaEUsQ0FBQztvQkFDSixPQUFPLENBQUMsRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUUsQ0FBQyxDQUFDO2dCQUN4QyxDQUFDO2dCQUNELEtBQUssRUFBRSxDQUFDLEtBQUssRUFBRSxFQUFFO29CQUNmLE9BQU8sQ0FBQyxHQUFHLENBQUMsc0JBQXNCLEdBQUcsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO29CQUM1RCxPQUFPLENBQUMsRUFBRSxPQUFPLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxLQUFLLEVBQUUsQ0FBQyxDQUFDO2dCQUM1QyxDQUFDO2dCQUNELFFBQVEsRUFBRSxHQUFHLEVBQUU7b0JBQ2IsT0FBTyxDQUFDLEdBQUcsQ0FBQyxtQkFBbUIsQ0FBQyxDQUFDO29CQUNqQyxPQUFPLENBQUMsRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFLEtBQUssRUFBRSxFQUFFLEVBQUUsQ0FBQyxDQUFDO2dCQUN4QyxDQUFDO2FBQ0YsQ0FBQyxDQUFDO1FBQ0wsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO0lBQ0QsS0FBSyxDQUFDLGVBQWU7UUFDbkIsSUFBSSxDQUFDO1lBQ0gsTUFBTSxTQUFTLEdBQUcsQ0FBQyxNQUFNLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBUSxDQUFDO1lBQ2pELElBQUksU0FBUyxDQUFDLE9BQU8sRUFBRSxDQUFDO2dCQUN0QixPQUFPLElBQUksQ0FBQyxLQUFLLENBQUM7WUFDcEIsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLE1BQU0sSUFBSSxLQUFLLENBQUMsd0JBQXdCLENBQUMsQ0FBQztZQUM1QyxDQUFDO1FBQ0gsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixPQUFPLENBQUMsS0FBSyxDQUFDLHlCQUF5QixFQUFFLEtBQUssQ0FBQyxDQUFDO1lBQ2hELE1BQU0sS0FBSyxDQUFDO1FBQ2QsQ0FBQztJQUNILENBQUM7K0dBbEVVLFlBQVk7bUhBQVosWUFBWSxjQUZYLE1BQU07OzRGQUVQLFlBQVk7a0JBSHhCLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQWN0aXZhdGVkUm91dGUgfSBmcm9tICdAYW5ndWxhci9yb3V0ZXInO1xuaW1wb3J0IHsgQXV0aEh0dHBTZXJ2aWNlIH0gZnJvbSAnLi9hdXRoLWh0dHAuc2VydmljZSc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVG9rZW5SZXNwb25zZSB7XG4gIHRva2VuOiBzdHJpbmc7XG59XG5cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnLFxufSlcbmV4cG9ydCBjbGFzcyBQcm94eVNlcnZpY2Uge1xuICBwdWJsaWMgb3JkZXJJRDogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICBwdWJsaWMgYXBpVVJMOiBzdHJpbmcgPSAnaHR0cHM6Ly9hcGlyby5pZC1reWMuY29tJztcbiAgcHVibGljIHRva2VuITogc3RyaW5nO1xuICBwdWJsaWMgZW52OiBzdHJpbmcgPSAnJztcbiAgcHJpdmF0ZSB0b2tlblVybDogc3RyaW5nID0gJyc7XG4gIHB1YmxpYyBkZWJ1ZzogYm9vbGVhbiA9IGZhbHNlO1xuICBwdWJsaWMgbGFuZ0lzbzogc3RyaW5nID0gJyc7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSBodHRwOiBBdXRoSHR0cFNlcnZpY2UsIHByaXZhdGUgcm91dGU6IEFjdGl2YXRlZFJvdXRlKSB7XG4gICAgdGhpcy50b2tlbiA9ICcnO1xuICAgIHRoaXMuZGVidWcgPSB0aGlzLmVudiA9PSAnUUEnIHx8IHRoaXMuZW52ID09ICdERU1PJztcbiAgICB0aGlzLnRva2VuVXJsID1cbiAgICAgIHRoaXMuYXBpVVJMICtcbiAgICAgICh0aGlzLmVudiA9PSAnUFJPRCdcbiAgICAgICAgPyAnL3B1YmxpYy9hdXRob3JpemF0aW9uL3Rva2VuJ1xuICAgICAgICA6ICcvZGV2X3B1YmxpYy9hdXRob3JpemF0aW9uL3Rva2VuJyk7XG4gIH1cblxuICBnZXRWYWx1ZXNGb3JMb2coKSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGFwaVVSTDogdGhpcy5hcGlVUkwsXG4gICAgICBvcmRlcklEOiB0aGlzLm9yZGVySUQsXG4gICAgICBlbnY6IHRoaXMuZW52LFxuICAgICAgdG9rZW5Vcmw6IHRoaXMudG9rZW5VcmwsXG4gICAgICB0b2tlbjogdGhpcy50b2tlbixcbiAgICB9O1xuICB9XG5cbiAgZ2V0VG9rZW4oKSB7XG4gICAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIHZhciB0b2tlblVybEZ1bGwgPSB0aGlzLnRva2VuVXJsICsgJ25ldz9mbG93PW9yaWdpbic7XG4gICAgICBpZiAodGhpcy5kZWJ1ZykgY29uc29sZS5sb2coJ0dldFRva2VuIGNhbGxlZCBhdCcgKyB0b2tlblVybEZ1bGwpO1xuICAgICAgdGhpcy5odHRwLmdldDxUb2tlblJlc3BvbnNlPih0b2tlblVybEZ1bGwpLnN1YnNjcmliZSh7XG4gICAgICAgIG5leHQ6IChyZXNwKSA9PiB7XG4gICAgICAgICAgdGhpcy50b2tlbiA9IHJlc3AudG9rZW47XG4gICAgICAgICAgc2Vzc2lvblN0b3JhZ2Uuc2V0SXRlbSgndG9rZW4ta3ljJywgdGhpcy50b2tlbik7XG4gICAgICAgICAgaWYgKHRoaXMuZGVidWcpXG4gICAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgICAgJ0dldFRva2VuIHJlc3BvbmRlZDogJyArIEpTT04uc3RyaW5naWZ5KHRoaXMuZ2V0VmFsdWVzRm9yTG9nKCkpXG4gICAgICAgICAgICApO1xuICAgICAgICAgIHJlc29sdmUoeyBzdWNjZXNzOiB0cnVlLCBlcnJvcjogJycgfSk7XG4gICAgICAgIH0sXG4gICAgICAgIGVycm9yOiAoZXJyb3IpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZygnR2V0VG9rZW4gcmVzcG9uZGVkOiAnICsgSlNPTi5zdHJpbmdpZnkoZXJyb3IpKTtcbiAgICAgICAgICByZXNvbHZlKHsgc3VjY2VzczogZmFsc2UsIGVycm9yOiBlcnJvciB9KTtcbiAgICAgICAgfSxcbiAgICAgICAgY29tcGxldGU6ICgpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZygnR2V0VG9rZW4gY29tcGxldGUnKTtcbiAgICAgICAgICByZXNvbHZlKHsgc3VjY2VzczogdHJ1ZSwgZXJyb3I6ICcnIH0pO1xuICAgICAgICB9LFxuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cbiAgYXN5bmMgZ2V0VG9rZW5BbmRTYXZlKCk6IFByb21pc2U8c3RyaW5nPiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHRva2VuUmVzcCA9IChhd2FpdCB0aGlzLmdldFRva2VuKCkpIGFzIGFueTtcbiAgICAgIGlmICh0b2tlblJlc3Auc3VjY2Vzcykge1xuICAgICAgICByZXR1cm4gdGhpcy50b2tlbjtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcignVG9rZW4gcmV0cmlldmFsIGZhaWxlZCcpO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBjb25zb2xlLmVycm9yKCdFcnJvciByZXRyaWV2aW5nIHRva2VuOicsIGVycm9yKTtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH1cbiAgfVxufVxuIl19
|