latchkey 0.1.4 → 1.0.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.
Files changed (99) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +93 -56
  3. package/dist/integrations/SKILL.md +77 -0
  4. package/dist/package.json +67 -0
  5. package/dist/scripts/encryptFile.d.ts +21 -0
  6. package/dist/scripts/encryptFile.d.ts.map +1 -0
  7. package/dist/scripts/encryptFile.js +101 -0
  8. package/dist/scripts/encryptFile.js.map +1 -0
  9. package/dist/src/apiCredentialStore.d.ts +1 -0
  10. package/dist/src/apiCredentialStore.d.ts.map +1 -1
  11. package/dist/src/apiCredentialStore.js +12 -0
  12. package/dist/src/apiCredentialStore.js.map +1 -1
  13. package/dist/src/apiCredentials.d.ts +116 -1
  14. package/dist/src/apiCredentials.d.ts.map +1 -1
  15. package/dist/src/apiCredentials.js +119 -1
  16. package/dist/src/apiCredentials.js.map +1 -1
  17. package/dist/src/browserState.d.ts +8 -0
  18. package/dist/src/browserState.d.ts.map +1 -0
  19. package/dist/src/browserState.js +21 -0
  20. package/dist/src/browserState.js.map +1 -0
  21. package/dist/src/cli.js +5 -3
  22. package/dist/src/cli.js.map +1 -1
  23. package/dist/src/cliCommands.d.ts.map +1 -1
  24. package/dist/src/cliCommands.js +218 -81
  25. package/dist/src/cliCommands.js.map +1 -1
  26. package/dist/src/config.d.ts +13 -0
  27. package/dist/src/config.d.ts.map +1 -1
  28. package/dist/src/config.js +50 -4
  29. package/dist/src/config.js.map +1 -1
  30. package/dist/src/index.d.ts +1 -1
  31. package/dist/src/index.d.ts.map +1 -1
  32. package/dist/src/index.js +2 -2
  33. package/dist/src/index.js.map +1 -1
  34. package/dist/src/oauthUtils.d.ts +49 -0
  35. package/dist/src/oauthUtils.d.ts.map +1 -0
  36. package/dist/src/oauthUtils.js +183 -0
  37. package/dist/src/oauthUtils.js.map +1 -0
  38. package/dist/src/playwrightUtils.d.ts +14 -1
  39. package/dist/src/playwrightUtils.d.ts.map +1 -1
  40. package/dist/src/playwrightUtils.js +37 -8
  41. package/dist/src/playwrightUtils.js.map +1 -1
  42. package/dist/src/registry.d.ts.map +1 -1
  43. package/dist/src/registry.js +20 -4
  44. package/dist/src/registry.js.map +1 -1
  45. package/dist/src/services/base.d.ts +43 -15
  46. package/dist/src/services/base.d.ts.map +1 -1
  47. package/dist/src/services/base.js +49 -9
  48. package/dist/src/services/base.js.map +1 -1
  49. package/dist/src/services/discord.d.ts +4 -3
  50. package/dist/src/services/discord.d.ts.map +1 -1
  51. package/dist/src/services/discord.js +6 -22
  52. package/dist/src/services/discord.js.map +1 -1
  53. package/dist/src/services/dropbox.d.ts +5 -4
  54. package/dist/src/services/dropbox.d.ts.map +1 -1
  55. package/dist/src/services/dropbox.js +10 -27
  56. package/dist/src/services/dropbox.js.map +1 -1
  57. package/dist/src/services/github.d.ts +5 -4
  58. package/dist/src/services/github.d.ts.map +1 -1
  59. package/dist/src/services/github.js +21 -30
  60. package/dist/src/services/github.js.map +1 -1
  61. package/dist/src/services/google.d.ts +34 -0
  62. package/dist/src/services/google.d.ts.map +1 -0
  63. package/dist/src/services/google.js +336 -0
  64. package/dist/src/services/google.js.map +1 -0
  65. package/dist/src/services/index.d.ts +4 -2
  66. package/dist/src/services/index.d.ts.map +1 -1
  67. package/dist/src/services/index.js +4 -1
  68. package/dist/src/services/index.js.map +1 -1
  69. package/dist/src/services/linear.d.ts +5 -4
  70. package/dist/src/services/linear.d.ts.map +1 -1
  71. package/dist/src/services/linear.js +10 -29
  72. package/dist/src/services/linear.js.map +1 -1
  73. package/dist/src/services/mailchimp.d.ts +11 -0
  74. package/dist/src/services/mailchimp.d.ts.map +1 -0
  75. package/dist/src/services/mailchimp.js +16 -0
  76. package/dist/src/services/mailchimp.js.map +1 -0
  77. package/dist/src/services/notion.d.ts +29 -0
  78. package/dist/src/services/notion.d.ts.map +1 -0
  79. package/dist/src/services/notion.js +102 -0
  80. package/dist/src/services/notion.js.map +1 -0
  81. package/dist/src/services/slack.d.ts +3 -1
  82. package/dist/src/services/slack.d.ts.map +1 -1
  83. package/dist/src/services/slack.js +5 -5
  84. package/dist/src/services/slack.js.map +1 -1
  85. package/dist/src/skillMd.d.ts +2 -0
  86. package/dist/src/skillMd.d.ts.map +1 -0
  87. package/dist/src/skillMd.js +19 -0
  88. package/dist/src/skillMd.js.map +1 -0
  89. package/dist/tests/apiCredentials.test.js +59 -1
  90. package/dist/tests/apiCredentials.test.js.map +1 -1
  91. package/dist/tests/cli.test.js +283 -104
  92. package/dist/tests/cli.test.js.map +1 -1
  93. package/dist/tests/playwrightDownload.test.js +2 -2
  94. package/dist/tests/playwrightDownload.test.js.map +1 -1
  95. package/dist/tests/registry.test.js +28 -3
  96. package/dist/tests/registry.test.js.map +1 -1
  97. package/dist/tests/servicesAgainstRecordings.test.js +3 -0
  98. package/dist/tests/servicesAgainstRecordings.test.js.map +1 -1
  99. package/package.json +6 -6
