finary-community 0.1.0 → 0.1.1

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 CHANGED
@@ -1,229 +1,234 @@
1
- # Finary Community Library
2
-
3
- Une bibliothèque non-officielle et communautaire pour s'interfacer avec l'API Finary. Elle permet de récupérer vos données de patrimoine, transactions, insights et bien plus encore de manière programmatique.
4
-
5
- ## Installation
6
-
7
- ```bash
8
- npm install finary-community
9
- ```
10
-
11
- ## Configuration & Authentification
12
-
13
- L'authentification sur Finary étant complexe (protection anti-bot), cette librairie fonctionne actuellement en utilisant une session navigateur existante.
14
-
15
- Vous devez créer un fichier `credentials.json` à la racine de votre projet.
16
- Vous pouvez le générer automatiquement en lançant la commande interactive :
17
-
18
- ```bash
19
- npm run setup
20
- ```
21
-
22
- Ou le créer manuellement :
23
-
24
- ### Structure du fichier `credentials.json`
25
-
26
- ```json
27
- {
28
- "token": "JWT_TOKEN_ICI",
29
- "clerkSession": "CLERK_SESSION_ID_ICI",
30
- "headers": {
31
- "user-agent": "Mozilla/5.0 ...",
32
- "cookie": "..."
33
- }
34
- }
35
- ```
36
-
37
- > **Note :** Le `clerkSession` et les `headers` (cookie) sont essentiels pour permettre le rafraîchissement automatique du token.
38
-
39
- ## API Reference
40
-
41
- Voici la documentation détaillée de chaque fonction disponible dans la librairie.
42
-
43
- ### Types et Enums Communs
44
-
45
- Plusieurs fonctions utilisent les énumérations suivantes pour filtrer les données :
46
-
47
- * **`FinaryPeriod`** : Période temporelle pour les graphiques et variations.
48
- * `'1d'` (24h), `'1w'` (Semaine), `'1m'` (Mois), `'ytd'` (Depuis début d'année), `'1y'` (1 an), `'all'` (Tout).
49
- * **`TimeseriesType`** : Type de série temporelle.
50
- * `'sum'` (Valeur cumulée), `'all'` (Détail).
51
- * **`DistributionType`** : Type de répartition.
52
- * `'account'` (Par compte).
53
- * **`InvestmentDistributionType`** : Type de répartition investissement.
54
- * `'stock'` (Par action/actif).
55
-
56
- ---
57
-
58
- ### 1. Client Principal (`FinaryClient`)
59
-
60
- Point d'entrée de la librairie.
61
-
62
- #### `constructor(config)`
63
- Initialise le client.
64
- * **Paramètres** :
65
- * `config` (`AuthConfig`) : Objet de configuration `{ credentialsPath: string }`.
66
-
67
- #### `getUserProfile()`
68
- Récupère les informations du profil utilisateur connecté.
69
- * **Retourne** : `Promise<UserProfile>` (Nom, email, paramètres, etc.)
70
-
71
- #### `getOrganizations()`
72
- Liste les organisations (souvent appelées "familles" dans l'UI) associées à l'utilisateur. Vous aurez besoin de l'ID de l'organisation et du membre pour la plupart des autres appels.
73
- * **Retourne** : `Promise<Organization[]>`
74
-
75
- ---
76
-
77
- ### 2. Investissements (`client.investments`)
78
-
79
- Tous les appels nécessitent généralement `organizationId` et `membershipId`.
80
-
81
- #### `getPortfolio(organizationId, membershipId, period)`
82
- Vue d'ensemble du portefeuille d'investissements (Actions, Crypto, Immo, etc.).
83
- * **Paramètres** :
84
- * `organizationId` (`string`) : ID de l'organisation.
85
- * `membershipId` (`string`) : ID du membre.
86
- * `period` (`FinaryPeriod`, défaut: `YTD`) : Période de calcul des plus/moins-values.
87
- * **Retourne** : `Promise<InvestmentPortfolio>`
88
-
89
- #### `getTimeseries(organizationId, membershipId, period, type)`
90
- Récupère l'historique de la valeur du portefeuille (graphique).
91
- * **Paramètres** :
92
- * `period` (`FinaryPeriod`, défaut: `YTD`).
93
- * `type` (`TimeseriesType`, défaut: `SUM`).
94
- * **Retourne** : `Promise<Timeseries[]>`
95
-
96
- #### `getDistribution(organizationId, membershipId, type, period)`
97
- Répartition des actifs.
98
- * **Paramètres** :
99
- * `type` (`InvestmentDistributionType`, défaut: `STOCK`).
100
- * **Retourne** : `Promise<Distribution>`
101
-
102
- #### `getDividends(organizationId, membershipId, withRealEstate)`
103
- Liste les dividendes perçus et à venir.
104
- * **Paramètres** :
105
- * `withRealEstate` (`boolean`, défaut: `true`) : Inclure les loyers/dividendes immobiliers (SCPI).
106
- * **Retourne** : `Promise<DividendsResponse>`
107
-
108
- #### `getSectorAllocation(organizationId, membershipId)`
109
- Répartition sectorielle du portefeuille (Technologie, Santé, Finance...).
110
- * **Retourne** : `Promise<SectorAllocationResponse>`
111
-
112
- #### `getGeographicalAllocation(organizationId, membershipId)`
113
- Répartition géographique du portefeuille (USA, Europe, Asie...).
114
- * **Retourne** : `Promise<GeographicalAllocationResponse>`
115
-
116
- #### `getFees(organizationId, membershipId)`
117
- Analyse des frais sur les enveloppes (PEA, CTO, AV...).
118
- * **Retourne** : `Promise<FeesResponse>`
119
-
120
- #### `getAccount(organizationId, membershipId, accountId, period)`
121
- Détails d'un compte spécifique (ex: un PEA particulier).
122
- * **Paramètres** :
123
- * `accountId` (`string`) : ID du compte.
124
- * **Retourne** : `Promise<InvestmentAccount>`
125
-
126
- #### Méthodes "Account" Spécifiques
127
- De la même manière que pour le portefeuille global, vous pouvez appeler ces méthodes pour un compte précis :
128
- * `getAccountTimeseries(orgId, memId, accountId, period, type)`
129
- * `getAccountDistribution(orgId, memId, accountId, type, period)`
130
- * `getAccountDividends(orgId, memId, accountId, withRealEstate)`
131
- * `getAccountSectorAllocation(orgId, memId, accountId)`
132
- * `getAccountGeographicalAllocation(orgId, memId, accountId)`
133
- * `getAccountFees(orgId, memId, accountId)`
134
-
135
- ---
136
-
137
- ### 3. Épargne (`client.savings`)
138
-
139
- Concerne les livrets bancaires, fonds euros, etc.
140
-
141
- #### `getPortfolio(organizationId, membershipId, period)`
142
- Vue d'ensemble de l'épargne.
143
- * **Retourne** : `Promise<SavingsPortfolio>`
144
-
145
- #### `getAccounts(organizationId, membershipId, period)`
146
- Liste tous les comptes d'épargne.
147
- * **Retourne** : `Promise<SavingsAccount[]>`
148
-
149
- #### `getAccount(organizationId, membershipId, accountId, period)`
150
- Détails d'un compte épargne spécifique.
151
- * **Retourne** : `Promise<SavingsAccount>`
152
-
153
- #### `getTransaction(organizationId, membershipId, page, perPage, query, accountId)`
154
- Récupère les transactions (virements, intérêts).
155
- * **Paramètres** :
156
- * `page` (`number`, défaut: `1`) : Numéro de page.
157
- * `perPage` (`number`, défaut: `50`) : Nombre d'éléments par page.
158
- * `query` (`string`, optionnel) : Recherche textuelle.
159
- * `accountId` (`string`, optionnel) : Filtrer par compte spécifique.
160
- * **Retourne** : `Promise<Transaction[]>`
161
-
162
- #### `getTimeseries(...)` et `getAccountTimeseries(...)`
163
- Historique de valeur.
164
-
165
- ---
166
-
167
- ### 4. Comptes Courants (`client.checkings`)
168
-
169
- #### `getPortfolio(organizationId, membershipId, period)`
170
- Solde total des comptes courants.
171
- * **Retourne** : `Promise<CheckingPortfolio>`
172
-
173
- #### `getAccounts(organizationId, membershipId, period)`
174
- Liste des comptes bancaires.
175
- * **Retourne** : `Promise<CheckingAccount[]>`
176
-
177
- #### `getTransactions(...)`
178
- Récupère les transactions bancaires. Fonctionne identiquement à celle de l'épargne.
179
-
180
- ---
181
-
182
- ### 5. Benchmarks (`client.benchmarks`)
183
-
184
- #### `getAvailableAssets(organizationId, membershipId, period)`
185
- Récupère la liste des actifs disponibles pour faire des comparaisons de performance.
186
- * **Retourne** : `Promise<BenchmarkAsset[]>`
187
-
188
- ## Exemple Complet
189
-
190
- ```typescript
191
- import { FinaryClient, FinaryPeriod } from 'finary-community';
192
-
193
- const client = new FinaryClient({ credentialsPath: './credentials.json' });
194
-
195
- async function main() {
196
- // 1. Setup
197
- const orgs = await client.getOrganizations();
198
- const orgId = orgs[0].id;
199
- const memberId = orgs[0].members[0].id;
200
-
201
- console.log(`Utilisateur : ${orgs[0].name}`);
202
-
203
- // 2. Investissements
204
- const investments = await client.investments.getPortfolio(orgId, memberId, FinaryPeriod.YTD);
205
- console.log(`--- Investissements ---`);
206
- console.log(`Total : ${investments.total.amount} €`);
207
- console.log(`Plus-value latente : ${investments.total.unrealized_pnl} €`);
208
-
209
- // 3. Dividendes à venir
210
- const dividends = await client.investments.getDividends(orgId, memberId);
211
- if(dividends.upcoming_dividends.length > 0) {
212
- console.log(`--- Prochains Dividendes ---`);
213
- dividends.upcoming_dividends.forEach(div => {
214
- console.log(`${div.date} : ${div.amount}€ (${div.asset.name})`);
215
- });
216
- }
217
-
218
- // 4. Recherche de transactions Livret A
219
- const savings = await client.savings.getAccounts(orgId, memberId);
220
- const livretA = savings.find(s => s.name.includes("Livret A"));
221
- if (livretA) {
222
- console.log(`--- Transactions Livret A ---`);
223
- const txs = await client.savings.getTransactions(orgId, memberId, 1, 5, '', livretA.id);
224
- txs.forEach(t => console.log(`${t.date} : ${t.amount}€ - ${t.description}`));
225
- }
226
- }
227
-
228
- main().catch(console.error);
229
- ```
1
+ # Finary Community Library
2
+
3
+ Une bibliothèque non-officielle et communautaire pour s'interfacer avec l'API Finary. Elle permet de récupérer vos données de patrimoine, transactions, insights et bien plus encore de manière programmatique.
4
+
5
+ ## Installation
6
+
7
+ ```bash
8
+ npm install finary-community
9
+ ```
10
+
11
+ ## Configuration & Authentification
12
+
13
+ L'authentification sur Finary étant complexe (protection anti-bot), cette librairie fonctionne en utilisant une session navigateur existante qu'elle capture pour vous.
14
+
15
+ Vous devez générer un fichier `credentials.json` qui permettra à votre code de se connecter.
16
+
17
+ ### Méthode Recommandée (Assistant Interactif)
18
+
19
+ Si vous avez installé la librairie via NPM dans votre projet, executez simplement cette commande dans votre terminal (**à la racine où se trouve votre `node_modules`**) :
20
+
21
+ ```bash
22
+ npx finary-community setup
23
+ ```
24
+
25
+ Cela ouvrira une fenêtre de navigateur contrôlée. Connectez-vous simplement à Finary, et le script capturera automatiquement vos tokens et cookies pour générer le fichier `credentials.json`.
26
+
27
+ ### Méthode Manuelle
28
+
29
+ Vous pouvez créer manuellement un fichier `credentials.json` à la racine de votre projet avec la structure suivante :
30
+
31
+ ```json
32
+ {
33
+ "token": "VOTRE_JWT_TOKEN",
34
+ "clerkSession": "VOTRE_CLERK_SESSION_ID",
35
+ "headers": {
36
+ "user-agent": "Mozilla/5.0 ...",
37
+ "cookie": "..."
38
+ }
39
+ }
40
+ ```
41
+
42
+ > **Important :** Le `clerkSession` (cookie `__session`) et les `headers` (notamment le `User-Agent` et `Cookie`) sont **obligatoires** pour permettre à la librairie de rafraîchir le token automatiquement lorsqu'il expire. Sans cela, le script cessera de fonctionner après 20 minutes.
43
+
44
+ ## API Reference
45
+
46
+ Voici la documentation détaillée de chaque fonction disponible dans la librairie.
47
+
48
+ ### Types et Enums Communs
49
+
50
+ Plusieurs fonctions utilisent les énumérations suivantes pour filtrer les données :
51
+
52
+ * **`FinaryPeriod`** : Période temporelle pour les graphiques et variations.
53
+ * `'1d'` (24h), `'1w'` (Semaine), `'1m'` (Mois), `'ytd'` (Depuis début d'année), `'1y'` (1 an), `'all'` (Tout).
54
+ * **`TimeseriesType`** : Type de série temporelle.
55
+ * `'sum'` (Valeur cumulée), `'all'` (Détail).
56
+ * **`DistributionType`** : Type de répartition.
57
+ * `'account'` (Par compte).
58
+ * **`InvestmentDistributionType`** : Type de répartition investissement.
59
+ * `'stock'` (Par action/actif).
60
+
61
+ ---
62
+
63
+ ### 1. Client Principal (`FinaryClient`)
64
+
65
+ Point d'entrée de la librairie.
66
+
67
+ #### `constructor(config)`
68
+ Initialise le client.
69
+ * **Paramètres** :
70
+ * `config` (`AuthConfig`) : Objet de configuration `{ credentialsPath: string }`.
71
+
72
+ #### `getUserProfile()`
73
+ Récupère les informations du profil utilisateur connecté.
74
+ * **Retourne** : `Promise<UserProfile>` (Nom, email, paramètres, etc.)
75
+
76
+ #### `getOrganizations()`
77
+ Liste les organisations (souvent appelées "familles" dans l'UI) associées à l'utilisateur. Vous aurez besoin de l'ID de l'organisation et du membre pour la plupart des autres appels.
78
+ * **Retourne** : `Promise<Organization[]>`
79
+
80
+ ---
81
+
82
+ ### 2. Investissements (`client.investments`)
83
+
84
+ Tous les appels nécessitent généralement `organizationId` et `membershipId`.
85
+
86
+ #### `getPortfolio(organizationId, membershipId, period)`
87
+ Vue d'ensemble du portefeuille d'investissements (Actions, Crypto, Immo, etc.).
88
+ * **Paramètres** :
89
+ * `organizationId` (`string`) : ID de l'organisation.
90
+ * `membershipId` (`string`) : ID du membre.
91
+ * `period` (`FinaryPeriod`, défaut: `YTD`) : Période de calcul des plus/moins-values.
92
+ * **Retourne** : `Promise<InvestmentPortfolio>`
93
+
94
+ #### `getTimeseries(organizationId, membershipId, period, type)`
95
+ Récupère l'historique de la valeur du portefeuille (graphique).
96
+ * **Paramètres** :
97
+ * `period` (`FinaryPeriod`, défaut: `YTD`).
98
+ * `type` (`TimeseriesType`, défaut: `SUM`).
99
+ * **Retourne** : `Promise<Timeseries[]>`
100
+
101
+ #### `getDistribution(organizationId, membershipId, type, period)`
102
+ Répartition des actifs.
103
+ * **Paramètres** :
104
+ * `type` (`InvestmentDistributionType`, défaut: `STOCK`).
105
+ * **Retourne** : `Promise<Distribution>`
106
+
107
+ #### `getDividends(organizationId, membershipId, withRealEstate)`
108
+ Liste les dividendes perçus et à venir.
109
+ * **Paramètres** :
110
+ * `withRealEstate` (`boolean`, défaut: `true`) : Inclure les loyers/dividendes immobiliers (SCPI).
111
+ * **Retourne** : `Promise<DividendsResponse>`
112
+
113
+ #### `getSectorAllocation(organizationId, membershipId)`
114
+ Répartition sectorielle du portefeuille (Technologie, Santé, Finance...).
115
+ * **Retourne** : `Promise<SectorAllocationResponse>`
116
+
117
+ #### `getGeographicalAllocation(organizationId, membershipId)`
118
+ Répartition géographique du portefeuille (USA, Europe, Asie...).
119
+ * **Retourne** : `Promise<GeographicalAllocationResponse>`
120
+
121
+ #### `getFees(organizationId, membershipId)`
122
+ Analyse des frais sur les enveloppes (PEA, CTO, AV...).
123
+ * **Retourne** : `Promise<FeesResponse>`
124
+
125
+ #### `getAccount(organizationId, membershipId, accountId, period)`
126
+ Détails d'un compte spécifique (ex: un PEA particulier).
127
+ * **Paramètres** :
128
+ * `accountId` (`string`) : ID du compte.
129
+ * **Retourne** : `Promise<InvestmentAccount>`
130
+
131
+ #### Méthodes "Account" Spécifiques
132
+ De la même manière que pour le portefeuille global, vous pouvez appeler ces méthodes pour un compte précis :
133
+ * `getAccountTimeseries(orgId, memId, accountId, period, type)`
134
+ * `getAccountDistribution(orgId, memId, accountId, type, period)`
135
+ * `getAccountDividends(orgId, memId, accountId, withRealEstate)`
136
+ * `getAccountSectorAllocation(orgId, memId, accountId)`
137
+ * `getAccountGeographicalAllocation(orgId, memId, accountId)`
138
+ * `getAccountFees(orgId, memId, accountId)`
139
+
140
+ ---
141
+
142
+ ### 3. Épargne (`client.savings`)
143
+
144
+ Concerne les livrets bancaires, fonds euros, etc.
145
+
146
+ #### `getPortfolio(organizationId, membershipId, period)`
147
+ Vue d'ensemble de l'épargne.
148
+ * **Retourne** : `Promise<SavingsPortfolio>`
149
+
150
+ #### `getAccounts(organizationId, membershipId, period)`
151
+ Liste tous les comptes d'épargne.
152
+ * **Retourne** : `Promise<SavingsAccount[]>`
153
+
154
+ #### `getAccount(organizationId, membershipId, accountId, period)`
155
+ Détails d'un compte épargne spécifique.
156
+ * **Retourne** : `Promise<SavingsAccount>`
157
+
158
+ #### `getTransaction(organizationId, membershipId, page, perPage, query, accountId)`
159
+ Récupère les transactions (virements, intérêts).
160
+ * **Paramètres** :
161
+ * `page` (`number`, défaut: `1`) : Numéro de page.
162
+ * `perPage` (`number`, défaut: `50`) : Nombre d'éléments par page.
163
+ * `query` (`string`, optionnel) : Recherche textuelle.
164
+ * `accountId` (`string`, optionnel) : Filtrer par compte spécifique.
165
+ * **Retourne** : `Promise<Transaction[]>`
166
+
167
+ #### `getTimeseries(...)` et `getAccountTimeseries(...)`
168
+ Historique de valeur.
169
+
170
+ ---
171
+
172
+ ### 4. Comptes Courants (`client.checkings`)
173
+
174
+ #### `getPortfolio(organizationId, membershipId, period)`
175
+ Solde total des comptes courants.
176
+ * **Retourne** : `Promise<CheckingPortfolio>`
177
+
178
+ #### `getAccounts(organizationId, membershipId, period)`
179
+ Liste des comptes bancaires.
180
+ * **Retourne** : `Promise<CheckingAccount[]>`
181
+
182
+ #### `getTransactions(...)`
183
+ Récupère les transactions bancaires. Fonctionne identiquement à celle de l'épargne.
184
+
185
+ ---
186
+
187
+ ### 5. Benchmarks (`client.benchmarks`)
188
+
189
+ #### `getAvailableAssets(organizationId, membershipId, period)`
190
+ Récupère la liste des actifs disponibles pour faire des comparaisons de performance.
191
+ * **Retourne** : `Promise<BenchmarkAsset[]>`
192
+
193
+ ## Exemple Complet
194
+
195
+ ```typescript
196
+ import { FinaryClient, FinaryPeriod } from 'finary-community';
197
+
198
+ const client = new FinaryClient({ credentialsPath: './credentials.json' });
199
+
200
+ async function main() {
201
+ // 1. Setup
202
+ const orgs = await client.getOrganizations();
203
+ const orgId = orgs[0].id;
204
+ const memberId = orgs[0].members[0].id;
205
+
206
+ console.log(`Utilisateur : ${orgs[0].name}`);
207
+
208
+ // 2. Investissements
209
+ const investments = await client.investments.getPortfolio(orgId, memberId, FinaryPeriod.YTD);
210
+ console.log(`--- Investissements ---`);
211
+ console.log(`Total : ${investments.total.amount} €`);
212
+ console.log(`Plus-value latente : ${investments.total.unrealized_pnl} €`);
213
+
214
+ // 3. Dividendes à venir
215
+ const dividends = await client.investments.getDividends(orgId, memberId);
216
+ if(dividends.upcoming_dividends.length > 0) {
217
+ console.log(`--- Prochains Dividendes ---`);
218
+ dividends.upcoming_dividends.forEach(div => {
219
+ console.log(`${div.date} : ${div.amount}€ (${div.asset.name})`);
220
+ });
221
+ }
222
+
223
+ // 4. Recherche de transactions Livret A
224
+ const savings = await client.savings.getAccounts(orgId, memberId);
225
+ const livretA = savings.find(s => s.name.includes("Livret A"));
226
+ if (livretA) {
227
+ console.log(`--- Transactions Livret A ---`);
228
+ const txs = await client.savings.getTransactions(orgId, memberId, 1, 5, '', livretA.id);
229
+ txs.forEach(t => console.log(`${t.date} : ${t.amount}€ - ${t.description}`));
230
+ }
231
+ }
232
+
233
+ main().catch(console.error);
234
+ ```
@@ -117,4 +117,4 @@ class AuthManager {
117
117
  }
118
118
  }
119
119
  exports.AuthManager = AuthManager;
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth-manager.js","sourceRoot":"","sources":["../../src/auth/auth-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import { readFileSync, existsSync } from \"node:fs\";\r\nimport { AuthConfig, Credentials } from \"../interfaces\";\r\n\r\nexport class AuthManager {\r\n    private token: string | null = null;\r\n    private credentialsPath: string;\r\n    private cachedCredentials: Credentials | null = null;\r\n\r\n    constructor(config: AuthConfig) {\r\n        this.credentialsPath = config.credentialsPath;\r\n    }\r\n\r\n    private loadCredentials(): Credentials {\r\n        if (this.cachedCredentials) return this.cachedCredentials;\r\n\r\n        if (existsSync(this.credentialsPath)) {\r\n            try {\r\n                const data = readFileSync(this.credentialsPath, \"utf-8\");\r\n                this.cachedCredentials = JSON.parse(data) as Credentials;\r\n                return this.cachedCredentials;\r\n            } catch (e) {\r\n                console.error(`Failed to parse credentials at ${this.credentialsPath}: ${e}`);\r\n            }\r\n        }\r\n\r\n        throw new Error(`credentials.json not found at ${this.credentialsPath}`);\r\n    }\r\n\r\n    public async getToken(): Promise<string> {\r\n        if (this.token) return this.token;\r\n\r\n        const creds = this.loadCredentials();\r\n        if (creds.token) {\r\n            this.token = creds.token;\r\n            return this.token;\r\n        }\r\n\r\n        throw new Error(\"No token found in credentials.\");\r\n    }\r\n\r\n    public getCredentials(): Credentials {\r\n        return this.loadCredentials();\r\n    }\r\n\r\n    public async refreshToken(): Promise<string> {\r\n        const creds = this.loadCredentials();\r\n\r\n        // 1. Clerk Refresh\r\n        if (creds.clerkSession && this.token) {\r\n            try {\r\n                const decoded = this.parseJwt(this.token);\r\n                const sessionId = decoded ? decoded.sid : null;\r\n\r\n                if (sessionId) {\r\n                    const clerkUrl = `https://clerk.finary.com/v1/client/sessions/${sessionId}/tokens?__clerk_api_version=2025-11-10&_clerk_js_version=5.117.0`;\r\n                    const cookieHeader = creds.cookieHeader || `__session=${creds.clerkSession}`;\r\n\r\n                    const response = await fetch(clerkUrl, {\r\n                        method: \"POST\",\r\n                        headers: {\r\n                            \"Cookie\": cookieHeader,\r\n                            \"Content-Type\": \"application/x-www-form-urlencoded\",\r\n                            \"Origin\": \"https://app.finary.com\",\r\n                            \"Referer\": \"https://app.finary.com/\",\r\n                            \"User-Agent\": creds.headers?.['user-agent'] || \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/143.0.0.0 Safari/537.36\"\r\n                        },\r\n                        body: \"organization_id=\"\r\n                    });\r\n\r\n                    if (response.ok) {\r\n                        const data = (await response.json()) as any;\r\n                        if (data && data.jwt) {\r\n                            this.token = data.jwt;\r\n                            return this.token as string;\r\n                        }\r\n                    } else {\r\n                        console.warn(`Clerk refresh failed: ${response.status}`);\r\n                    }\r\n                }\r\n            } catch (e) {\r\n                console.error(`Clerk refresh error: ${e}`);\r\n            }\r\n        }\r\n\r\n        // 2. Legacy Refresh\r\n        if (!creds.refreshUrl || !creds.headers) {\r\n            throw new Error(\"Missing info to refresh token (legacy).\");\r\n        }\r\n\r\n        try {\r\n            const headers: Record<string, string> = { ...creds.headers };\r\n            // Cleanup headers manually since we can't use arbitrary types easily on headers object\r\n            delete headers['content-length'];\r\n            delete headers['host'];\r\n            delete headers['connection'];\r\n            delete headers['origin'];\r\n            delete headers['referer'];\r\n            delete headers['cookie'];\r\n\r\n            Object.keys(headers).forEach(key => { if (key.startsWith(':')) delete headers[key]; });\r\n\r\n            const response = await fetch(creds.refreshUrl, {\r\n                method: \"POST\",\r\n                headers: headers,\r\n            });\r\n\r\n            if (!response.ok) {\r\n                throw new Error(`Legacy refresh failed: ${response.status}`);\r\n            }\r\n\r\n            const data = (await response.json()) as any;\r\n            if (data && data.jwt) {\r\n                this.token = data.jwt;\r\n                return this.token as string;\r\n            }\r\n            throw new Error(\"No JWT in refresh response\");\r\n\r\n        } catch (e) {\r\n            console.error(`Token Refresh Error: ${e}`);\r\n            throw e;\r\n        }\r\n    }\r\n\r\n    private parseJwt(token: string): any {\r\n        try {\r\n            return JSON.parse(Buffer.from(token.split('.')[1], 'base64').toString());\r\n        } catch {\r\n            return null;\r\n        }\r\n    }\r\n}\r\n"]}
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth-manager.js","sourceRoot":"","sources":["../../src/auth/auth-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import { readFileSync, existsSync } from \"node:fs\";\nimport { AuthConfig, Credentials } from \"../interfaces\";\n\nexport class AuthManager {\n    private token: string | null = null;\n    private credentialsPath: string;\n    private cachedCredentials: Credentials | null = null;\n\n    constructor(config: AuthConfig) {\n        this.credentialsPath = config.credentialsPath;\n    }\n\n    private loadCredentials(): Credentials {\n        if (this.cachedCredentials) return this.cachedCredentials;\n\n        if (existsSync(this.credentialsPath)) {\n            try {\n                const data = readFileSync(this.credentialsPath, \"utf-8\");\n                this.cachedCredentials = JSON.parse(data) as Credentials;\n                return this.cachedCredentials;\n            } catch (e) {\n                console.error(`Failed to parse credentials at ${this.credentialsPath}: ${e}`);\n            }\n        }\n\n        throw new Error(`credentials.json not found at ${this.credentialsPath}`);\n    }\n\n    public async getToken(): Promise<string> {\n        if (this.token) return this.token;\n\n        const creds = this.loadCredentials();\n        if (creds.token) {\n            this.token = creds.token;\n            return this.token;\n        }\n\n        throw new Error(\"No token found in credentials.\");\n    }\n\n    public getCredentials(): Credentials {\n        return this.loadCredentials();\n    }\n\n    public async refreshToken(): Promise<string> {\n        const creds = this.loadCredentials();\n\n        // 1. Clerk Refresh\n        if (creds.clerkSession && this.token) {\n            try {\n                const decoded = this.parseJwt(this.token);\n                const sessionId = decoded ? decoded.sid : null;\n\n                if (sessionId) {\n                    const clerkUrl = `https://clerk.finary.com/v1/client/sessions/${sessionId}/tokens?__clerk_api_version=2025-11-10&_clerk_js_version=5.117.0`;\n                    const cookieHeader = creds.cookieHeader || `__session=${creds.clerkSession}`;\n\n                    const response = await fetch(clerkUrl, {\n                        method: \"POST\",\n                        headers: {\n                            \"Cookie\": cookieHeader,\n                            \"Content-Type\": \"application/x-www-form-urlencoded\",\n                            \"Origin\": \"https://app.finary.com\",\n                            \"Referer\": \"https://app.finary.com/\",\n                            \"User-Agent\": creds.headers?.['user-agent'] || \"Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/143.0.0.0 Safari/537.36\"\n                        },\n                        body: \"organization_id=\"\n                    });\n\n                    if (response.ok) {\n                        const data = (await response.json()) as any;\n                        if (data && data.jwt) {\n                            this.token = data.jwt;\n                            return this.token as string;\n                        }\n                    } else {\n                        console.warn(`Clerk refresh failed: ${response.status}`);\n                    }\n                }\n            } catch (e) {\n                console.error(`Clerk refresh error: ${e}`);\n            }\n        }\n\n        // 2. Legacy Refresh\n        if (!creds.refreshUrl || !creds.headers) {\n            throw new Error(\"Missing info to refresh token (legacy).\");\n        }\n\n        try {\n            const headers: Record<string, string> = { ...creds.headers };\n            // Cleanup headers manually since we can't use arbitrary types easily on headers object\n            delete headers['content-length'];\n            delete headers['host'];\n            delete headers['connection'];\n            delete headers['origin'];\n            delete headers['referer'];\n            delete headers['cookie'];\n\n            Object.keys(headers).forEach(key => { if (key.startsWith(':')) delete headers[key]; });\n\n            const response = await fetch(creds.refreshUrl, {\n                method: \"POST\",\n                headers: headers,\n            });\n\n            if (!response.ok) {\n                throw new Error(`Legacy refresh failed: ${response.status}`);\n            }\n\n            const data = (await response.json()) as any;\n            if (data && data.jwt) {\n                this.token = data.jwt;\n                return this.token as string;\n            }\n            throw new Error(\"No JWT in refresh response\");\n\n        } catch (e) {\n            console.error(`Token Refresh Error: ${e}`);\n            throw e;\n        }\n    }\n\n    private parseJwt(token: string): any {\n        try {\n            return JSON.parse(Buffer.from(token.split('.')[1], 'base64').toString());\n        } catch {\n            return null;\n        }\n    }\n}\n"]}
@@ -160,4 +160,4 @@ async function createSession(options = {}) {
160
160
  await browser.close();
161
161
  }
