finary-community 0.1.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 +229 -0
- package/dist/auth/auth-manager.d.ts +12 -0
- package/dist/auth/auth-manager.js +120 -0
- package/dist/auth/login-helper.d.ts +11 -0
- package/dist/auth/login-helper.js +163 -0
- package/dist/client/benchmarks.d.ts +7 -0
- package/dist/client/benchmarks.js +15 -0
- package/dist/client/checkings.d.ts +15 -0
- package/dist/client/checkings.js +46 -0
- package/dist/client/finary-client.d.ts +16 -0
- package/dist/client/finary-client.js +70 -0
- package/dist/client/investments.d.ts +20 -0
- package/dist/client/investments.js +67 -0
- package/dist/client/savings.d.ts +13 -0
- package/dist/client/savings.js +46 -0
- package/dist/greetings.d.ts +2 -0
- package/dist/greetings.js +12 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.js +25 -0
- package/dist/interfaces/benchmark.d.ts +10 -0
- package/dist/interfaces/benchmark.js +3 -0
- package/dist/interfaces/checking.d.ts +78 -0
- package/dist/interfaces/checking.js +3 -0
- package/dist/interfaces/common.d.ts +15 -0
- package/dist/interfaces/common.js +22 -0
- package/dist/interfaces/distribution.d.ts +9 -0
- package/dist/interfaces/distribution.js +3 -0
- package/dist/interfaces/index.d.ts +22 -0
- package/dist/interfaces/index.js +28 -0
- package/dist/interfaces/investment.d.ts +346 -0
- package/dist/interfaces/investment.js +8 -0
- package/dist/interfaces/organization.d.ts +9 -0
- package/dist/interfaces/organization.js +3 -0
- package/dist/interfaces/savings.d.ts +83 -0
- package/dist/interfaces/savings.js +3 -0
- package/dist/interfaces/timeseries.d.ts +9 -0
- package/dist/interfaces/timeseries.js +3 -0
- package/dist/interfaces/transaction.d.ts +78 -0
- package/dist/interfaces/transaction.js +3 -0
- package/dist/interfaces/user.d.ts +163 -0
- package/dist/interfaces/user.js +3 -0
- package/dist/scripts/setup-credentials.d.ts +1 -0
- package/dist/scripts/setup-credentials.js +38 -0
- package/dist/utils/console-logger.d.ts +6 -0
- package/dist/utils/console-logger.js +16 -0
- package/package.json +34 -0
package/README.md
ADDED
|
@@ -0,0 +1,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 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
|
+
```
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { AuthConfig, Credentials } from "../interfaces";
|
|
2
|
+
export declare class AuthManager {
|
|
3
|
+
private token;
|
|
4
|
+
private credentialsPath;
|
|
5
|
+
private cachedCredentials;
|
|
6
|
+
constructor(config: AuthConfig);
|
|
7
|
+
private loadCredentials;
|
|
8
|
+
getToken(): Promise<string>;
|
|
9
|
+
getCredentials(): Credentials;
|
|
10
|
+
refreshToken(): Promise<string>;
|
|
11
|
+
private parseJwt;
|
|
12
|
+
}
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.AuthManager = void 0;
|
|
4
|
+
const node_fs_1 = require("node:fs");
|
|
5
|
+
class AuthManager {
|
|
6
|
+
constructor(config) {
|
|
7
|
+
this.token = null;
|
|
8
|
+
this.cachedCredentials = null;
|
|
9
|
+
this.credentialsPath = config.credentialsPath;
|
|
10
|
+
}
|
|
11
|
+
loadCredentials() {
|
|
12
|
+
if (this.cachedCredentials)
|
|
13
|
+
return this.cachedCredentials;
|
|
14
|
+
if ((0, node_fs_1.existsSync)(this.credentialsPath)) {
|
|
15
|
+
try {
|
|
16
|
+
const data = (0, node_fs_1.readFileSync)(this.credentialsPath, "utf-8");
|
|
17
|
+
this.cachedCredentials = JSON.parse(data);
|
|
18
|
+
return this.cachedCredentials;
|
|
19
|
+
}
|
|
20
|
+
catch (e) {
|
|
21
|
+
console.error(`Failed to parse credentials at ${this.credentialsPath}: ${e}`);
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
throw new Error(`credentials.json not found at ${this.credentialsPath}`);
|
|
25
|
+
}
|
|
26
|
+
async getToken() {
|
|
27
|
+
if (this.token)
|
|
28
|
+
return this.token;
|
|
29
|
+
const creds = this.loadCredentials();
|
|
30
|
+
if (creds.token) {
|
|
31
|
+
this.token = creds.token;
|
|
32
|
+
return this.token;
|
|
33
|
+
}
|
|
34
|
+
throw new Error("No token found in credentials.");
|
|
35
|
+
}
|
|
36
|
+
getCredentials() {
|
|
37
|
+
return this.loadCredentials();
|
|
38
|
+
}
|
|
39
|
+
async refreshToken() {
|
|
40
|
+
const creds = this.loadCredentials();
|
|
41
|
+
// 1. Clerk Refresh
|
|
42
|
+
if (creds.clerkSession && this.token) {
|
|
43
|
+
try {
|
|
44
|
+
const decoded = this.parseJwt(this.token);
|
|
45
|
+
const sessionId = decoded ? decoded.sid : null;
|
|
46
|
+
if (sessionId) {
|
|
47
|
+
const clerkUrl = `https://clerk.finary.com/v1/client/sessions/${sessionId}/tokens?__clerk_api_version=2025-11-10&_clerk_js_version=5.117.0`;
|
|
48
|
+
const cookieHeader = creds.cookieHeader || `__session=${creds.clerkSession}`;
|
|
49
|
+
const response = await fetch(clerkUrl, {
|
|
50
|
+
method: "POST",
|
|
51
|
+
headers: {
|
|
52
|
+
"Cookie": cookieHeader,
|
|
53
|
+
"Content-Type": "application/x-www-form-urlencoded",
|
|
54
|
+
"Origin": "https://app.finary.com",
|
|
55
|
+
"Referer": "https://app.finary.com/",
|
|
56
|
+
"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"
|
|
57
|
+
},
|
|
58
|
+
body: "organization_id="
|
|
59
|
+
});
|
|
60
|
+
if (response.ok) {
|
|
61
|
+
const data = (await response.json());
|
|
62
|
+
if (data && data.jwt) {
|
|
63
|
+
this.token = data.jwt;
|
|
64
|
+
return this.token;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
console.warn(`Clerk refresh failed: ${response.status}`);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
catch (e) {
|
|
73
|
+
console.error(`Clerk refresh error: ${e}`);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
// 2. Legacy Refresh
|
|
77
|
+
if (!creds.refreshUrl || !creds.headers) {
|
|
78
|
+
throw new Error("Missing info to refresh token (legacy).");
|
|
79
|
+
}
|
|
80
|
+
try {
|
|
81
|
+
const headers = { ...creds.headers };
|
|
82
|
+
// Cleanup headers manually since we can't use arbitrary types easily on headers object
|
|
83
|
+
delete headers['content-length'];
|
|
84
|
+
delete headers['host'];
|
|
85
|
+
delete headers['connection'];
|
|
86
|
+
delete headers['origin'];
|
|
87
|
+
delete headers['referer'];
|
|
88
|
+
delete headers['cookie'];
|
|
89
|
+
Object.keys(headers).forEach(key => { if (key.startsWith(':'))
|
|
90
|
+
delete headers[key]; });
|
|
91
|
+
const response = await fetch(creds.refreshUrl, {
|
|
92
|
+
method: "POST",
|
|
93
|
+
headers: headers,
|
|
94
|
+
});
|
|
95
|
+
if (!response.ok) {
|
|
96
|
+
throw new Error(`Legacy refresh failed: ${response.status}`);
|
|
97
|
+
}
|
|
98
|
+
const data = (await response.json());
|
|
99
|
+
if (data && data.jwt) {
|
|
100
|
+
this.token = data.jwt;
|
|
101
|
+
return this.token;
|
|
102
|
+
}
|
|
103
|
+
throw new Error("No JWT in refresh response");
|
|
104
|
+
}
|
|
105
|
+
catch (e) {
|
|
106
|
+
console.error(`Token Refresh Error: ${e}`);
|
|
107
|
+
throw e;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
parseJwt(token) {
|
|
111
|
+
try {
|
|
112
|
+
return JSON.parse(Buffer.from(token.split('.')[1], 'base64').toString());
|
|
113
|
+
}
|
|
114
|
+
catch {
|
|
115
|
+
return null;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
}
|
|
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"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
export interface CreateSessionOptions {
|
|
2
|
+
headless?: boolean;
|
|
3
|
+
outputPath?: string;
|
|
4
|
+
}
|
|
5
|
+
export declare function createSession(options?: CreateSessionOptions): Promise<{
|
|
6
|
+
token: string;
|
|
7
|
+
refreshUrl: string;
|
|
8
|
+
headers: Record<string, string>;
|
|
9
|
+
clerkSession: string | null;
|
|
10
|
+
cookieHeader: string;
|
|
11
|
+
}>;
|
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.createSession = createSession;
|
|
7
|
+
const puppeteer_1 = __importDefault(require("puppeteer"));
|
|
8
|
+
const node_fs_1 = __importDefault(require("node:fs"));
|
|
9
|
+
const node_path_1 = __importDefault(require("node:path"));
|
|
10
|
+
async function createSession(options = {}) {
|
|
11
|
+
const { headless = false, outputPath } = options;
|
|
12
|
+
console.log("------------------------------------------");
|
|
13
|
+
console.log(" Finary Setup - Browser Login");
|
|
14
|
+
console.log("------------------------------------------");
|
|
15
|
+
if (outputPath) {
|
|
16
|
+
// Ensure directory exists
|
|
17
|
+
const dir = node_path_1.default.dirname(outputPath);
|
|
18
|
+
if (!node_fs_1.default.existsSync(dir)) {
|
|
19
|
+
try {
|
|
20
|
+
node_fs_1.default.mkdirSync(dir, { recursive: true });
|
|
21
|
+
}
|
|
22
|
+
catch (e) {
|
|
23
|
+
throw new Error(`Error creating directory ${dir}: ${e.message}`);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
console.log(`Will save to: ${outputPath}`);
|
|
27
|
+
}
|
|
28
|
+
console.log("Launching browser for authentication...");
|
|
29
|
+
let browser;
|
|
30
|
+
try {
|
|
31
|
+
browser = await puppeteer_1.default.launch({
|
|
32
|
+
headless: headless,
|
|
33
|
+
defaultViewport: null,
|
|
34
|
+
args: ['--start-maximized'] // Find browser window easier
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
catch (e) {
|
|
38
|
+
console.error("Failed to launch browser. Make sure you installed puppeteer: npm install puppeteer");
|
|
39
|
+
throw e;
|
|
40
|
+
}
|
|
41
|
+
try {
|
|
42
|
+
const page = await browser.newPage();
|
|
43
|
+
await page.setRequestInterception(false); // Ensure interception does not block unless intended
|
|
44
|
+
page.on('request', request => {
|
|
45
|
+
const url = request.url();
|
|
46
|
+
if (url.includes('clerk.finary.com')) {
|
|
47
|
+
// Optional: log or handle clerk requests
|
|
48
|
+
}
|
|
49
|
+
});
|
|
50
|
+
console.log("Navigating to Finary login page...");
|
|
51
|
+
await page.goto('https://app.finary.com/v2', { waitUntil: 'domcontentloaded' });
|
|
52
|
+
console.log("\nACTION REQUIRED: Please log in to your Finary account in the opened browser window.\n");
|
|
53
|
+
console.log("Scanning network traffic for authentication token...");
|
|
54
|
+
// Promise that resolves when the valid token is found
|
|
55
|
+
const waitForToken = new Promise((resolve) => {
|
|
56
|
+
page.on('response', response => {
|
|
57
|
+
const request = response.request();
|
|
58
|
+
const url = request.url();
|
|
59
|
+
// Filter API requests
|
|
60
|
+
if (url.includes('api.finary.com') || url.includes('clerk.finary.com')) {
|
|
61
|
+
// We only want successful requests to ensure the token is valid
|
|
62
|
+
if (response.status() === 200) {
|
|
63
|
+
const headers = request.headers();
|
|
64
|
+
const auth = headers['authorization'];
|
|
65
|
+
if (auth && auth.startsWith('Bearer ey')) {
|
|
66
|
+
const token = auth.replace('Bearer ', '');
|
|
67
|
+
console.log("✅ Valid token successfully detected from successful request!");
|
|
68
|
+
resolve({ token, headers });
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
const { token, headers } = await waitForToken;
|
|
75
|
+
console.log("Token detected! capturing cookies...");
|
|
76
|
+
// Capture cookies for both the main app and the clerk subdomain to ensure we get __client, __session, etc.
|
|
77
|
+
const cookies = await page.cookies('https://app.finary.com', 'https://clerk.finary.com');
|
|
78
|
+
const cookieHeader = cookies.map(c => `${c.name}=${c.value}`).join('; ');
|
|
79
|
+
const sessionCookie = cookies.find(c => c.name === '__session');
|
|
80
|
+
const clerkSession = sessionCookie ? sessionCookie.value : null;
|
|
81
|
+
if (cookieHeader) {
|
|
82
|
+
console.log("✅ Cookies captured!");
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
console.warn("⚠️ No cookies found. Refresh verify might fail.");
|
|
86
|
+
}
|
|
87
|
+
console.log("Fetching user details...");
|
|
88
|
+
// Use the token to identify the user
|
|
89
|
+
let email = "unknown_user@example.com";
|
|
90
|
+
try {
|
|
91
|
+
const meResp = await fetch("https://api.finary.com/users/me", {
|
|
92
|
+
headers: {
|
|
93
|
+
"Authorization": `Bearer ${token}`,
|
|
94
|
+
"User-Agent": "Mozilla/5.0"
|
|
95
|
+
}
|
|
96
|
+
});
|
|
97
|
+
if (meResp.ok) {
|
|
98
|
+
const data = await meResp.json();
|
|
99
|
+
if (data.result && data.result.email) {
|
|
100
|
+
email = data.result.email;
|
|
101
|
+
console.log(`Identified as: ${email}`);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
catch (e) {
|
|
106
|
+
console.warn("Could not fetch user email (minor issue), proceeding anyway.");
|
|
107
|
+
}
|
|
108
|
+
// Filter headers based on user request ("only these specific headers")
|
|
109
|
+
const allowedHeaderKeys = [
|
|
110
|
+
"sec-ch-ua-platform",
|
|
111
|
+
"x-client-api-version",
|
|
112
|
+
"referer",
|
|
113
|
+
"accept-language",
|
|
114
|
+
"sec-ch-ua",
|
|
115
|
+
"sec-ch-ua-mobile",
|
|
116
|
+
"user-agent",
|
|
117
|
+
"x-finary-client-id",
|
|
118
|
+
":authority",
|
|
119
|
+
":method",
|
|
120
|
+
":scheme",
|
|
121
|
+
"accept",
|
|
122
|
+
"accept-encoding",
|
|
123
|
+
"origin",
|
|
124
|
+
"priority",
|
|
125
|
+
"sec-fetch-dest",
|
|
126
|
+
"sec-fetch-mode",
|
|
127
|
+
"sec-fetch-site"
|
|
128
|
+
];
|
|
129
|
+
const filteredHeaders = {};
|
|
130
|
+
for (const key of allowedHeaderKeys) {
|
|
131
|
+
if (headers[key]) {
|
|
132
|
+
filteredHeaders[key] = headers[key];
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
const credentialsData = {
|
|
136
|
+
token: token,
|
|
137
|
+
refreshUrl: "https://clerk.finary.com/v1/client/sessions",
|
|
138
|
+
headers: filteredHeaders,
|
|
139
|
+
clerkSession: clerkSession, // Kept for backward compatibility
|
|
140
|
+
cookieHeader: cookieHeader,
|
|
141
|
+
};
|
|
142
|
+
if (outputPath) {
|
|
143
|
+
const jsonContent = JSON.stringify(credentialsData, null, 4);
|
|
144
|
+
try {
|
|
145
|
+
node_fs_1.default.writeFileSync(outputPath, jsonContent);
|
|
146
|
+
console.log(`[OK] Saved credentials to: ${outputPath}`);
|
|
147
|
+
}
|
|
148
|
+
catch (e) {
|
|
149
|
+
console.error(`[ERR] Failed to save to ${outputPath}: ${e.message}`);
|
|
150
|
+
throw e;
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
console.log("\n------------------------------------------");
|
|
154
|
+
console.log("SUCCESS! Setup complete.");
|
|
155
|
+
console.log("------------------------------------------");
|
|
156
|
+
return credentialsData;
|
|
157
|
+
}
|
|
158
|
+
finally {
|
|
159
|
+
if (browser)
|
|
160
|
+
await browser.close();
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"login-helper.js","sourceRoot":"","sources":["../../src/auth/login-helper.ts"],"names":[],"mappings":";;;;;AASA,sCA0KC;AAnLD,0DAAkC;AAClC,sDAAyB;AACzB,0DAA6B;AAOtB,KAAK,UAAU,aAAa,CAAC,UAAgC,EAAE;IAClE,MAAM,EAAE,QAAQ,GAAG,KAAK,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC;IAEjD,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAC;IAC1D,OAAO,CAAC,GAAG,CAAC,iCAAiC,CAAC,CAAC;IAC/C,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAC;IAE1D,IAAI,UAAU,EAAE,CAAC;QACb,0BAA0B;QAC1B,MAAM,GAAG,GAAG,mBAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QACrC,IAAI,CAAC,iBAAE,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;YACtB,IAAI,CAAC;gBACD,iBAAE,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YAC3C,CAAC;YAAC,OAAO,CAAM,EAAE,CAAC;gBACd,MAAM,IAAI,KAAK,CAAC,4BAA4B,GAAG,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;YACrE,CAAC;QACL,CAAC;QACD,OAAO,CAAC,GAAG,CAAC,iBAAiB,UAAU,EAAE,CAAC,CAAC;IAC/C,CAAC;IAED,OAAO,CAAC,GAAG,CAAC,yCAAyC,CAAC,CAAC;IAEvD,IAAI,OAAO,CAAC;IACZ,IAAI,CAAC;QACD,OAAO,GAAG,MAAM,mBAAS,CAAC,MAAM,CAAC;YAC7B,QAAQ,EAAE,QAAQ;YAClB,eAAe,EAAE,IAAI;YACrB,IAAI,EAAE,CAAC,mBAAmB,CAAC,CAAC,6BAA6B;SAC5D,CAAC,CAAC;IACP,CAAC;IAAC,OAAO,CAAM,EAAE,CAAC;QACd,OAAO,CAAC,KAAK,CAAC,oFAAoF,CAAC,CAAC;QACpG,MAAM,CAAC,CAAC;IACZ,CAAC;IAED,IAAI,CAAC;QACD,MAAM,IAAI,GAAG,MAAM,OAAO,CAAC,OAAO,EAAE,CAAC;QAErC,MAAM,IAAI,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC,CAAC,qDAAqD;QAC/F,IAAI,CAAC,EAAE,CAAC,SAAS,EAAE,OAAO,CAAC,EAAE;YACzB,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;YAC1B,IAAI,GAAG,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAAC;gBACnC,yCAAyC;YAC7C,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,OAAO,CAAC,GAAG,CAAC,oCAAoC,CAAC,CAAC;QAClD,MAAM,IAAI,CAAC,IAAI,CAAC,2BAA2B,EAAE,EAAE,SAAS,EAAE,kBAAkB,EAAE,CAAC,CAAC;QAEhF,OAAO,CAAC,GAAG,CAAC,yFAAyF,CAAC,CAAC;QACvG,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;QAEpE,sDAAsD;QACtD,MAAM,YAAY,GAAG,IAAI,OAAO,CAAqD,CAAC,OAAO,EAAE,EAAE;YAC7F,IAAI,CAAC,EAAE,CAAC,UAAU,EAAE,QAAQ,CAAC,EAAE;gBAC3B,MAAM,OAAO,GAAG,QAAQ,CAAC,OAAO,EAAE,CAAC;gBACnC,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;gBAE1B,sBAAsB;gBACtB,IAAI,GAAG,CAAC,QAAQ,CAAC,gBAAgB,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAAC;oBACrE,gEAAgE;oBAChE,IAAI,QAAQ,CAAC,MAAM,EAAE,KAAK,GAAG,EAAE,CAAC;wBAC5B,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC;wBAClC,MAAM,IAAI,GAAG,OAAO,CAAC,eAAe,CAAC,CAAC;wBAEtC,IAAI,IAAI,IAAI,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;4BACvC,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;4BAC1C,OAAO,CAAC,GAAG,CAAC,8DAA8D,CAAC,CAAC;4BAC5E,OAAO,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;wBAChC,CAAC;oBACL,CAAC;gBACL,CAAC;YACL,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,MAAM,EAAE,KAAK,EAAE,OAAO,EAAE,GAAG,MAAM,YAAY,CAAC;QAE9C,OAAO,CAAC,GAAG,CAAC,sCAAsC,CAAC,CAAC;QACpD,2GAA2G;QAC3G,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,wBAAwB,EAAE,0BAA0B,CAAC,CAAC;QACzF,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEzE,MAAM,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,WAAW,CAAC,CAAC;QAChE,MAAM,YAAY,GAAG,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;QAEhE,IAAI,YAAY,EAAE,CAAC;YACf,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC;QACvC,CAAC;aAAM,CAAC;YACJ,OAAO,CAAC,IAAI,CAAC,iDAAiD,CAAC,CAAC;QACpE,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC;QAExC,qCAAqC;QACrC,IAAI,KAAK,GAAG,0BAA0B,CAAC;QACvC,IAAI,CAAC;YACD,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,iCAAiC,EAAE;gBAC1D,OAAO,EAAE;oBACL,eAAe,EAAE,UAAU,KAAK,EAAE;oBAClC,YAAY,EAAE,aAAa;iBAC9B;aACJ,CAAC,CAAC;YAEH,IAAI,MAAM,CAAC,EAAE,EAAE,CAAC;gBACZ,MAAM,IAAI,GAAG,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;gBACjC,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;oBACnC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC;oBAC1B,OAAO,CAAC,GAAG,CAAC,kBAAkB,KAAK,EAAE,CAAC,CAAC;gBAC3C,CAAC;YACL,CAAC;QACL,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YACd,OAAO,CAAC,IAAI,CAAC,8DAA8D,CAAC,CAAC;QACjF,CAAC;QAED,uEAAuE;QACvE,MAAM,iBAAiB,GAAG;YACtB,oBAAoB;YACpB,sBAAsB;YACtB,SAAS;YACT,iBAAiB;YACjB,WAAW;YACX,kBAAkB;YAClB,YAAY;YACZ,oBAAoB;YACpB,YAAY;YACZ,SAAS;YACT,SAAS;YACT,QAAQ;YACR,iBAAiB;YACjB,QAAQ;YACR,UAAU;YACV,gBAAgB;YAChB,gBAAgB;YAChB,gBAAgB;SACnB,CAAC;QAEF,MAAM,eAAe,GAA2B,EAAE,CAAC;QACnD,KAAK,MAAM,GAAG,IAAI,iBAAiB,EAAE,CAAC;YAClC,IAAI,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC;gBACf,eAAe,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC;YACxC,CAAC;QACL,CAAC;QAED,MAAM,eAAe,GAAG;YACpB,KAAK,EAAE,KAAK;YACZ,UAAU,EAAE,6CAA6C;YACzD,OAAO,EAAE,eAAe;YACxB,YAAY,EAAE,YAAY,EAAE,kCAAkC;YAC9D,YAAY,EAAE,YAAY;SAC7B,CAAC;QAEF,IAAI,UAAU,EAAE,CAAC;YACb,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,eAAe,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;YAC7D,IAAI,CAAC;gBACD,iBAAE,CAAC,aAAa,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;gBAC1C,OAAO,CAAC,GAAG,CAAC,8BAA8B,UAAU,EAAE,CAAC,CAAC;YAC5D,CAAC;YAAC,OAAO,CAAM,EAAE,CAAC;gBACd,OAAO,CAAC,KAAK,CAAC,2BAA2B,UAAU,KAAK,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;gBACrE,MAAM,CAAC,CAAC;YACZ,CAAC;QACL,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,8CAA8C,CAAC,CAAC;QAC5D,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC;QACxC,OAAO,CAAC,GAAG,CAAC,4CAA4C,CAAC,CAAC;QAE1D,OAAO,eAAe,CAAC;IAE3B,CAAC;YAAS,CAAC;QACP,IAAI,OAAO;YAAE,MAAM,OAAO,CAAC,KAAK,EAAE,CAAC;IACvC,CAAC;AACL,CAAC","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"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { BenchmarkAsset, FinaryPeriod } from "../interfaces";
|
|
2
|
+
import { RequestHandler } from "./checkings";
|
|
3
|
+
export declare class BenchmarkClient {
|
|
4
|
+
private requestHandler;
|
|
5
|
+
constructor(requestHandler: RequestHandler);
|
|
6
|
+
getAvailableAssets(organizationId: string, membershipId: string, period?: FinaryPeriod): Promise<BenchmarkAsset[]>;
|
|
7
|
+
}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.BenchmarkClient = void 0;
|
|
4
|
+
const interfaces_1 = require("../interfaces");
|
|
5
|
+
class BenchmarkClient {
|
|
6
|
+
constructor(requestHandler) {
|
|
7
|
+
this.requestHandler = requestHandler;
|
|
8
|
+
}
|
|
9
|
+
async getAvailableAssets(organizationId, membershipId, period = interfaces_1.FinaryPeriod.YTD) {
|
|
10
|
+
const response = await this.requestHandler.authenticatedRequest(`/organizations/${organizationId}/memberships/${membershipId}/benchmarks/available_assets?period=${period}`);
|
|
11
|
+
return response.result;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
exports.BenchmarkClient = BenchmarkClient;
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmVuY2htYXJrcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jbGllbnQvYmVuY2htYXJrcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw4Q0FBNkU7QUFHN0UsTUFBYSxlQUFlO0lBQ3hCLFlBQW9CLGNBQThCO1FBQTlCLG1CQUFjLEdBQWQsY0FBYyxDQUFnQjtJQUFJLENBQUM7SUFFaEQsS0FBSyxDQUFDLGtCQUFrQixDQUFDLGNBQXNCLEVBQUUsWUFBb0IsRUFBRSxTQUF1Qix5QkFBWSxDQUFDLEdBQUc7UUFDakgsTUFBTSxRQUFRLEdBQUcsTUFBTSxJQUFJLENBQUMsY0FBYyxDQUFDLG9CQUFvQixDQUMzRCxrQkFBa0IsY0FBYyxnQkFBZ0IsWUFBWSx1Q0FBdUMsTUFBTSxFQUFFLENBQzlHLENBQUM7UUFDRixPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUM7SUFDM0IsQ0FBQztDQUNKO0FBVEQsMENBU0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBGaW5hcnlSZXNwb25zZSwgQmVuY2htYXJrQXNzZXQsIEZpbmFyeVBlcmlvZCB9IGZyb20gXCIuLi9pbnRlcmZhY2VzXCI7XHJcbmltcG9ydCB7IFJlcXVlc3RIYW5kbGVyIH0gZnJvbSBcIi4vY2hlY2tpbmdzXCI7XHJcblxyXG5leHBvcnQgY2xhc3MgQmVuY2htYXJrQ2xpZW50IHtcclxuICAgIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVxdWVzdEhhbmRsZXI6IFJlcXVlc3RIYW5kbGVyKSB7IH1cclxuXHJcbiAgICBwdWJsaWMgYXN5bmMgZ2V0QXZhaWxhYmxlQXNzZXRzKG9yZ2FuaXphdGlvbklkOiBzdHJpbmcsIG1lbWJlcnNoaXBJZDogc3RyaW5nLCBwZXJpb2Q6IEZpbmFyeVBlcmlvZCA9IEZpbmFyeVBlcmlvZC5ZVEQpOiBQcm9taXNlPEJlbmNobWFya0Fzc2V0W10+IHtcclxuICAgICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IHRoaXMucmVxdWVzdEhhbmRsZXIuYXV0aGVudGljYXRlZFJlcXVlc3Q8RmluYXJ5UmVzcG9uc2U8QmVuY2htYXJrQXNzZXRbXT4+KFxyXG4gICAgICAgICAgICBgL29yZ2FuaXphdGlvbnMvJHtvcmdhbml6YXRpb25JZH0vbWVtYmVyc2hpcHMvJHttZW1iZXJzaGlwSWR9L2JlbmNobWFya3MvYXZhaWxhYmxlX2Fzc2V0cz9wZXJpb2Q9JHtwZXJpb2R9YFxyXG4gICAgICAgICk7XHJcbiAgICAgICAgcmV0dXJuIHJlc3BvbnNlLnJlc3VsdDtcclxuICAgIH1cclxufVxyXG4iXX0=
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { CheckingAccount, CheckingPortfolio, FinaryPeriod, Timeseries, TimeseriesType, Distribution, DistributionType, Transaction } from "../interfaces";
|
|
2
|
+
export interface RequestHandler {
|
|
3
|
+
authenticatedRequest<T>(endpoint: string, method?: string, body?: any): Promise<T>;
|
|
4
|
+
}
|
|
5
|
+
export declare class CheckingClient {
|
|
6
|
+
private requestHandler;
|
|
7
|
+
constructor(requestHandler: RequestHandler);
|
|
8
|
+
getAccounts(organizationId: string, membershipId: string, period?: FinaryPeriod): Promise<CheckingAccount[]>;
|
|
9
|
+
getAccount(organizationId: string, membershipId: string, accountId: string, period?: FinaryPeriod): Promise<CheckingAccount>;
|
|
10
|
+
getPortfolio(organizationId: string, membershipId: string, period?: FinaryPeriod): Promise<CheckingPortfolio>;
|
|
11
|
+
getTimeseries(organizationId: string, membershipId: string, period?: FinaryPeriod, type?: TimeseriesType): Promise<Timeseries[]>;
|
|
12
|
+
getAccountTimeseries(organizationId: string, membershipId: string, accountId: string, period?: FinaryPeriod, type?: TimeseriesType): Promise<Timeseries[]>;
|
|
13
|
+
getDistribution(organizationId: string, membershipId: string, type?: DistributionType): Promise<Distribution>;
|
|
14
|
+
getTransactions(organizationId: string, membershipId: string, page?: number, perPage?: number, query?: string, accountId?: string): Promise<Transaction[]>;
|
|
15
|
+
}
|