@@ -0,0 +1,336 @@
1
+ /**
2
+ * Google service implementation with OAuth flow.
3
+ */
4
+ import fs from 'node:fs/promises';
5
+ import { OAuthCredentials } from '../apiCredentials.js';
6
+ import { generateLatchkeyAppName, showSpinnerPage, withTempBrowserContext, } from '../playwrightUtils.js';
7
+ import { exchangeCodeForTokens, refreshAccessToken, startOAuthCallbackServer, } from '../oauthUtils.js';
8
+ import { Service, BrowserFollowupServiceSession, LoginFailedError, LoginCancelledError, isBrowserClosedError, } from './base.js';
9
+ const DEFAULT_TIMEOUT_MS = 8000;
10
+ const LOGIN_TIMEOUT_MS = 120000;
11
+ const GOOGLE_TOKEN_ENDPOINT = 'https://oauth2.googleapis.com/token';
12
+ const APIS = [
13
+ 'gmail.googleapis.com',
14
+ 'calendar-json.googleapis.com',
15
+ 'drive.googleapis.com',
16
+ 'sheets.googleapis.com',
17
+ 'docs.googleapis.com',
18
+ 'people.googleapis.com', // Contacts API
19
+ ];
20
+ const OAUTH_SCOPES = [
21
+ // User info (for credential validation)
22
+ 'https://www.googleapis.com/auth/userinfo.profile',
23
+ 'https://www.googleapis.com/auth/userinfo.email',
24
+ // Gmail API
25
+ 'https://www.googleapis.com/auth/gmail.readonly',
26
+ 'https://www.googleapis.com/auth/gmail.modify',
27
+ 'https://www.googleapis.com/auth/gmail.compose',
28
+ 'https://www.googleapis.com/auth/gmail.send',
29
+ // Calendar API
30
+ 'https://www.googleapis.com/auth/calendar',
31
+ 'https://www.googleapis.com/auth/calendar.events',
32
+ // Drive API
33
+ 'https://www.googleapis.com/auth/drive',
34
+ 'https://www.googleapis.com/auth/drive.file',
35
+ // Sheets API
36
+ 'https://www.googleapis.com/auth/spreadsheets',
37
+ // Docs API
38
+ 'https://www.googleapis.com/auth/documents',
39
+ // Contacts API
40
+ 'https://www.googleapis.com/auth/contacts',
41
+ 'https://www.googleapis.com/auth/contacts.readonly',
42
+ ];
43
+ /**
44
+ * Parse client_secret.json content.
45
+ */
46
+ function parseClientSecretJson(content) {
47
+ try {
48
+ const json = JSON.parse(content);
49
+ const config = json.installed ?? json.web;
50
+ if (!config) {
51
+ throw new LoginFailedError('Invalid client_secret.json: missing "installed" or "web" configuration.');
52
+ }
53
+ return {
54
+ clientId: config.client_id,
55
+ clientSecret: config.client_secret,
56
+ };
57
+ }
58
+ catch (error) {
59
+ if (error instanceof LoginFailedError) {
60
+ throw error;
61
+ }
62
+ throw new LoginFailedError(`Failed to parse client_secret.json: ${error instanceof Error ? error.message : String(error)}`);
63
+ }
64
+ }
65
+ async function createProject(page) {
66
+ // Navigate to the projects page
67
+ await page.goto('https://console.cloud.google.com/projectselector2/home/dashboard', {
68
+ timeout: DEFAULT_TIMEOUT_MS,
69
+ });
70
+ // Always create a new project
71
+ const createProjectButton = page.locator('.projectselector-project-create');
72
+ await createProjectButton.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
73
+ await createProjectButton.click();
74
+ const projectNameInput = page.locator('proj-name-id-input input');
75
+ await projectNameInput.waitFor({ timeout: DEFAULT_TIMEOUT_MS * 100 });
76
+ await projectNameInput.fill(generateLatchkeyAppName());
77
+ const createButton = page.locator('button[type="submit"]');
78
+ await createButton.click();
79
+ await page.waitForURL('https://console.cloud.google.com/home/dashboard?project=**', {
80
+ timeout: 16000,
81
+ });
82
+ const urlObj = new URL(page.url());
83
+ const projectId = urlObj.searchParams.get('project');
84
+ if (!projectId) {
85
+ throw new LoginFailedError('Failed to create or retrieve Google Cloud project ID.');
86
+ }
87
+ return projectId;
88
+ }
89
+ async function enableApis(page, projectSlug) {
90
+ for (const api of APIS) {
91
+ await enableApi(page, projectSlug, api);
92
+ }
93
+ }
94
+ async function enableApi(page, projectSlug, apiName) {
95
+ await page.goto(`https://console.cloud.google.com/apis/library/${apiName}?project=${projectSlug}`, {
96
+ timeout: DEFAULT_TIMEOUT_MS,
97
+ });
98
+ const successIcon = page.locator('.cfc-icon-status-success'); // Present when API is already enabled.
99
+ const enableButton = page
100
+ .locator('.mp-details-cta-button-primary button .mdc-button__label')
101
+ .filter({ visible: true });
102
+ const sucessOrEnableButton = successIcon.or(enableButton);
103
+ await sucessOrEnableButton.isVisible({ timeout: DEFAULT_TIMEOUT_MS });
104
+ if (await successIcon.isVisible()) {
105
+ return;
106
+ }
107
+ await enableButton.click();
108
+ const stopIndicator = page.locator('.cfc-icon-stop');
109
+ await stopIndicator.waitFor({ timeout: 18000 });
110
+ }
111
+ async function configureBranding(page, projectSlug) {
112
+ await page.goto(`https://console.cloud.google.com/auth/branding?project=${projectSlug}`, {
113
+ timeout: DEFAULT_TIMEOUT_MS,
114
+ });
115
+ const getStartedButton = page.locator('cfc-empty-state-actions .mdc-button__label');
116
+ await getStartedButton.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
117
+ await getStartedButton.click();
118
+ const appNameInput = page.locator('input[formcontrolname="displayName"]');
119
+ await appNameInput.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
120
+ await appNameInput.fill(generateLatchkeyAppName());
121
+ const emailSelector = page.locator('svg[data-icon-name="arrowDropDownIcon"]').nth(0);
122
+ await emailSelector.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
123
+ await emailSelector.click();
124
+ const supportEmailOption = page.locator('mat-option > span:nth-child(1)').first();
125
+ await supportEmailOption.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
126
+ const supportEmailValue = await supportEmailOption.textContent();
127
+ await supportEmailOption.click();
128
+ const nextButton = page.locator('.cfc-stepper-step-button');
129
+ await nextButton.click();
130
+ const internalAudienceRadio = page.locator('.mdc-radio').nth(0);
131
+ await internalAudienceRadio.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
132
+ await internalAudienceRadio.click();
133
+ await nextButton.click();
134
+ const contactEmailInput = page.locator('mat-chip-grid[formcontrolname="emails"] input');
135
+ await contactEmailInput.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
136
+ await contactEmailInput.fill(supportEmailValue ?? '');
137
+ await nextButton.click();
138
+ const agreeCheckbox = page.locator('input[type="checkbox"]');
139
+ await agreeCheckbox.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
140
+ await agreeCheckbox.click();
141
+ await nextButton.click();
142
+ const createButton = page.locator('.cfc-stepper-submit-button button');
143
+ await createButton.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
144
+ await createButton.click();
145
+ }
146
+ async function createOAuthClient(page) {
147
+ // Navigate to credentials page
148
+ await page.goto('https://console.cloud.google.com/apis/credentials', {
149
+ timeout: DEFAULT_TIMEOUT_MS,
150
+ });
151
+ // Click "Create Credentials" button
152
+ const createCredentialsButton = page.locator('services-create-credentials-menu button');
153
+ await createCredentialsButton.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
154
+ await createCredentialsButton.click();
155
+ // Click "OAuth client ID"
156
+ const oauthClientIdOption = page.locator('cfc-menu-item[track-metadata-type="OAUTH_CLIENT"]');
157
+ await oauthClientIdOption.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
158
+ await oauthClientIdOption.click();
159
+ const applicationTypeDropdown = page.locator('svg[data-icon-name="arrowDropDownIcon"]').nth(0);
160
+ await applicationTypeDropdown.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
161
+ await applicationTypeDropdown.click();
162
+ const desktopAppOption = page.locator('#_1rif_mat-option-5');
163
+ await desktopAppOption.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
164
+ await desktopAppOption.click();
165
+ const clientNameInput = page.locator('input[formcontrolname="displayName"]');
166
+ await clientNameInput.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
167
+ await clientNameInput.fill(generateLatchkeyAppName());
168
+ // Click Create
169
+ const createButton = page.locator('cfc-progress-button button');
170
+ await createButton.click();
171
+ // Download the JSON file
172
+ const downloadButton = page.locator('cfc-icon[icon="download"]');
173
+ await downloadButton.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
174
+ const [download] = await Promise.all([page.waitForEvent('download'), downloadButton.click()]);
175
+ const path = await download.path();
176
+ if (!path) {
177
+ throw new LoginFailedError('Failed to download client_secret.json');
178
+ }
179
+ const content = await fs.readFile(path, 'utf-8');
180
+ return parseClientSecretJson(content);
181
+ }
182
+ function checkGoogleLoginResponse(response, loginDetector) {
183
+ if (loginDetector.isLoggedIn) {
184
+ return;
185
+ }
186
+ const request = response.request();
187
+ // Detect successful login by checking for Google account access
188
+ if (request.url().startsWith('https://console.cloud.google.com/')) {
189
+ if (response.status() === 200) {
190
+ void response.text().then((text) => {
191
+ // Check if we're actually logged in (not on login page)
192
+ if (!text.includes('accounts.google.com/signin')) {
193
+ loginDetector.isLoggedIn = true;
194
+ }
195
+ });
196
+ }
197
+ }
198
+ }
199
+ async function waitForGoogleLogin(page) {
200
+ const loginDetector = { isLoggedIn: false };
201
+ const responseHandler = (response) => {
202
+ checkGoogleLoginResponse(response, loginDetector);
203
+ };
204
+ page.on('response', responseHandler);
205
+ while (!loginDetector.isLoggedIn) {
206
+ await page.waitForTimeout(100);
207
+ }
208
+ page.off('response', responseHandler);
209
+ }
210
+ class GoogleServiceSession extends BrowserFollowupServiceSession {
211
+ loginDetector = { isLoggedIn: false };
212
+ onResponse(response) {
213
+ checkGoogleLoginResponse(response, this.loginDetector);
214
+ }
215
+ isLoginComplete() {
216
+ return this.loginDetector.isLoggedIn;
217
+ }
218
+ /**
219
+ * Override to skip the spinner page since the OAuth flow requires user interaction
220
+ * (granting consent on Google's authorization page).
221
+ */
222
+ finalizeCredentials(_browser, context, oldCredentials) {
223
+ return this.performBrowserFollowup(context, oldCredentials);
224
+ }
225
+ async performBrowserFollowup(context, oldCredentials) {
226
+ const page = context.pages()[0];
227
+ if (!page) {
228
+ throw new LoginFailedError('No page available in browser context.');
229
+ }
230
+ // Require existing credentials with client ID and secret
231
+ if (!(oldCredentials instanceof OAuthCredentials)) {
232
+ throw new LoginFailedError('Google login requires existing OAuth client credentials. Run browser-prepare first.');
233
+ }
234
+ const clientId = oldCredentials.clientId;
235
+ const clientSecret = oldCredentials.clientSecret;
236
+ // Perform OAuth flow with localhost server
237
+ const { accessToken, refreshToken, accessTokenExpiresAt, refreshTokenExpiresAt } = await this.performOAuthFlow(context, page, clientId, clientSecret);
238
+ await page.close();
239
+ return new OAuthCredentials(clientId, clientSecret, accessToken, refreshToken, accessTokenExpiresAt, refreshTokenExpiresAt);
240
+ }
241
+ async prepare(encryptedStorage, launchOptions) {
242
+ return withTempBrowserContext(encryptedStorage, launchOptions ?? {}, async ({ context }) => {
243
+ const page = await context.newPage();
244
+ await page.goto(this.service.loginUrl);
245
+ await waitForGoogleLogin(page);
246
+ await showSpinnerPage(context, `Finalizing ${this.service.displayName} login...\nThis can take a few minutes.`);
247
+ const projectSlug = await createProject(page);
248
+ await enableApis(page, projectSlug);
249
+ await configureBranding(page, projectSlug);
250
+ const { clientId, clientSecret } = await createOAuthClient(page);
251
+ await page.close();
252
+ return new OAuthCredentials(clientId, clientSecret);
253
+ });
254
+ }
255
+ async performOAuthFlow(context, page, clientId, clientSecret) {
256
+ // Use an AbortController to signal the OAuth callback server to shut down
257
+ // when the browser is closed. Listen to both page and context close events
258
+ // to handle all cases where the user might close the browser.
259
+ const abortController = new AbortController();
260
+ const closeHandler = () => {
261
+ abortController.abort();
262
+ };
263
+ page.on('close', closeHandler);
264
+ context.on('close', closeHandler);
265
+ try {
266
+ // Start the callback server first to get the auto-assigned port
267
+ const { port, codePromise } = await startOAuthCallbackServer(LOGIN_TIMEOUT_MS, abortController.signal);
268
+ const redirectUri = `http://localhost:${port.toString()}/oauth2callback`;
269
+ const authUrl = new URL('https://accounts.google.com/o/oauth2/v2/auth');
270
+ authUrl.searchParams.set('client_id', clientId);
271
+ authUrl.searchParams.set('redirect_uri', redirectUri);
272
+ authUrl.searchParams.set('response_type', 'code');
273
+ authUrl.searchParams.set('scope', OAUTH_SCOPES.join(' '));
274
+ authUrl.searchParams.set('access_type', 'offline');
275
+ authUrl.searchParams.set('prompt', 'consent');
276
+ await page.goto(authUrl.toString());
277
+ // Wait for the authorization code from the callback
278
+ const code = await codePromise;
279
+ const tokens = exchangeCodeForTokens(GOOGLE_TOKEN_ENDPOINT, code, clientId, clientSecret, redirectUri);
280
+ const accessTokenExpiresAt = new Date(Date.now() + tokens.expires_in * 1000).toISOString();
281
+ // Google refresh tokens typically don't expire, so we don't set refreshTokenExpiresAt
282
+ return {
283
+ accessToken: tokens.access_token,
284
+ // exchangeCodeForTokens() guarantees refresh_token is present
285
+ refreshToken: tokens.refresh_token,
286
+ accessTokenExpiresAt,
287
+ };
288
+ }
289
+ catch (error) {
290
+ if (error instanceof Error && isBrowserClosedError(error)) {
291
+ throw new LoginCancelledError();
292
+ }
293
+ throw error;
294
+ }
295
+ finally {
296
+ // Remove the close handlers to prevent them from firing when context
297
+ // is closed normally during cleanup
298
+ page.off('close', closeHandler);
299
+ context.off('close', closeHandler);
300
+ }
301
+ }
302
+ }
303
+ export class Google extends Service {
304
+ name = 'google';
305
+ displayName = 'Google Workspace';
306
+ baseApiUrls = ['https://www.googleapis.com/'];
307
+ loginUrl = 'https://console.cloud.google.com/';
308
+ info = 'Supports some Google Workspace APIs: Gmail, Calendar, Drive, Sheets, Docs, and Contacts. ' +
309
+ 'If needed, run "latchkey auth browser-prepare google" to create an OAuth client first. ' +
310
+ 'It may take a few minutes before the OAuth client is ready to use.';
311
+ credentialCheckCurlArguments = [
312
+ 'https://www.googleapis.com/oauth2/v1/userinfo',
313
+ ];
314
+ getSession() {
315
+ return new GoogleServiceSession(this);
316
+ }
317
+ refreshCredentials(apiCredentials) {
318
+ if (!(apiCredentials instanceof OAuthCredentials)) {
319
+ return Promise.resolve(null);
320
+ }
321
+ if (!apiCredentials.refreshToken) {
322
+ return Promise.resolve(null);
323
+ }
324
+ const tokens = refreshAccessToken(GOOGLE_TOKEN_ENDPOINT, apiCredentials.refreshToken, apiCredentials.clientId, apiCredentials.clientSecret);
325
+ if (tokens === null) {
326
+ return Promise.resolve(null);
327
+ }
328
+ // Calculate access token expiration from the expires_in field
329
+ const accessTokenExpiresAt = new Date(Date.now() + tokens.expires_in * 1000).toISOString();
330
+ // Return new credentials with refreshed access token
331
+ // Keep the same refresh token unless a new one is provided
332
+ return Promise.resolve(new OAuthCredentials(apiCredentials.clientId, apiCredentials.clientSecret, tokens.access_token, tokens.refresh_token ?? apiCredentials.refreshToken, accessTokenExpiresAt, apiCredentials.refreshTokenExpiresAt));
333
+ }
334
+ }
335
+ export const GOOGLE = new Google();
336
+ //# sourceMappingURL=google.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"google.js","sourceRoot":"","sources":["../../../src/services/google.ts"],"names":[],"mappings":"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"}
@@ -1,12 +1,14 @@
1
1
  /**
2
2
  * Re-export all services.
3
3
  */