162
162
  }
163
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"login-helper.js","sourceRoot":"","sources":["../../src/auth/login-helper.ts"],"names":[],"mappings":"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","sourcesContent":["import puppeteer from 'puppeteer';\r\nimport fs from 'node:fs';\r\nimport path from 'node:path';\r\n\r\nexport interface CreateSessionOptions {\r\n    headless?: boolean;\r\n    outputPath?: string;\r\n}\r\n\r\nexport async function createSession(options: CreateSessionOptions = {}) {\r\n    const { headless = false, outputPath } = options;\r\n\r\n    console.log(\"------------------------------------------\");\r\n    console.log(\"   Finary Setup - Browser Login\");\r\n    console.log(\"------------------------------------------\");\r\n\r\n    if (outputPath) {\r\n        // Ensure directory exists\r\n        const dir = path.dirname(outputPath);\r\n        if (!fs.existsSync(dir)) {\r\n            try {\r\n                fs.mkdirSync(dir, { recursive: true });\r\n            } catch (e: any) {\r\n                throw new Error(`Error creating directory ${dir}: ${e.message}`);\r\n            }\r\n        }\r\n        console.log(`Will save to: ${outputPath}`);\r\n    }\r\n\r\n    console.log(\"Launching browser for authentication...\");\r\n\r\n    let browser;\r\n    try {\r\n        browser = await puppeteer.launch({\r\n            headless: headless,\r\n            defaultViewport: null,\r\n            args: ['--start-maximized'] // Find browser window easier\r\n        });\r\n    } catch (e: any) {\r\n        console.error(\"Failed to launch browser. Make sure you installed puppeteer: npm install puppeteer\");\r\n        throw e;\r\n    }\r\n\r\n    try {\r\n        const page = await browser.newPage();\r\n\r\n        await page.setRequestInterception(false); // Ensure interception does not block unless intended\r\n        page.on('request', request => {\r\n            const url = request.url();\r\n            if (url.includes('clerk.finary.com')) {\r\n                // Optional: log or handle clerk requests\r\n            }\r\n        });\r\n\r\n        console.log(\"Navigating to Finary login page...\");\r\n        await page.goto('https://app.finary.com/v2', { waitUntil: 'domcontentloaded' });\r\n\r\n        console.log(\"\\nACTION REQUIRED: Please log in to your Finary account in the opened browser window.\\n\");\r\n        console.log(\"Scanning network traffic for authentication token...\");\r\n\r\n        // Promise that resolves when the valid token is found\r\n        const waitForToken = new Promise<{ token: string, headers: Record<string, string> }>((resolve) => {\r\n            page.on('response', response => {\r\n                const request = response.request();\r\n                const url = request.url();\r\n\r\n                // Filter API requests\r\n                if (url.includes('api.finary.com') || url.includes('clerk.finary.com')) {\r\n                    // We only want successful requests to ensure the token is valid\r\n                    if (response.status() === 200) {\r\n                        const headers = request.headers();\r\n                        const auth = headers['authorization'];\r\n\r\n                        if (auth && auth.startsWith('Bearer ey')) {\r\n                            const token = auth.replace('Bearer ', '');\r\n                            console.log(\"✅ Valid token successfully detected from successful request!\");\r\n                            resolve({ token, headers });\r\n                        }\r\n                    }\r\n                }\r\n            });\r\n        });\r\n\r\n        const { token, headers } = await waitForToken;\r\n\r\n        console.log(\"Token detected! capturing cookies...\");\r\n        // Capture cookies for both the main app and the clerk subdomain to ensure we get __client, __session, etc.\r\n        const cookies = await page.cookies('https://app.finary.com', 'https://clerk.finary.com');\r\n        const cookieHeader = cookies.map(c => `${c.name}=${c.value}`).join('; ');\r\n\r\n        const sessionCookie = cookies.find(c => c.name === '__session');\r\n        const clerkSession = sessionCookie ? sessionCookie.value : null;\r\n\r\n        if (cookieHeader) {\r\n            console.log(\"✅ Cookies captured!\");\r\n        } else {\r\n            console.warn(\"⚠️ No cookies found. Refresh verify might fail.\");\r\n        }\r\n\r\n        console.log(\"Fetching user details...\");\r\n\r\n        // Use the token to identify the user\r\n        let email = \"unknown_user@example.com\";\r\n        try {\r\n            const meResp = await fetch(\"https://api.finary.com/users/me\", {\r\n                headers: {\r\n                    \"Authorization\": `Bearer ${token}`,\r\n                    \"User-Agent\": \"Mozilla/5.0\"\r\n                }\r\n            });\r\n\r\n            if (meResp.ok) {\r\n                const data = await meResp.json();\r\n                if (data.result && data.result.email) {\r\n                    email = data.result.email;\r\n                    console.log(`Identified as: ${email}`);\r\n                }\r\n            }\r\n        } catch (e: any) {\r\n            console.warn(\"Could not fetch user email (minor issue), proceeding anyway.\");\r\n        }\r\n\r\n        // Filter headers based on user request (\"only these specific headers\")\r\n        const allowedHeaderKeys = [\r\n            \"sec-ch-ua-platform\",\r\n            \"x-client-api-version\",\r\n            \"referer\",\r\n            \"accept-language\",\r\n            \"sec-ch-ua\",\r\n            \"sec-ch-ua-mobile\",\r\n            \"user-agent\",\r\n            \"x-finary-client-id\",\r\n            \":authority\",\r\n            \":method\",\r\n            \":scheme\",\r\n            \"accept\",\r\n            \"accept-encoding\",\r\n            \"origin\",\r\n            \"priority\",\r\n            \"sec-fetch-dest\",\r\n            \"sec-fetch-mode\",\r\n            \"sec-fetch-site\"\r\n        ];\r\n\r\n        const filteredHeaders: Record<string, string> = {};\r\n        for (const key of allowedHeaderKeys) {\r\n            if (headers[key]) {\r\n                filteredHeaders[key] = headers[key];\r\n            }\r\n        }\r\n\r\n        const credentialsData = {\r\n            token: token,\r\n            refreshUrl: \"https://clerk.finary.com/v1/client/sessions\",\r\n            headers: filteredHeaders,\r\n            clerkSession: clerkSession, // Kept for backward compatibility\r\n            cookieHeader: cookieHeader,\r\n        };\r\n\r\n        if (outputPath) {\r\n            const jsonContent = JSON.stringify(credentialsData, null, 4);\r\n            try {\r\n                fs.writeFileSync(outputPath, jsonContent);\r\n                console.log(`[OK] Saved credentials to: ${outputPath}`);\r\n            } catch (e: any) {\r\n                console.error(`[ERR] Failed to save to ${outputPath}: ${e.message}`);\r\n                throw e;\r\n            }\r\n        }\r\n\r\n        console.log(\"\\n------------------------------------------\");\r\n        console.log(\"SUCCESS! Setup complete.\");\r\n        console.log(\"------------------------------------------\");\r\n\r\n        return credentialsData;\r\n\r\n    } finally {\r\n        if (browser) await browser.close();\r\n    }\r\n}\r\n"]}
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"login-helper.js","sourceRoot":"","sources":["../../src/auth/login-helper.ts"],"names":[],"mappings":"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","sourcesContent":["import puppeteer from 'puppeteer';\nimport fs from 'node:fs';\nimport path from 'node:path';\n\nexport interface CreateSessionOptions {\n    headless?: boolean;\n    outputPath?: string;\n}\n\nexport async function createSession(options: CreateSessionOptions = {}) {\n    const { headless = false, outputPath } = options;\n\n    console.log(\"------------------------------------------\");\n    console.log(\"   Finary Setup - Browser Login\");\n    console.log(\"------------------------------------------\");\n\n    if (outputPath) {\n        // Ensure directory exists\n        const dir = path.dirname(outputPath);\n        if (!fs.existsSync(dir)) {\n            try {\n                fs.mkdirSync(dir, { recursive: true });\n            } catch (e: any) {\n                throw new Error(`Error creating directory ${dir}: ${e.message}`);\n            }\n        }\n        console.log(`Will save to: ${outputPath}`);\n    }\n\n    console.log(\"Launching browser for authentication...\");\n\n    let browser;\n    try {\n        browser = await puppeteer.launch({\n            headless: headless,\n            defaultViewport: null,\n            args: ['--start-maximized'] // Find browser window easier\n        });\n    } catch (e: any) {\n        console.error(\"Failed to launch browser. Make sure you installed puppeteer: npm install puppeteer\");\n        throw e;\n    }\n\n    try {\n        const page = await browser.newPage();\n\n        await page.setRequestInterception(false); // Ensure interception does not block unless intended\n        page.on('request', request => {\n            const url = request.url();\n            if (url.includes('clerk.finary.com')) {\n                // Optional: log or handle clerk requests\n            }\n        });\n\n        console.log(\"Navigating to Finary login page...\");\n        await page.goto('https://app.finary.com/v2', { waitUntil: 'domcontentloaded' });\n\n        console.log(\"\\nACTION REQUIRED: Please log in to your Finary account in the opened browser window.\\n\");\n        console.log(\"Scanning network traffic for authentication token...\");\n\n        // Promise that resolves when the valid token is found\n        const waitForToken = new Promise<{ token: string, headers: Record<string, string> }>((resolve) => {\n            page.on('response', response => {\n                const request = response.request();\n                const url = request.url();\n\n                // Filter API requests\n                if (url.includes('api.finary.com') || url.includes('clerk.finary.com')) {\n                    // We only want successful requests to ensure the token is valid\n                    if (response.status() === 200) {\n                        const headers = request.headers();\n                        const auth = headers['authorization'];\n\n                        if (auth && auth.startsWith('Bearer ey')) {\n                            const token = auth.replace('Bearer ', '');\n                            console.log(\"✅ Valid token successfully detected from successful request!\");\n                            resolve({ token, headers });\n                        }\n                    }\n                }\n            });\n        });\n\n        const { token, headers } = await waitForToken;\n\n        console.log(\"Token detected! capturing cookies...\");\n        // Capture cookies for both the main app and the clerk subdomain to ensure we get __client, __session, etc.\n        const cookies = await page.cookies('https://app.finary.com', 'https://clerk.finary.com');\n        const cookieHeader = cookies.map(c => `${c.name}=${c.value}`).join('; ');\n\n        const sessionCookie = cookies.find(c => c.name === '__session');\n        const clerkSession = sessionCookie ? sessionCookie.value : null;\n\n        if (cookieHeader) {\n            console.log(\"✅ Cookies captured!\");\n        } else {\n            console.warn(\"⚠️ No cookies found. Refresh verify might fail.\");\n        }\n\n        console.log(\"Fetching user details...\");\n\n        // Use the token to identify the user\n        let email = \"unknown_user@example.com\";\n        try {\n            const meResp = await fetch(\"https://api.finary.com/users/me\", {\n                headers: {\n                    \"Authorization\": `Bearer ${token}`,\n                    \"User-Agent\": \"Mozilla/5.0\"\n                }\n            });\n\n            if (meResp.ok) {\n                const data = await meResp.json();\n                if (data.result && data.result.email) {\n                    email = data.result.email;\n                    console.log(`Identified as: ${email}`);\n                }\n            }\n        } catch (e: any) {\n            console.warn(\"Could not fetch user email (minor issue), proceeding anyway.\");\n        }\n\n        // Filter headers based on user request (\"only these specific headers\")\n        const allowedHeaderKeys = [\n            \"sec-ch-ua-platform\",\n            \"x-client-api-version\",\n            \"referer\",\n            \"accept-language\",\n            \"sec-ch-ua\",\n            \"sec-ch-ua-mobile\",\n            \"user-agent\",\n            \"x-finary-client-id\",\n            \":authority\",\n            \":method\",\n            \":scheme\",\n            \"accept\",\n            \"accept-encoding\",\n            \"origin\",\n            \"priority\",\n            \"sec-fetch-dest\",\n            \"sec-fetch-mode\",\n            \"sec-fetch-site\"\n        ];\n\n        const filteredHeaders: Record<string, string> = {};\n        for (const key of allowedHeaderKeys) {\n            if (headers[key]) {\n                filteredHeaders[key] = headers[key];\n            }\n        }\n\n        const credentialsData = {\n            token: token,\n            refreshUrl: \"https://clerk.finary.com/v1/client/sessions\",\n            headers: filteredHeaders,\n            clerkSession: clerkSession, // Kept for backward compatibility\n            cookieHeader: cookieHeader,\n        };\n\n        if (outputPath) {\n            const jsonContent = JSON.stringify(credentialsData, null, 4);\n            try {\n                fs.writeFileSync(outputPath, jsonContent);\n                console.log(`[OK] Saved credentials to: ${outputPath}`);\n            } catch (e: any) {\n                console.error(`[ERR] Failed to save to ${outputPath}: ${e.message}`);\n                throw e;\n            }\n        }\n\n        console.log(\"\\n------------------------------------------\");\n        console.log(\"SUCCESS! Setup complete.\");\n        console.log(\"------------------------------------------\");\n\n        return credentialsData;\n\n    } finally {\n        if (browser) await browser.close();\n    }\n}\n"]}
@@ -12,4 +12,4 @@ class BenchmarkClient {
12
12
  }
