@rool-dev/client 0.3.0-dev.52960ac → 0.3.0-dev.f9b91b8
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 +29 -56
- package/dist/auth.d.ts +35 -7
- package/dist/auth.d.ts.map +1 -1
- package/dist/auth.js +326 -34
- package/dist/auth.js.map +1 -1
- package/dist/client.d.ts +2 -2
- package/dist/client.d.ts.map +1 -1
- package/dist/client.js +12 -12
- package/dist/client.js.map +1 -1
- package/dist/graphql.d.ts +1 -1
- package/dist/graphql.d.ts.map +1 -1
- package/dist/graphql.js +1 -1
- package/dist/graphql.js.map +1 -1
- package/dist/media.d.ts +1 -1
- package/dist/media.d.ts.map +1 -1
- package/dist/media.js +1 -1
- package/dist/media.js.map +1 -1
- package/dist/subscription.d.ts +1 -1
- package/dist/subscription.d.ts.map +1 -1
- package/dist/subscription.js +1 -1
- package/dist/subscription.js.map +1 -1
- package/dist/types.d.ts +10 -18
- package/dist/types.d.ts.map +1 -1
- package/package.json +2 -8
- package/dist/auth-browser.d.ts +0 -71
- package/dist/auth-browser.d.ts.map +0 -1
- package/dist/auth-browser.js +0 -328
- package/dist/auth-browser.js.map +0 -1
- package/dist/auth-node.d.ts +0 -33
- package/dist/auth-node.d.ts.map +0 -1
- package/dist/auth-node.js +0 -271
- package/dist/auth-node.js.map +0 -1
package/dist/auth-browser.js
DELETED
|
@@ -1,328 +0,0 @@
|
|
|
1
|
-
const GCIP_REFRESH_ENDPOINT = 'https://securetoken.googleapis.com/v1/token';
|
|
2
|
-
const REFRESH_BUFFER_MS = 5 * 60 * 1000; // Refresh 5 minutes before expiry
|
|
3
|
-
const DEFAULT_STORAGE_PREFIX = 'rool_';
|
|
4
|
-
export class BrowserAuthProvider {
|
|
5
|
-
config;
|
|
6
|
-
apiKey = null;
|
|
7
|
-
apiKeyFetchPromise = null;
|
|
8
|
-
refreshPromise = null;
|
|
9
|
-
refreshTimeoutId = null;
|
|
10
|
-
get storageKeys() {
|
|
11
|
-
const prefix = this.config.storagePrefix ?? DEFAULT_STORAGE_PREFIX;
|
|
12
|
-
return {
|
|
13
|
-
access: `${prefix}access_token`,
|
|
14
|
-
refresh: `${prefix}refresh_token`,
|
|
15
|
-
expiresAt: `${prefix}token_expires_at`,
|
|
16
|
-
state: `${prefix}auth_state`,
|
|
17
|
-
};
|
|
18
|
-
}
|
|
19
|
-
/** Auth URL without trailing slash */
|
|
20
|
-
get authBaseUrl() {
|
|
21
|
-
return this.config.authUrl.replace(/\/+$/, '');
|
|
22
|
-
}
|
|
23
|
-
constructor(config) {
|
|
24
|
-
this.config = config;
|
|
25
|
-
}
|
|
26
|
-
/**
|
|
27
|
-
* Initialize auth manager - should be called on app startup.
|
|
28
|
-
* Processes any auth callback in the URL and sets up auto-refresh.
|
|
29
|
-
*/
|
|
30
|
-
initialize() {
|
|
31
|
-
const wasCallback = this.processCallback();
|
|
32
|
-
this.scheduleTokenRefresh();
|
|
33
|
-
return wasCallback;
|
|
34
|
-
}
|
|
35
|
-
/**
|
|
36
|
-
* Check if user is currently authenticated (has valid token).
|
|
37
|
-
*/
|
|
38
|
-
isAuthenticated() {
|
|
39
|
-
const accessToken = this.readAccessToken();
|
|
40
|
-
const expiresAt = this.readExpiresAt();
|
|
41
|
-
if (!accessToken || !expiresAt)
|
|
42
|
-
return false;
|
|
43
|
-
return Date.now() < expiresAt;
|
|
44
|
-
}
|
|
45
|
-
/**
|
|
46
|
-
* Get current access token, refreshing if expired.
|
|
47
|
-
* Returns undefined if not authenticated.
|
|
48
|
-
*/
|
|
49
|
-
async getToken() {
|
|
50
|
-
const accessToken = this.readAccessToken();
|
|
51
|
-
const expiresAt = this.readExpiresAt();
|
|
52
|
-
if (!accessToken)
|
|
53
|
-
return undefined;
|
|
54
|
-
// Token expired or about to expire - try refresh
|
|
55
|
-
if (expiresAt && Date.now() >= expiresAt - REFRESH_BUFFER_MS) {
|
|
56
|
-
const refreshed = await this.tryRefreshToken();
|
|
57
|
-
if (!refreshed)
|
|
58
|
-
return undefined;
|
|
59
|
-
return this.readAccessToken() ?? undefined;
|
|
60
|
-
}
|
|
61
|
-
return accessToken;
|
|
62
|
-
}
|
|
63
|
-
/**
|
|
64
|
-
* Get user info decoded from JWT token.
|
|
65
|
-
*/
|
|
66
|
-
getUser() {
|
|
67
|
-
const accessToken = this.readAccessToken();
|
|
68
|
-
if (!accessToken)
|
|
69
|
-
return { email: null, name: null };
|
|
70
|
-
return this.decodeUserInfo(accessToken);
|
|
71
|
-
}
|
|
72
|
-
/**
|
|
73
|
-
* Initiate login by redirecting to auth page.
|
|
74
|
-
*/
|
|
75
|
-
login() {
|
|
76
|
-
const loginUrl = new URL(`${this.authBaseUrl}/`);
|
|
77
|
-
const redirectTarget = window.location.origin + window.location.pathname + window.location.search;
|
|
78
|
-
loginUrl.searchParams.set('redirect_uri', redirectTarget);
|
|
79
|
-
const state = this.generateState();
|
|
80
|
-
this.storeState(state);
|
|
81
|
-
loginUrl.searchParams.set('state', state);
|
|
82
|
-
window.location.href = loginUrl.toString();
|
|
83
|
-
}
|
|
84
|
-
/**
|
|
85
|
-
* Logout - clear all tokens and state.
|
|
86
|
-
*/
|
|
87
|
-
logout() {
|
|
88
|
-
this.clearTokens();
|
|
89
|
-
this.clearState();
|
|
90
|
-
this.cancelScheduledRefresh();
|
|
91
|
-
this.config.onAuthStateChanged(false);
|
|
92
|
-
}
|
|
93
|
-
/**
|
|
94
|
-
* Process auth callback from URL fragment.
|
|
95
|
-
* Should be called on page load.
|
|
96
|
-
* @returns true if callback was processed
|
|
97
|
-
*/
|
|
98
|
-
processCallback() {
|
|
99
|
-
const hash = window.location.hash ?? '';
|
|
100
|
-
const trimmed = hash.startsWith('#') ? hash.slice(1) : hash;
|
|
101
|
-
if (!trimmed)
|
|
102
|
-
return false;
|
|
103
|
-
const params = new URLSearchParams(trimmed);
|
|
104
|
-
const idToken = params.get('id_token');
|
|
105
|
-
if (!idToken)
|
|
106
|
-
return false;
|
|
107
|
-
const refreshToken = params.get('refresh_token');
|
|
108
|
-
const expiresIn = params.get('expires_in');
|
|
109
|
-
const expiresAt = expiresIn ? Date.now() + Number(expiresIn) * 1000 : NaN;
|
|
110
|
-
const incomingState = params.get('state');
|
|
111
|
-
const storedState = this.readState();
|
|
112
|
-
// Validate state - if we stored one, require it back
|
|
113
|
-
if (storedState && incomingState !== storedState) {
|
|
114
|
-
console.error('[RoolClient] Auth state mismatch. Token fragment ignored.');
|
|
115
|
-
return false;
|
|
116
|
-
}
|
|
117
|
-
if (!Number.isFinite(expiresAt)) {
|
|
118
|
-
console.error('[RoolClient] Auth response missing expires_in. Token ignored.');
|
|
119
|
-
return false;
|
|
120
|
-
}
|
|
121
|
-
// Clear state and store tokens
|
|
122
|
-
this.clearState();
|
|
123
|
-
this.writeTokens(idToken, refreshToken, expiresAt);
|
|
124
|
-
// Clean URL
|
|
125
|
-
const cleanUrl = window.location.origin + window.location.pathname + window.location.search;
|
|
126
|
-
window.history.replaceState({}, document.title, cleanUrl);
|
|
127
|
-
// Schedule refresh and notify
|
|
128
|
-
this.scheduleTokenRefresh();
|
|
129
|
-
this.config.onAuthStateChanged(true);
|
|
130
|
-
return true;
|
|
131
|
-
}
|
|
132
|
-
/**
|
|
133
|
-
* Destroy auth manager - clear refresh timers.
|
|
134
|
-
*/
|
|
135
|
-
destroy() {
|
|
136
|
-
this.cancelScheduledRefresh();
|
|
137
|
-
}
|
|
138
|
-
// ===========================================================================
|
|
139
|
-
// Private methods
|
|
140
|
-
// ===========================================================================
|
|
141
|
-
/**
|
|
142
|
-
* Get the API key, fetching from server if not provided in config.
|
|
143
|
-
*/
|
|
144
|
-
async getApiKey() {
|
|
145
|
-
// Already have it
|
|
146
|
-
if (this.apiKey)
|
|
147
|
-
return this.apiKey;
|
|
148
|
-
// Already fetching
|
|
149
|
-
if (this.apiKeyFetchPromise)
|
|
150
|
-
return this.apiKeyFetchPromise;
|
|
151
|
-
// Fetch from server
|
|
152
|
-
this.apiKeyFetchPromise = fetch(`${this.authBaseUrl}/config.json`)
|
|
153
|
-
.then(async (response) => {
|
|
154
|
-
if (!response.ok) {
|
|
155
|
-
console.warn('[RoolClient] Failed to fetch API key from server');
|
|
156
|
-
return null;
|
|
157
|
-
}
|
|
158
|
-
const data = await response.json();
|
|
159
|
-
if (data.apiKey && typeof data.apiKey === 'string') {
|
|
160
|
-
this.apiKey = data.apiKey;
|
|
161
|
-
return this.apiKey;
|
|
162
|
-
}
|
|
163
|
-
return null;
|
|
164
|
-
})
|
|
165
|
-
.catch((error) => {
|
|
166
|
-
console.warn('[RoolClient] Failed to fetch API key:', error);
|
|
167
|
-
return null;
|
|
168
|
-
})
|
|
169
|
-
.finally(() => {
|
|
170
|
-
this.apiKeyFetchPromise = null;
|
|
171
|
-
});
|
|
172
|
-
return this.apiKeyFetchPromise;
|
|
173
|
-
}
|
|
174
|
-
async tryRefreshToken() {
|
|
175
|
-
// Deduplicate concurrent refresh attempts
|
|
176
|
-
if (this.refreshPromise) {
|
|
177
|
-
return this.refreshPromise;
|
|
178
|
-
}
|
|
179
|
-
const refreshToken = localStorage.getItem(this.storageKeys.refresh);
|
|
180
|
-
if (!refreshToken)
|
|
181
|
-
return false;
|
|
182
|
-
// Get API key (from config or server)
|
|
183
|
-
const apiKey = await this.getApiKey();
|
|
184
|
-
if (!apiKey) {
|
|
185
|
-
console.warn('[RoolClient] Cannot refresh token: no API key available');
|
|
186
|
-
return false;
|
|
187
|
-
}
|
|
188
|
-
const refreshUrl = new URL(GCIP_REFRESH_ENDPOINT);
|
|
189
|
-
refreshUrl.searchParams.set('key', apiKey);
|
|
190
|
-
this.refreshPromise = fetch(refreshUrl.toString(), {
|
|
191
|
-
method: 'POST',
|
|
192
|
-
headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
|
|
193
|
-
body: new URLSearchParams({
|
|
194
|
-
grant_type: 'refresh_token',
|
|
195
|
-
refresh_token: refreshToken,
|
|
196
|
-
}),
|
|
197
|
-
})
|
|
198
|
-
.then(async (response) => {
|
|
199
|
-
if (!response.ok) {
|
|
200
|
-
throw new Error(`Refresh failed: ${response.status} ${response.statusText}`);
|
|
201
|
-
}
|
|
202
|
-
const data = await response.json();
|
|
203
|
-
const accessToken = data.id_token ?? data.access_token ?? null;
|
|
204
|
-
const nextRefreshToken = data.refresh_token ?? refreshToken;
|
|
205
|
-
const expiresAt = data.expires_in ? Date.now() + data.expires_in * 1000 : NaN;
|
|
206
|
-
if (!accessToken || !Number.isFinite(expiresAt)) {
|
|
207
|
-
throw new Error('Refresh response missing access token or expires_in');
|
|
208
|
-
}
|
|
209
|
-
this.writeTokens(accessToken, nextRefreshToken, expiresAt);
|
|
210
|
-
this.scheduleTokenRefresh();
|
|
211
|
-
return true;
|
|
212
|
-
})
|
|
213
|
-
.catch((error) => {
|
|
214
|
-
console.error('[RoolClient] Token refresh failed:', error);
|
|
215
|
-
this.clearTokens();
|
|
216
|
-
this.config.onAuthStateChanged(false);
|
|
217
|
-
return false;
|
|
218
|
-
})
|
|
219
|
-
.finally(() => {
|
|
220
|
-
this.refreshPromise = null;
|
|
221
|
-
});
|
|
222
|
-
return this.refreshPromise;
|
|
223
|
-
}
|
|
224
|
-
scheduleTokenRefresh() {
|
|
225
|
-
this.cancelScheduledRefresh();
|
|
226
|
-
const expiresAt = this.readExpiresAt();
|
|
227
|
-
if (!expiresAt)
|
|
228
|
-
return;
|
|
229
|
-
const refreshAt = expiresAt - REFRESH_BUFFER_MS;
|
|
230
|
-
const delay = refreshAt - Date.now();
|
|
231
|
-
if (delay <= 0) {
|
|
232
|
-
// Already needs refresh
|
|
233
|
-
void this.tryRefreshToken();
|
|
234
|
-
}
|
|
235
|
-
else {
|
|
236
|
-
this.refreshTimeoutId = setTimeout(() => {
|
|
237
|
-
void this.tryRefreshToken();
|
|
238
|
-
}, delay);
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
cancelScheduledRefresh() {
|
|
242
|
-
if (this.refreshTimeoutId !== null) {
|
|
243
|
-
clearTimeout(this.refreshTimeoutId);
|
|
244
|
-
this.refreshTimeoutId = null;
|
|
245
|
-
}
|
|
246
|
-
}
|
|
247
|
-
readAccessToken() {
|
|
248
|
-
return localStorage.getItem(this.storageKeys.access);
|
|
249
|
-
}
|
|
250
|
-
readExpiresAt() {
|
|
251
|
-
const raw = localStorage.getItem(this.storageKeys.expiresAt);
|
|
252
|
-
if (!raw)
|
|
253
|
-
return null;
|
|
254
|
-
const parsed = Number.parseInt(raw, 10);
|
|
255
|
-
return Number.isFinite(parsed) ? parsed : null;
|
|
256
|
-
}
|
|
257
|
-
writeTokens(accessToken, refreshToken, expiresAt) {
|
|
258
|
-
if (accessToken) {
|
|
259
|
-
localStorage.setItem(this.storageKeys.access, accessToken);
|
|
260
|
-
}
|
|
261
|
-
else {
|
|
262
|
-
localStorage.removeItem(this.storageKeys.access);
|
|
263
|
-
}
|
|
264
|
-
if (refreshToken) {
|
|
265
|
-
localStorage.setItem(this.storageKeys.refresh, refreshToken);
|
|
266
|
-
}
|
|
267
|
-
else {
|
|
268
|
-
localStorage.removeItem(this.storageKeys.refresh);
|
|
269
|
-
}
|
|
270
|
-
if (expiresAt !== null && Number.isFinite(expiresAt)) {
|
|
271
|
-
localStorage.setItem(this.storageKeys.expiresAt, Math.floor(expiresAt).toString());
|
|
272
|
-
}
|
|
273
|
-
else {
|
|
274
|
-
localStorage.removeItem(this.storageKeys.expiresAt);
|
|
275
|
-
}
|
|
276
|
-
}
|
|
277
|
-
clearTokens() {
|
|
278
|
-
this.writeTokens(null, null, null);
|
|
279
|
-
}
|
|
280
|
-
storeState(value) {
|
|
281
|
-
try {
|
|
282
|
-
sessionStorage.setItem(this.storageKeys.state, value);
|
|
283
|
-
}
|
|
284
|
-
catch {
|
|
285
|
-
// Ignore storage restrictions
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
readState() {
|
|
289
|
-
try {
|
|
290
|
-
return sessionStorage.getItem(this.storageKeys.state);
|
|
291
|
-
}
|
|
292
|
-
catch {
|
|
293
|
-
return null;
|
|
294
|
-
}
|
|
295
|
-
}
|
|
296
|
-
clearState() {
|
|
297
|
-
try {
|
|
298
|
-
sessionStorage.removeItem(this.storageKeys.state);
|
|
299
|
-
}
|
|
300
|
-
catch {
|
|
301
|
-
// Ignore storage restrictions
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
generateState() {
|
|
305
|
-
try {
|
|
306
|
-
const buffer = new Uint8Array(16);
|
|
307
|
-
window.crypto.getRandomValues(buffer);
|
|
308
|
-
return Array.from(buffer, (value) => value.toString(16).padStart(2, '0')).join('');
|
|
309
|
-
}
|
|
310
|
-
catch {
|
|
311
|
-
return Math.random().toString(36).slice(2) + Math.random().toString(36).slice(2);
|
|
312
|
-
}
|
|
313
|
-
}
|
|
314
|
-
decodeUserInfo(accessToken) {
|
|
315
|
-
try {
|
|
316
|
-
const payload = JSON.parse(atob(accessToken.split('.')[1]));
|
|
317
|
-
return {
|
|
318
|
-
email: payload.email || null,
|
|
319
|
-
name: payload.name || null,
|
|
320
|
-
};
|
|
321
|
-
}
|
|
322
|
-
catch (error) {
|
|
323
|
-
console.error('[RoolClient] Failed to decode token:', error);
|
|
324
|
-
return { email: null, name: null };
|
|
325
|
-
}
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
//# sourceMappingURL=auth-browser.js.map
|
package/dist/auth-browser.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"auth-browser.js","sourceRoot":"","sources":["../src/auth-browser.ts"],"names":[],"mappings":"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"}
|
package/dist/auth-node.d.ts
DELETED
|
@@ -1,33 +0,0 @@
|
|
|
1
|
-
import type { AuthProvider, UserInfo } from './types.js';
|
|
2
|
-
export interface NodeAuthConfig {
|
|
3
|
-
/** Path to store credentials (default: ~/.config/rool/credentials.json) */
|
|
4
|
-
credentialsPath?: string;
|
|
5
|
-
/** Timeout for login flow in ms (default: 5 minutes) */
|
|
6
|
-
loginTimeoutMs?: number;
|
|
7
|
-
onAuthStateChanged?: (authenticated: boolean) => void;
|
|
8
|
-
}
|
|
9
|
-
export declare class NodeAuthProvider implements AuthProvider {
|
|
10
|
-
private config;
|
|
11
|
-
private apiKey;
|
|
12
|
-
private _authUrl;
|
|
13
|
-
constructor(config?: NodeAuthConfig);
|
|
14
|
-
/** Called by AuthManager to inject the auth URL */
|
|
15
|
-
setAuthUrl(url: string): void;
|
|
16
|
-
private get credentialsPath();
|
|
17
|
-
/**
|
|
18
|
-
* Get the auth endpoint URL (without trailing slash).
|
|
19
|
-
*/
|
|
20
|
-
private get authEndpoint();
|
|
21
|
-
initialize(): boolean;
|
|
22
|
-
getToken(): Promise<string | undefined>;
|
|
23
|
-
getUser(): UserInfo;
|
|
24
|
-
isAuthenticated(): boolean;
|
|
25
|
-
login(): Promise<void>;
|
|
26
|
-
logout(): void;
|
|
27
|
-
private readCredentials;
|
|
28
|
-
private writeCredentials;
|
|
29
|
-
private getApiKey;
|
|
30
|
-
private refreshToken;
|
|
31
|
-
private startLoopbackServer;
|
|
32
|
-
}
|
|
33
|
-
//# sourceMappingURL=auth-node.d.ts.map
|
package/dist/auth-node.d.ts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"auth-node.d.ts","sourceRoot":"","sources":["../src/auth-node.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,YAAY,EAAE,QAAQ,EAAE,MAAM,YAAY,CAAC;AAIzD,MAAM,WAAW,cAAc;IAC3B,2EAA2E;IAC3E,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,wDAAwD;IACxD,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,kBAAkB,CAAC,EAAE,CAAC,aAAa,EAAE,OAAO,KAAK,IAAI,CAAC;CACzD;AAQD,qBAAa,gBAAiB,YAAW,YAAY;IACjD,OAAO,CAAC,MAAM,CAAiB;IAC/B,OAAO,CAAC,MAAM,CAAuB;IACrC,OAAO,CAAC,QAAQ,CAAuB;gBAE3B,MAAM,GAAE,cAAmB;IAIvC,mDAAmD;IACnD,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI;IAI7B,OAAO,KAAK,eAAe,GAO1B;IAED;;OAEG;IACH,OAAO,KAAK,YAAY,GAKvB;IAED,UAAU,IAAI,OAAO;IAIf,QAAQ,IAAI,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;IAY7C,OAAO,IAAI,QAAQ;IAenB,eAAe,IAAI,OAAO;IAQpB,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IA6C5B,MAAM,IAAI,IAAI;IAYd,OAAO,CAAC,eAAe;IAWvB,OAAO,CAAC,gBAAgB;YAYV,SAAS;YAcT,YAAY;IAoD1B,OAAO,CAAC,mBAAmB;CAuE9B"}
|
package/dist/auth-node.js
DELETED
|
@@ -1,271 +0,0 @@
|
|
|
1
|
-
import * as http from 'node:http';
|
|
2
|
-
import * as fs from 'node:fs';
|
|
3
|
-
import * as path from 'node:path';
|
|
4
|
-
import * as os from 'node:os';
|
|
5
|
-
import open from 'open';
|
|
6
|
-
const GCIP_REFRESH_ENDPOINT = 'https://securetoken.googleapis.com/v1/token';
|
|
7
|
-
export class NodeAuthProvider {
|
|
8
|
-
config;
|
|
9
|
-
apiKey = null;
|
|
10
|
-
_authUrl = null;
|
|
11
|
-
constructor(config = {}) {
|
|
12
|
-
this.config = config;
|
|
13
|
-
}
|
|
14
|
-
/** Called by AuthManager to inject the auth URL */
|
|
15
|
-
setAuthUrl(url) {
|
|
16
|
-
this._authUrl = url;
|
|
17
|
-
}
|
|
18
|
-
get credentialsPath() {
|
|
19
|
-
if (this.config.credentialsPath) {
|
|
20
|
-
return this.config.credentialsPath;
|
|
21
|
-
}
|
|
22
|
-
const homeDir = os.homedir();
|
|
23
|
-
const configDir = path.join(homeDir, '.config', 'rool');
|
|
24
|
-
return path.join(configDir, 'credentials.json');
|
|
25
|
-
}
|
|
26
|
-
/**
|
|
27
|
-
* Get the auth endpoint URL (without trailing slash).
|
|
28
|
-
*/
|
|
29
|
-
get authEndpoint() {
|
|
30
|
-
if (!this._authUrl) {
|
|
31
|
-
throw new Error('Auth URL not set. Ensure RoolClient is configured correctly.');
|
|
32
|
-
}
|
|
33
|
-
return this._authUrl.replace(/\/+$/, '');
|
|
34
|
-
}
|
|
35
|
-
initialize() {
|
|
36
|
-
return this.isAuthenticated();
|
|
37
|
-
}
|
|
38
|
-
async getToken() {
|
|
39
|
-
const creds = this.readCredentials();
|
|
40
|
-
if (!creds)
|
|
41
|
-
return undefined;
|
|
42
|
-
// Refresh if expiring in less than 5 minutes
|
|
43
|
-
if (Date.now() >= creds.expires_at - 5 * 60 * 1000) {
|
|
44
|
-
return this.refreshToken(creds);
|
|
45
|
-
}
|
|
46
|
-
return creds.access_token;
|
|
47
|
-
}
|
|
48
|
-
getUser() {
|
|
49
|
-
const creds = this.readCredentials();
|
|
50
|
-
if (!creds?.access_token)
|
|
51
|
-
return { email: null, name: null };
|
|
52
|
-
try {
|
|
53
|
-
const payload = JSON.parse(Buffer.from(creds.access_token.split('.')[1], 'base64').toString());
|
|
54
|
-
return {
|
|
55
|
-
email: payload.email || null,
|
|
56
|
-
name: payload.name || null,
|
|
57
|
-
};
|
|
58
|
-
}
|
|
59
|
-
catch {
|
|
60
|
-
return { email: null, name: null };
|
|
61
|
-
}
|
|
62
|
-
}
|
|
63
|
-
isAuthenticated() {
|
|
64
|
-
const creds = this.readCredentials();
|
|
65
|
-
if (!creds)
|
|
66
|
-
return false;
|
|
67
|
-
// We consider it authenticated if we have a refresh token,
|
|
68
|
-
// effectively "remember me" behavior for CLI
|
|
69
|
-
return !!creds.refresh_token || Date.now() < creds.expires_at;
|
|
70
|
-
}
|
|
71
|
-
async login() {
|
|
72
|
-
const { server, closeAll } = await this.startLoopbackServer();
|
|
73
|
-
const port = server.address().port;
|
|
74
|
-
const redirectUri = `http://localhost:${port}`;
|
|
75
|
-
// Generate code verifier/state if needed, currently just state
|
|
76
|
-
const state = Math.random().toString(36).substring(2);
|
|
77
|
-
// Auth endpoint is the root of the auth service
|
|
78
|
-
const loginUrl = new URL(`${this.authEndpoint}/`);
|
|
79
|
-
loginUrl.searchParams.set('redirect_uri', redirectUri);
|
|
80
|
-
loginUrl.searchParams.set('state', state);
|
|
81
|
-
console.log('Opening browser to login:', loginUrl.toString());
|
|
82
|
-
await open(loginUrl.toString());
|
|
83
|
-
const timeoutMs = this.config.loginTimeoutMs ?? 5 * 60 * 1000; // 5 minutes default
|
|
84
|
-
return new Promise((resolve, reject) => {
|
|
85
|
-
const timeout = setTimeout(() => {
|
|
86
|
-
closeAll();
|
|
87
|
-
reject(new Error('Login timed out. Please try again.'));
|
|
88
|
-
}, timeoutMs);
|
|
89
|
-
server.on('authenticated', (tokens) => {
|
|
90
|
-
clearTimeout(timeout);
|
|
91
|
-
const expiresAt = Date.now() + (tokens.expires_in * 1000);
|
|
92
|
-
this.writeCredentials({
|
|
93
|
-
access_token: tokens.id_token, // GCIP returns id_token
|
|
94
|
-
refresh_token: tokens.refresh_token,
|
|
95
|
-
expires_at: expiresAt
|
|
96
|
-
});
|
|
97
|
-
this.config.onAuthStateChanged?.(true);
|
|
98
|
-
closeAll();
|
|
99
|
-
resolve();
|
|
100
|
-
});
|
|
101
|
-
server.on('error', (err) => {
|
|
102
|
-
clearTimeout(timeout);
|
|
103
|
-
closeAll();
|
|
104
|
-
reject(err);
|
|
105
|
-
});
|
|
106
|
-
});
|
|
107
|
-
}
|
|
108
|
-
logout() {
|
|
109
|
-
const filePath = this.credentialsPath;
|
|
110
|
-
if (fs.existsSync(filePath)) {
|
|
111
|
-
fs.unlinkSync(filePath);
|
|
112
|
-
}
|
|
113
|
-
this.config.onAuthStateChanged?.(false);
|
|
114
|
-
}
|
|
115
|
-
// ===========================================================================
|
|
116
|
-
// Private Helpers
|
|
117
|
-
// ===========================================================================
|
|
118
|
-
readCredentials() {
|
|
119
|
-
try {
|
|
120
|
-
const filePath = this.credentialsPath;
|
|
121
|
-
if (!fs.existsSync(filePath))
|
|
122
|
-
return null;
|
|
123
|
-
const data = fs.readFileSync(filePath, 'utf-8');
|
|
124
|
-
return JSON.parse(data);
|
|
125
|
-
}
|
|
126
|
-
catch {
|
|
127
|
-
return null;
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
writeCredentials(creds) {
|
|
131
|
-
const filePath = this.credentialsPath;
|
|
132
|
-
const dir = path.dirname(filePath);
|
|
133
|
-
try {
|
|
134
|
-
fs.mkdirSync(dir, { recursive: true });
|
|
135
|
-
fs.writeFileSync(filePath, JSON.stringify(creds, null, 2), { mode: 0o600 });
|
|
136
|
-
}
|
|
137
|
-
catch (error) {
|
|
138
|
-
console.error('[RoolClient] Failed to save credentials:', error);
|
|
139
|
-
}
|
|
140
|
-
}
|
|
141
|
-
async getApiKey() {
|
|
142
|
-
if (this.apiKey)
|
|
143
|
-
return this.apiKey;
|
|
144
|
-
try {
|
|
145
|
-
const response = await fetch(`${this.authEndpoint}/config.json`);
|
|
146
|
-
if (!response.ok)
|
|
147
|
-
return null;
|
|
148
|
-
const data = await response.json();
|
|
149
|
-
this.apiKey = data.apiKey;
|
|
150
|
-
return this.apiKey;
|
|
151
|
-
}
|
|
152
|
-
catch {
|
|
153
|
-
return null;
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
async refreshToken(creds) {
|
|
157
|
-
if (!creds.refresh_token)
|
|
158
|
-
return undefined;
|
|
159
|
-
const apiKey = await this.getApiKey();
|
|
160
|
-
if (!apiKey) {
|
|
161
|
-
console.warn('[RoolClient] Cannot refresh: API key not found');
|
|
162
|
-
return undefined;
|
|
163
|
-
}
|
|
164
|
-
try {
|
|
165
|
-
// Derive referer from auth URL (required for API key restrictions)
|
|
166
|
-
const referer = new URL(this.authEndpoint).origin;
|
|
167
|
-
const response = await fetch(`${GCIP_REFRESH_ENDPOINT}?key=${apiKey}`, {
|
|
168
|
-
method: 'POST',
|
|
169
|
-
headers: {
|
|
170
|
-
'Content-Type': 'application/x-www-form-urlencoded',
|
|
171
|
-
'Referer': referer,
|
|
172
|
-
},
|
|
173
|
-
body: new URLSearchParams({
|
|
174
|
-
grant_type: 'refresh_token',
|
|
175
|
-
refresh_token: creds.refresh_token,
|
|
176
|
-
}),
|
|
177
|
-
});
|
|
178
|
-
if (!response.ok) {
|
|
179
|
-
// 400 typically means invalid/expired refresh token - clear credentials
|
|
180
|
-
if (response.status === 400) {
|
|
181
|
-
console.warn('[RoolClient] Refresh token expired or invalid. Please login again.');
|
|
182
|
-
this.logout();
|
|
183
|
-
}
|
|
184
|
-
else {
|
|
185
|
-
console.warn(`[RoolClient] Refresh failed: ${response.status}`);
|
|
186
|
-
}
|
|
187
|
-
return undefined;
|
|
188
|
-
}
|
|
189
|
-
const data = await response.json();
|
|
190
|
-
const newCreds = {
|
|
191
|
-
access_token: data.id_token || data.access_token,
|
|
192
|
-
refresh_token: data.refresh_token || creds.refresh_token,
|
|
193
|
-
expires_at: Date.now() + (Number(data.expires_in) * 1000),
|
|
194
|
-
};
|
|
195
|
-
this.writeCredentials(newCreds);
|
|
196
|
-
return newCreds.access_token;
|
|
197
|
-
}
|
|
198
|
-
catch (error) {
|
|
199
|
-
console.error('[RoolClient] Refresh failed:', error);
|
|
200
|
-
return undefined;
|
|
201
|
-
}
|
|
202
|
-
}
|
|
203
|
-
startLoopbackServer() {
|
|
204
|
-
return new Promise((resolve) => {
|
|
205
|
-
const server = http.createServer((req, res) => {
|
|
206
|
-
const url = new URL(req.url || '/', `http://localhost`);
|
|
207
|
-
// 1. Serve the capture page if we just have a root request
|
|
208
|
-
if (url.pathname === '/') {
|
|
209
|
-
res.writeHead(200, { 'Content-Type': 'text/html' });
|
|
210
|
-
res.end(`
|
|
211
|
-
<html>
|
|
212
|
-
<body>
|
|
213
|
-
<h1>Authenticating...</h1>
|
|
214
|
-
<script>
|
|
215
|
-
// Extract hash and post to /callback
|
|
216
|
-
if (window.location.hash) {
|
|
217
|
-
const hash = window.location.hash.substring(1);
|
|
218
|
-
fetch('/callback', {
|
|
219
|
-
method: 'POST',
|
|
220
|
-
headers: {'Content-Type': 'application/x-www-form-urlencoded'},
|
|
221
|
-
body: hash
|
|
222
|
-
})
|
|
223
|
-
.then(() => document.body.innerHTML = '<h1>Login Successful. You can close this window.</h1>')
|
|
224
|
-
.catch(err => document.body.innerHTML = '<h1>Error: ' + err.message + '</h1>');
|
|
225
|
-
}
|
|
226
|
-
</script>
|
|
227
|
-
</body>
|
|
228
|
-
</html>
|
|
229
|
-
`);
|
|
230
|
-
return;
|
|
231
|
-
}
|
|
232
|
-
// 2. Handle the callback POST
|
|
233
|
-
if (url.pathname === '/callback' && req.method === 'POST') {
|
|
234
|
-
let body = '';
|
|
235
|
-
req.on('data', (chunk) => body += chunk.toString());
|
|
236
|
-
req.on('end', () => {
|
|
237
|
-
const params = new URLSearchParams(body);
|
|
238
|
-
const idToken = params.get('id_token');
|
|
239
|
-
const refreshToken = params.get('refresh_token');
|
|
240
|
-
const expiresIn = params.get('expires_in');
|
|
241
|
-
if (idToken && expiresIn) {
|
|
242
|
-
res.writeHead(200);
|
|
243
|
-
res.end('OK');
|
|
244
|
-
server.emit('authenticated', {
|
|
245
|
-
id_token: idToken,
|
|
246
|
-
refresh_token: refreshToken,
|
|
247
|
-
expires_in: Number(expiresIn)
|
|
248
|
-
});
|
|
249
|
-
}
|
|
250
|
-
else {
|
|
251
|
-
res.writeHead(400);
|
|
252
|
-
res.end('Invalid tokens');
|
|
253
|
-
}
|
|
254
|
-
});
|
|
255
|
-
return;
|
|
256
|
-
}
|
|
257
|
-
res.writeHead(404);
|
|
258
|
-
res.end();
|
|
259
|
-
});
|
|
260
|
-
// Listen on random port
|
|
261
|
-
server.listen(0, '127.0.0.1', () => resolve({
|
|
262
|
-
server,
|
|
263
|
-
closeAll: () => {
|
|
264
|
-
server.close();
|
|
265
|
-
server.closeAllConnections();
|
|
266
|
-
},
|
|
267
|
-
}));
|
|
268
|
-
});
|
|
269
|
-
}
|
|
270
|
-
}
|
|
271
|
-
//# sourceMappingURL=auth-node.js.map
|
package/dist/auth-node.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"auth-node.js","sourceRoot":"","sources":["../src/auth-node.ts"],"names":[],"mappings":"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"}
|