4
- export type { Service } from './base.js';
5
- export { ServiceSession, SimpleServiceSession, BrowserFollowupServiceSession } from './base.js';
4
+ export { Service, ServiceSession, SimpleServiceSession, BrowserFollowupServiceSession, } from './base.js';
6
5
  export { LoginCancelledError, LoginFailedError } from './base.js';
7
6
  export { Slack, SLACK } from './slack.js';
8
7
  export { Discord, DISCORD } from './discord.js';
9
8
  export { Github, GITHUB } from './github.js';
10
9
  export { Dropbox, DROPBOX } from './dropbox.js';
11
10
  export { Linear, LINEAR } from './linear.js';
11
+ export { Google, GOOGLE } from './google.js';
12
+ export { Notion, NOTION } from './notion.js';
13
+ export { Mailchimp, MAILCHIMP } from './mailchimp.js';
12
14
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/services/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,YAAY,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACzC,OAAO,EAAE,cAAc,EAAE,oBAAoB,EAAE,6BAA6B,EAAE,MAAM,WAAW,CAAC;AAChG,OAAO,EAAE,mBAAmB,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAElE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/services/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EACL,OAAO,EACP,cAAc,EACd,oBAAoB,EACpB,6BAA6B,GAC9B,MAAM,WAAW,CAAC;AACnB,OAAO,EAAE,mBAAmB,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAElE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC"}
@@ -1,11 +1,14 @@
1
1
  /**
2
2
  * Re-export all services.
3
3
  */