13
13
  }
14
14
  exports.BenchmarkClient = BenchmarkClient;
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmVuY2htYXJrcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jbGllbnQvYmVuY2htYXJrcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw4Q0FBNkU7QUFHN0UsTUFBYSxlQUFlO0lBQ3hCLFlBQW9CLGNBQThCO1FBQTlCLG1CQUFjLEdBQWQsY0FBYyxDQUFnQjtJQUFJLENBQUM7SUFFaEQsS0FBSyxDQUFDLGtCQUFrQixDQUFDLGNBQXNCLEVBQUUsWUFBb0IsRUFBRSxTQUF1Qix5QkFBWSxDQUFDLEdBQUc7UUFDakgsTUFBTSxRQUFRLEdBQUcsTUFBTSxJQUFJLENBQUMsY0FBYyxDQUFDLG9CQUFvQixDQUMzRCxrQkFBa0IsY0FBYyxnQkFBZ0IsWUFBWSx1Q0FBdUMsTUFBTSxFQUFFLENBQzlHLENBQUM7UUFDRixPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUM7SUFDM0IsQ0FBQztDQUNKO0FBVEQsMENBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGaW5hcnlSZXNwb25zZSwgQmVuY2htYXJrQXNzZXQsIEZpbmFyeVBlcmlvZCB9IGZyb20gXCIuLi9pbnRlcmZhY2VzXCI7XHJcbmltcG9ydCB7IFJlcXVlc3RIYW5kbGVyIH0gZnJvbSBcIi4vY2hlY2tpbmdzXCI7XHJcblxyXG5leHBvcnQgY2xhc3MgQmVuY2htYXJrQ2xpZW50IHtcclxuICAgIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVxdWVzdEhhbmRsZXI6IFJlcXVlc3RIYW5kbGVyKSB7IH1cclxuXHJcbiAgICBwdWJsaWMgYXN5bmMgZ2V0QXZhaWxhYmxlQXNzZXRzKG9yZ2FuaXphdGlvbklkOiBzdHJpbmcsIG1lbWJlcnNoaXBJZDogc3RyaW5nLCBwZXJpb2Q6IEZpbmFyeVBlcmlvZCA9IEZpbmFyeVBlcmlvZC5ZVEQpOiBQcm9taXNlPEJlbmNobWFya0Fzc2V0W10+IHtcclxuICAgICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IHRoaXMucmVxdWVzdEhhbmRsZXIuYXV0aGVudGljYXRlZFJlcXVlc3Q8RmluYXJ5UmVzcG9uc2U8QmVuY2htYXJrQXNzZXRbXT4+KFxyXG4gICAgICAgICAgICBgL29yZ2FuaXphdGlvbnMvJHtvcmdhbml6YXRpb25JZH0vbWVtYmVyc2hpcHMvJHttZW1iZXJzaGlwSWR9L2JlbmNobWFya3MvYXZhaWxhYmxlX2Fzc2V0cz9wZXJpb2Q9JHtwZXJpb2R9YFxyXG4gICAgICAgICk7XHJcbiAgICAgICAgcmV0dXJuIHJlc3BvbnNlLnJlc3VsdDtcclxuICAgIH1cclxufVxyXG4iXX0=
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmVuY2htYXJrcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jbGllbnQvYmVuY2htYXJrcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw4Q0FBNkU7QUFHN0UsTUFBYSxlQUFlO0lBQ3hCLFlBQW9CLGNBQThCO1FBQTlCLG1CQUFjLEdBQWQsY0FBYyxDQUFnQjtJQUFJLENBQUM7SUFFaEQsS0FBSyxDQUFDLGtCQUFrQixDQUFDLGNBQXNCLEVBQUUsWUFBb0IsRUFBRSxTQUF1Qix5QkFBWSxDQUFDLEdBQUc7UUFDakgsTUFBTSxRQUFRLEdBQUcsTUFBTSxJQUFJLENBQUMsY0FBYyxDQUFDLG9CQUFvQixDQUMzRCxrQkFBa0IsY0FBYyxnQkFBZ0IsWUFBWSx1Q0FBdUMsTUFBTSxFQUFFLENBQzlHLENBQUM7UUFDRixPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUM7SUFDM0IsQ0FBQztDQUNKO0FBVEQsMENBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGaW5hcnlSZXNwb25zZSwgQmVuY2htYXJrQXNzZXQsIEZpbmFyeVBlcmlvZCB9IGZyb20gXCIuLi9pbnRlcmZhY2VzXCI7XG5pbXBvcnQgeyBSZXF1ZXN0SGFuZGxlciB9IGZyb20gXCIuL2NoZWNraW5nc1wiO1xuXG5leHBvcnQgY2xhc3MgQmVuY2htYXJrQ2xpZW50IHtcbiAgICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlcXVlc3RIYW5kbGVyOiBSZXF1ZXN0SGFuZGxlcikgeyB9XG5cbiAgICBwdWJsaWMgYXN5bmMgZ2V0QXZhaWxhYmxlQXNzZXRzKG9yZ2FuaXphdGlvbklkOiBzdHJpbmcsIG1lbWJlcnNoaXBJZDogc3RyaW5nLCBwZXJpb2Q6IEZpbmFyeVBlcmlvZCA9IEZpbmFyeVBlcmlvZC5ZVEQpOiBQcm9taXNlPEJlbmNobWFya0Fzc2V0W10+IHtcbiAgICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLnJlcXVlc3RIYW5kbGVyLmF1dGhlbnRpY2F0ZWRSZXF1ZXN0PEZpbmFyeVJlc3BvbnNlPEJlbmNobWFya0Fzc2V0W10+PihcbiAgICAgICAgICAgIGAvb3JnYW5pemF0aW9ucy8ke29yZ2FuaXphdGlvbklkfS9tZW1iZXJzaGlwcy8ke21lbWJlcnNoaXBJZH0vYmVuY2htYXJrcy9hdmFpbGFibGVfYXNzZXRzP3BlcmlvZD0ke3BlcmlvZH1gXG4gICAgICAgICk7XG4gICAgICAgIHJldHVybiByZXNwb25zZS5yZXN1bHQ7XG4gICAgfVxufVxuIl19