4
- export { ServiceSession, SimpleServiceSession, BrowserFollowupServiceSession } from './base.js';
4
+ export { Service, ServiceSession, SimpleServiceSession, BrowserFollowupServiceSession, } from './base.js';
5
5
  export { LoginCancelledError, LoginFailedError } from './base.js';
6
6
  export { Slack, SLACK } from './slack.js';
7
7
  export { Discord, DISCORD } from './discord.js';
8
8
  export { Github, GITHUB } from './github.js';
9
9
  export { Dropbox, DROPBOX } from './dropbox.js';
10
10
  export { Linear, LINEAR } from './linear.js';
11
+ export { Google, GOOGLE } from './google.js';
12
+ export { Notion, NOTION } from './notion.js';
13
+ export { Mailchimp, MAILCHIMP } from './mailchimp.js';
11
14
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/services/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,EAAE,cAAc,EAAE,oBAAoB,EAAE,6BAA6B,EAAE,MAAM,WAAW,CAAC;AAChG,OAAO,EAAE,mBAAmB,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAElE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/services/index.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EACL,OAAO,EACP,cAAc,EACd,oBAAoB,EACpB,6BAA6B,GAC9B,MAAM,WAAW,CAAC;AACnB,OAAO,EAAE,mBAAmB,EAAE,gBAAgB,EAAE,MAAM,WAAW,CAAC;AAElE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,YAAY,CAAC;AAC1C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,cAAc,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,aAAa,CAAC;AAC7C,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC"}
@@ -2,21 +2,22 @@
2
2
  * Linear service implementation.
3
3
  */
4
4
  import type { Response, BrowserContext } from 'playwright';
5
- import { ApiCredentialStatus, ApiCredentials } from '../apiCredentials.js';
5
+ import { ApiCredentials } from '../apiCredentials.js';
6
6
  import { Service, BrowserFollowupServiceSession } from './base.js';
7
7
  declare class LinearServiceSession extends BrowserFollowupServiceSession {
8
8
  private isLoggedIn;
9
9
  onResponse(response: Response): void;
10
10
  protected isLoginComplete(): boolean;
11
- protected performBrowserFollowup(context: BrowserContext): Promise<ApiCredentials | null>;
11
+ protected performBrowserFollowup(context: BrowserContext, _oldCredentials?: ApiCredentials): Promise<ApiCredentials | null>;
12
12
  }
13
- export declare class Linear implements Service {
13
+ export declare class Linear extends Service {
14
14
  readonly name = "linear";
15
+ readonly displayName = "Linear";
15
16
  readonly baseApiUrls: readonly ["https://api.linear.app/"];
16
17
  readonly loginUrl = "https://linear.app/imbue/settings/account/security/api-keys/new";
18
+ readonly info = "https://linear.app/developers/graphql";
17
19
  readonly credentialCheckCurlArguments: readonly ["-X", "POST", "-H", "Content-Type: application/json", "-d", "{\"query\": \"{ viewer { id } }\"}", "https://api.linear.app/graphql"];
18
20
  getSession(): LinearServiceSession;
19
- checkApiCredentials(apiCredentials: ApiCredentials): ApiCredentialStatus;
20
21
  }
21
22
  export declare const LINEAR: Linear;
22
23
  export {};
@@ -1 +1 @@
1
- {"version":3,"file":"linear.d.ts","sourceRoot":"","sources":["../../../src/services/linear.ts"],"names":[],"mappings":"AAAA;;GAEG;AAGH,OAAO,KAAK,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC3D,OAAO,EAAE,mBAAmB,EAAE,cAAc,EAAqB,MAAM,sBAAsB,CAAC;AAG9F,OAAO,EAAE,OAAO,EAAE,6BAA6B,EAAoB,MAAM,WAAW,CAAC;AAOrF,cAAM,oBAAqB,SAAQ,6BAA6B;IAC9D,OAAO,CAAC,UAAU,CAAS;IAE3B,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IA+BpC,SAAS,CAAC,eAAe,IAAI,OAAO;cAIpB,sBAAsB,CAAC,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC;CAgChG;AAED,qBAAa,MAAO,YAAW,OAAO;IACpC,QAAQ,CAAC,IAAI,YAAY;IACzB,QAAQ,CAAC,WAAW,uCAAwC;IAC5D,QAAQ,CAAC,QAAQ,qEAA0B;IAE3C,QAAQ,CAAC,4BAA4B,gJAQ1B;IAEX,UAAU,IAAI,oBAAoB;IAIlC,mBAAmB,CAAC,cAAc,EAAE,cAAc,GAAG,mBAAmB;CAwBzE;AAED,eAAO,MAAM,MAAM,QAAe,CAAC"}
1
+ {"version":3,"file":"linear.d.ts","sourceRoot":"","sources":["../../../src/services/linear.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,KAAK,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC3D,OAAO,EAAE,cAAc,EAAqB,MAAM,sBAAsB,CAAC;AAEzE,OAAO,EAAE,OAAO,EAAE,6BAA6B,EAAoB,MAAM,WAAW,CAAC;AAOrF,cAAM,oBAAqB,SAAQ,6BAA6B;IAC9D,OAAO,CAAC,UAAU,CAAS;IAE3B,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IA+BpC,SAAS,CAAC,eAAe,IAAI,OAAO;cAIpB,sBAAsB,CACpC,OAAO,EAAE,cAAc,EACvB,eAAe,CAAC,EAAE,cAAc,GAC/B,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC;CAgClC;AAED,qBAAa,MAAO,SAAQ,OAAO;IACjC,QAAQ,CAAC,IAAI,YAAY;IACzB,QAAQ,CAAC,WAAW,YAAY;IAChC,QAAQ,CAAC,WAAW,uCAAwC;IAC5D,QAAQ,CAAC,QAAQ,qEAA0B;IAC3C,QAAQ,CAAC,IAAI,2CAA2C;IAExD,QAAQ,CAAC,4BAA4B,gJAQ1B;IAEF,UAAU,IAAI,oBAAoB;CAG5C;AAED,eAAO,MAAM,MAAM,QAAe,CAAC"}
@@ -1,11 +1,9 @@
1
1
  /**
2
2
  * Linear service implementation.
3
3
  */
4
- import { randomUUID } from 'node:crypto';
5
- import { ApiCredentialStatus, AuthorizationBare } from '../apiCredentials.js';
6
- import { runCaptured } from '../curl.js';
7
- import { typeLikeHuman } from '../playwrightUtils.js';
8
- import { BrowserFollowupServiceSession, LoginFailedError } from './base.js';
4
+ import { AuthorizationBare } from '../apiCredentials.js';
5
+ import { generateLatchkeyAppName, typeLikeHuman } from '../playwrightUtils.js';
6
+ import { Service, BrowserFollowupServiceSession, LoginFailedError } from './base.js';
9
7
  const DEFAULT_TIMEOUT_MS = 8000;
10
8
  // URL for creating a new personal API key (also used as login URL)
11
9
  const LINEAR_NEW_API_KEY_URL = 'https://linear.app/imbue/settings/account/security/api-keys/new';
@@ -44,19 +42,19 @@ class LinearServiceSession extends BrowserFollowupServiceSession {
44
42
  isLoginComplete() {
45
43
  return this.isLoggedIn;
46
44
  }
47
- async performBrowserFollowup(context) {
45
+ async performBrowserFollowup(context, _oldCredentials) {
48
46
  const page = context.pages()[0];
49
47
  if (!page) {
50
48
  throw new LoginFailedError('No page available in browser context.');
51
49
  }
52
50
  await page.goto(LINEAR_NEW_API_KEY_URL);
53
51
  // Fill in the key name
54
- const keyName = `Latchkey-${randomUUID().slice(0, 8)}`;
55
- const keyNameInput = page.getByRole('textbox', { name: 'Key name' });
52
+ const keyName = generateLatchkeyAppName();
53
+ const keyNameInput = page.locator('//*[@id="label"]');
56
54
  await keyNameInput.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
57
55
  await typeLikeHuman(page, keyNameInput, keyName);
58
56
  // Click the Create button
59
- const createButton = page.getByRole('button', { name: 'Create' });
57
+ const createButton = page.locator('button[type="submit"]');
60
58
  await createButton.waitFor({ timeout: DEFAULT_TIMEOUT_MS });
61
59
  await createButton.click();
62
60
  // Wait for and extract the token from span element containing lin_api_ prefix
@@ -70,10 +68,12 @@ class LinearServiceSession extends BrowserFollowupServiceSession {
70
68
  return new AuthorizationBare(token);
71
69
  }
72
70
  }
73
- export class Linear {
71
+ export class Linear extends Service {
74
72
  name = 'linear';
73
+ displayName = 'Linear';
75
74
  baseApiUrls = ['https://api.linear.app/'];
76
75
  loginUrl = LINEAR_NEW_API_KEY_URL;
76
+ info = 'https://linear.app/developers/graphql';
77
77
  credentialCheckCurlArguments = [
78
78
  '-X',
79
79
  'POST',
@@ -86,25 +86,6 @@ export class Linear {
86
86
  getSession() {
87
87
  return new LinearServiceSession(this);
88
88
  }
89
- checkApiCredentials(apiCredentials) {
90
- if (!(apiCredentials instanceof AuthorizationBare)) {
91
- return ApiCredentialStatus.Invalid;
92
- }
93
- // Linear uses GraphQL API - check credentials with a simple query
94
- const result = runCaptured([
95
- '-s',
96
- '-o',
97
- '/dev/null',
98
- '-w',
99
- '%{http_code}',
100
- ...apiCredentials.asCurlArguments(),
101
- ...this.credentialCheckCurlArguments,
102
- ], 10);
103
- if (result.stdout === '200') {
104
- return ApiCredentialStatus.Valid;
105
- }
106
- return ApiCredentialStatus.Invalid;
107
- }
108
89
  }
109
90
  export const LINEAR = new Linear();
110
91
  //# sourceMappingURL=linear.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"linear.js","sourceRoot":"","sources":["../../../src/services/linear.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"linear.js","sourceRoot":"","sources":["../../../src/services/linear.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ import { Service } from './base.js';
2
+ export declare class Mailchimp extends Service {
3
+ readonly name = "mailchimp";
4
+ readonly displayName = "Mailchimp";
5
+ readonly baseApiUrls: readonly ["https://api.mailchimp.com/", RegExp];
6
+ readonly loginUrl = "https://login.mailchimp.com/";
7
+ readonly info: string;
8
+ readonly credentialCheckCurlArguments: readonly ["https://login.mailchimp.com/oauth2/metadata"];
9
+ }
10
+ export declare const MAILCHIMP: Mailchimp;
11
+ //# sourceMappingURL=mailchimp.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mailchimp.d.ts","sourceRoot":"","sources":["../../../src/services/mailchimp.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAEpC,qBAAa,SAAU,SAAQ,OAAO;IACpC,QAAQ,CAAC,IAAI,eAAe;IAC5B,QAAQ,CAAC,WAAW,eAAe;IACnC,QAAQ,CAAC,WAAW,kDAGT;IACX,QAAQ,CAAC,QAAQ,kCAAkC;IACnD,QAAQ,CAAC,IAAI,SAGyF;IAEtG,QAAQ,CAAC,4BAA4B,2DAA4D;CAClG;AAED,eAAO,MAAM,SAAS,WAAkB,CAAC"}
@@ -0,0 +1,16 @@
1
+ import { Service } from './base.js';
2
+ export class Mailchimp extends Service {
3
+ name = 'mailchimp';
4
+ displayName = 'Mailchimp';
5
+ baseApiUrls = [
6
+ 'https://api.mailchimp.com/',
7
+ /^https:\/\/[^/]+\.api\.mailchimp\.com\//,
8
+ ];
9
+ loginUrl = 'https://login.mailchimp.com/';
10
+ info = 'https://mailchimp.com/developer/marketing/api/. ' +
11
+ 'Browser-based authentication is not supported. ' +
12
+ 'Use `latchkey auth set mailchimp -H "Authorization: Bearer <token>"` to add credentials manually.';
13
+ credentialCheckCurlArguments = ['https://login.mailchimp.com/oauth2/metadata'];
14
+ }
15
+ export const MAILCHIMP = new Mailchimp();
16
+ //# sourceMappingURL=mailchimp.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mailchimp.js","sourceRoot":"","sources":["../../../src/services/mailchimp.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAEpC,MAAM,OAAO,SAAU,SAAQ,OAAO;IAC3B,IAAI,GAAG,WAAW,CAAC;IACnB,WAAW,GAAG,WAAW,CAAC;IAC1B,WAAW,GAAG;QACrB,4BAA4B;QAC5B,yCAAyC;KACjC,CAAC;IACF,QAAQ,GAAG,8BAA8B,CAAC;IAC1C,IAAI,GACX,kDAAkD;QAClD,iDAAiD;QACjD,mGAAmG,CAAC;IAE7F,4BAA4B,GAAG,CAAC,6CAA6C,CAAU,CAAC;CAClG;AAED,MAAM,CAAC,MAAM,SAAS,GAAG,IAAI,SAAS,EAAE,CAAC"}
@@ -0,0 +1,29 @@
1
+ /**
2
+ * Notion service implementation.
3
+ *
4
+ * This has some severe limitations:
5
+ *
6
+ * - It requires the UI to be in English.
7
+ * - It only grants access to the private pages that existed at the time of login.
8
+ */
9
+ import type { Response, BrowserContext } from 'playwright';
10
+ import { ApiCredentials } from '../apiCredentials.js';
11
+ import { Service, BrowserFollowupServiceSession } from './base.js';
12
+ declare class NotionServiceSession extends BrowserFollowupServiceSession {
13
+ private isLoggedIn;
14
+ onResponse(response: Response): void;
15
+ protected isLoginComplete(): boolean;
16
+ protected performBrowserFollowup(context: BrowserContext): Promise<ApiCredentials | null>;
17
+ }
18
+ export declare class Notion extends Service {
19
+ readonly name = "notion";
20
+ readonly displayName = "Notion";
21
+ readonly baseApiUrls: readonly ["https://api.notion.com/"];
22
+ readonly loginUrl = "https://www.notion.so/profile/integrations/form/new-integration";
23
+ readonly info: string;
24
+ readonly credentialCheckCurlArguments: readonly ["-H", "Notion-Version: 2022-06-28", "https://api.notion.com/v1/users/me"];
25
+ getSession(): NotionServiceSession;
26
+ }
27
+ export declare const NOTION: Notion;
28
+ export {};
29
+ //# sourceMappingURL=notion.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"notion.d.ts","sourceRoot":"","sources":["../../../src/services/notion.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AAEH,OAAO,KAAK,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC3D,OAAO,EAAE,cAAc,EAAuB,MAAM,sBAAsB,CAAC;AAE3E,OAAO,EAAE,OAAO,EAAE,6BAA6B,EAAoB,MAAM,WAAW,CAAC;AAMrF,cAAM,oBAAqB,SAAQ,6BAA6B;IAC9D,OAAO,CAAC,UAAU,CAAS;IAE3B,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IASpC,SAAS,CAAC,eAAe,IAAI,OAAO;cAIpB,sBAAsB,CAAC,OAAO,EAAE,cAAc,GAAG,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC;CAgEhG;AAED,qBAAa,MAAO,SAAQ,OAAO;IACjC,QAAQ,CAAC,IAAI,YAAY;IACzB,QAAQ,CAAC,WAAW,YAAY;IAChC,QAAQ,CAAC,WAAW,uCAAwC;IAC5D,QAAQ,CAAC,QAAQ,qEAA2B;IAC5C,QAAQ,CAAC,IAAI,SAG2E;IAExF,QAAQ,CAAC,4BAA4B,sFAI1B;IAEF,UAAU,IAAI,oBAAoB;CAG5C;AAED,eAAO,MAAM,MAAM,QAAe,CAAC"}