@stackmemoryai/stackmemory 0.2.4 → 0.2.6

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 (179) hide show
  1. package/README.md +108 -0
  2. package/dist/index.js +382 -0
  3. package/dist/src/analytics/api/analytics-api.d.ts +24 -0
  4. package/dist/src/analytics/api/analytics-api.d.ts.map +1 -0
  5. package/dist/src/analytics/api/analytics-api.js +279 -0
  6. package/dist/src/analytics/api/analytics-api.js.map +1 -0
  7. package/dist/src/analytics/core/analytics-service.d.ts +23 -0
  8. package/dist/src/analytics/core/analytics-service.d.ts.map +1 -0
  9. package/dist/src/analytics/core/analytics-service.js +160 -0
  10. package/dist/src/analytics/core/analytics-service.js.map +1 -0
  11. package/dist/src/analytics/index.d.ts +12 -0
  12. package/dist/src/analytics/index.d.ts.map +1 -0
  13. package/dist/src/analytics/index.js +11 -0
  14. package/dist/src/analytics/index.js.map +1 -0
  15. package/dist/src/analytics/queries/metrics-queries.d.ts +11 -0
  16. package/dist/src/analytics/queries/metrics-queries.d.ts.map +1 -0
  17. package/dist/src/analytics/queries/metrics-queries.js +179 -0
  18. package/dist/src/analytics/queries/metrics-queries.js.map +1 -0
  19. package/dist/src/analytics/types/metrics.d.ts +60 -0
  20. package/dist/src/analytics/types/metrics.d.ts.map +1 -0
  21. package/dist/src/analytics/types/metrics.js +2 -0
  22. package/dist/src/analytics/types/metrics.js.map +1 -0
  23. package/dist/src/cli/analytics-viewer.d.ts +3 -0
  24. package/dist/src/cli/analytics-viewer.d.ts.map +1 -0
  25. package/dist/src/cli/analytics-viewer.js +89 -0
  26. package/dist/src/cli/analytics-viewer.js.map +1 -0
  27. package/dist/src/cli/browser-test.d.ts +6 -0
  28. package/dist/src/cli/browser-test.d.ts.map +1 -0
  29. package/dist/src/cli/browser-test.js +32 -0
  30. package/dist/src/cli/browser-test.js.map +1 -0
  31. package/dist/src/cli/cli.js +157 -0
  32. package/dist/src/cli/cli.js.map +1 -1
  33. package/dist/src/cli/commands/projects.d.ts +8 -0
  34. package/dist/src/cli/commands/projects.d.ts.map +1 -0
  35. package/dist/src/cli/commands/projects.js +220 -0
  36. package/dist/src/cli/commands/projects.js.map +1 -0
  37. package/dist/src/cli/index.d.ts +7 -0
  38. package/dist/src/cli/index.d.ts.map +1 -0
  39. package/dist/src/cli/index.js +704 -0
  40. package/dist/src/cli/index.js.map +1 -0
  41. package/dist/src/cli/project-commands.d.ts +8 -0
  42. package/dist/src/cli/project-commands.d.ts.map +1 -0
  43. package/dist/src/cli/project-commands.js +212 -0
  44. package/dist/src/cli/project-commands.js.map +1 -0
  45. package/dist/src/cli/utils/viewer.d.ts +3 -0
  46. package/dist/src/cli/utils/viewer.d.ts.map +1 -0
  47. package/dist/src/cli/utils/viewer.js +89 -0
  48. package/dist/src/cli/utils/viewer.js.map +1 -0
  49. package/dist/src/core/context/frame-manager.d.ts +106 -0
  50. package/dist/src/core/context/frame-manager.d.ts.map +1 -0
  51. package/dist/src/core/context/frame-manager.js +387 -0
  52. package/dist/src/core/context/frame-manager.js.map +1 -0
  53. package/dist/src/core/logger.test.js +1 -1
  54. package/dist/src/core/logger.test.js.map +1 -1
  55. package/dist/src/core/monitoring/error-handler.d.ts +46 -0
  56. package/dist/src/core/monitoring/error-handler.d.ts.map +1 -0
  57. package/dist/src/core/monitoring/error-handler.js +212 -0
  58. package/dist/src/core/monitoring/error-handler.js.map +1 -0
  59. package/dist/src/core/monitoring/logger.d.ts +24 -0
  60. package/dist/src/core/monitoring/logger.d.ts.map +1 -0
  61. package/dist/src/core/monitoring/logger.js +121 -0
  62. package/dist/src/core/monitoring/logger.js.map +1 -0
  63. package/dist/src/core/monitoring/metrics.d.ts +7 -0
  64. package/dist/src/core/monitoring/metrics.d.ts.map +1 -0
  65. package/dist/src/core/monitoring/metrics.js +13 -0
  66. package/dist/src/core/monitoring/metrics.js.map +1 -0
  67. package/dist/src/core/monitoring/progress-tracker.d.ts +95 -0
  68. package/dist/src/core/monitoring/progress-tracker.d.ts.map +1 -0
  69. package/dist/src/core/monitoring/progress-tracker.js +178 -0
  70. package/dist/src/core/monitoring/progress-tracker.js.map +1 -0
  71. package/dist/src/core/project-manager.d.ts +130 -0
  72. package/dist/src/core/project-manager.d.ts.map +1 -0
  73. package/dist/src/core/project-manager.js +582 -0
  74. package/dist/src/core/project-manager.js.map +1 -0
  75. package/dist/src/core/projects/project-manager.d.ts +130 -0
  76. package/dist/src/core/projects/project-manager.d.ts.map +1 -0
  77. package/dist/src/core/projects/project-manager.js +591 -0
  78. package/dist/src/core/projects/project-manager.js.map +1 -0
  79. package/dist/src/core/utils/update-checker.d.ts +38 -0
  80. package/dist/src/core/utils/update-checker.d.ts.map +1 -0
  81. package/dist/src/core/utils/update-checker.js +156 -0
  82. package/dist/src/core/utils/update-checker.js.map +1 -0
  83. package/dist/src/features/analytics/api/analytics-api.d.ts +24 -0
  84. package/dist/src/features/analytics/api/analytics-api.d.ts.map +1 -0
  85. package/dist/src/features/analytics/api/analytics-api.js +289 -0
  86. package/dist/src/features/analytics/api/analytics-api.js.map +1 -0
  87. package/dist/src/features/analytics/core/analytics-service.d.ts +23 -0
  88. package/dist/src/features/analytics/core/analytics-service.d.ts.map +1 -0
  89. package/dist/src/features/analytics/core/analytics-service.js +160 -0
  90. package/dist/src/features/analytics/core/analytics-service.js.map +1 -0
  91. package/dist/src/features/analytics/index.d.ts +12 -0
  92. package/dist/src/features/analytics/index.d.ts.map +1 -0
  93. package/dist/src/features/analytics/index.js +11 -0
  94. package/dist/src/features/analytics/index.js.map +1 -0
  95. package/dist/src/features/analytics/queries/metrics-queries.d.ts +11 -0
  96. package/dist/src/features/analytics/queries/metrics-queries.d.ts.map +1 -0
  97. package/dist/src/features/analytics/queries/metrics-queries.js +183 -0
  98. package/dist/src/features/analytics/queries/metrics-queries.js.map +1 -0
  99. package/dist/src/features/analytics/types/metrics.d.ts +60 -0
  100. package/dist/src/features/analytics/types/metrics.d.ts.map +1 -0
  101. package/dist/src/features/analytics/types/metrics.js +2 -0
  102. package/dist/src/features/analytics/types/metrics.js.map +1 -0
  103. package/dist/src/features/browser/browser-mcp.d.ts +94 -0
  104. package/dist/src/features/browser/browser-mcp.d.ts.map +1 -0
  105. package/dist/src/features/browser/browser-mcp.js +456 -0
  106. package/dist/src/features/browser/browser-mcp.js.map +1 -0
  107. package/dist/src/features/tasks/pebbles-task-store.d.ts +117 -0
  108. package/dist/src/features/tasks/pebbles-task-store.d.ts.map +1 -0
  109. package/dist/src/features/tasks/pebbles-task-store.js +335 -0
  110. package/dist/src/features/tasks/pebbles-task-store.js.map +1 -0
  111. package/dist/src/features/tasks/task-aware-context.d.ts +103 -0
  112. package/dist/src/features/tasks/task-aware-context.d.ts.map +1 -0
  113. package/dist/src/features/tasks/task-aware-context.js +412 -0
  114. package/dist/src/features/tasks/task-aware-context.js.map +1 -0
  115. package/dist/src/index.d.ts +4 -4
  116. package/dist/src/index.d.ts.map +1 -1
  117. package/dist/src/index.js +4 -4
  118. package/dist/src/index.js.map +1 -1
  119. package/dist/src/integrations/browser-mcp.d.ts +94 -0
  120. package/dist/src/integrations/browser-mcp.d.ts.map +1 -0
  121. package/dist/src/integrations/browser-mcp.js +431 -0
  122. package/dist/src/integrations/browser-mcp.js.map +1 -0
  123. package/dist/src/integrations/linear/auth.d.ts +99 -0
  124. package/dist/src/integrations/linear/auth.d.ts.map +1 -0
  125. package/dist/src/integrations/linear/auth.js +319 -0
  126. package/dist/src/integrations/linear/auth.js.map +1 -0
  127. package/dist/src/integrations/linear/auto-sync.d.ts +77 -0
  128. package/dist/src/integrations/linear/auto-sync.d.ts.map +1 -0
  129. package/dist/src/integrations/linear/auto-sync.js +268 -0
  130. package/dist/src/integrations/linear/auto-sync.js.map +1 -0
  131. package/dist/src/integrations/linear/client.d.ts +86 -0
  132. package/dist/src/integrations/linear/client.d.ts.map +1 -0
  133. package/dist/src/integrations/linear/client.js +277 -0
  134. package/dist/src/integrations/linear/client.js.map +1 -0
  135. package/dist/src/integrations/linear/config.d.ts +51 -0
  136. package/dist/src/integrations/linear/config.d.ts.map +1 -0
  137. package/dist/src/integrations/linear/config.js +103 -0
  138. package/dist/src/integrations/linear/config.js.map +1 -0
  139. package/dist/src/integrations/linear/sync.d.ts +97 -0
  140. package/dist/src/integrations/linear/sync.d.ts.map +1 -0
  141. package/dist/src/integrations/linear/sync.js +391 -0
  142. package/dist/src/integrations/linear/sync.js.map +1 -0
  143. package/dist/src/integrations/mcp/server.d.ts +40 -0
  144. package/dist/src/integrations/mcp/server.d.ts.map +1 -0
  145. package/dist/src/integrations/mcp/server.js +828 -0
  146. package/dist/src/integrations/mcp/server.js.map +1 -0
  147. package/dist/src/mcp/mcp-server.d.ts +1 -0
  148. package/dist/src/mcp/mcp-server.d.ts.map +1 -1
  149. package/dist/src/mcp/mcp-server.js +11 -0
  150. package/dist/src/mcp/mcp-server.js.map +1 -1
  151. package/dist/src/railway/index.d.ts +7 -0
  152. package/dist/src/railway/index.d.ts.map +1 -0
  153. package/dist/src/railway/index.js +401 -0
  154. package/dist/src/railway/index.js.map +1 -0
  155. package/dist/src/runway/auth/auth-middleware.d.ts +66 -0
  156. package/dist/src/runway/auth/auth-middleware.d.ts.map +1 -0
  157. package/dist/src/runway/auth/auth-middleware.js +337 -0
  158. package/dist/src/runway/auth/auth-middleware.js.map +1 -0
  159. package/dist/src/runway/server/runway-mcp-server.d.ts +46 -0
  160. package/dist/src/runway/server/runway-mcp-server.d.ts.map +1 -0
  161. package/dist/src/runway/server/runway-mcp-server.js +601 -0
  162. package/dist/src/runway/server/runway-mcp-server.js.map +1 -0
  163. package/dist/src/runway.bak/auth/auth-middleware.d.ts +66 -0
  164. package/dist/src/runway.bak/auth/auth-middleware.d.ts.map +1 -0
  165. package/dist/src/runway.bak/auth/auth-middleware.js +337 -0
  166. package/dist/src/runway.bak/auth/auth-middleware.js.map +1 -0
  167. package/dist/src/runway.bak/server/runway-mcp-server.d.ts +46 -0
  168. package/dist/src/runway.bak/server/runway-mcp-server.d.ts.map +1 -0
  169. package/dist/src/runway.bak/server/runway-mcp-server.js +601 -0
  170. package/dist/src/runway.bak/server/runway-mcp-server.js.map +1 -0
  171. package/dist/src/servers/production/auth-middleware.d.ts +66 -0
  172. package/dist/src/servers/production/auth-middleware.d.ts.map +1 -0
  173. package/dist/src/servers/production/auth-middleware.js +346 -0
  174. package/dist/src/servers/production/auth-middleware.js.map +1 -0
  175. package/dist/src/servers/railway/index.d.ts +7 -0
  176. package/dist/src/servers/railway/index.d.ts.map +1 -0
  177. package/dist/src/servers/railway/index.js +401 -0
  178. package/dist/src/servers/railway/index.js.map +1 -0
  179. package/package.json +27 -5
@@ -0,0 +1,319 @@
1
+ /**
2
+ * Linear OAuth Authentication Setup
3
+ * Handles initial OAuth flow and token management for Linear integration
4
+ */
5
+ import { createHash, randomBytes } from 'crypto';
6
+ import { readFileSync, writeFileSync, existsSync } from 'fs';
7
+ import { join } from 'path';
8
+ import { logger } from '../../core/monitoring/logger.js';
9
+ export class LinearAuthManager {
10
+ configPath;
11
+ tokensPath;
12
+ config;
13
+ constructor(projectRoot) {
14
+ const configDir = join(projectRoot, '.stackmemory');
15
+ this.configPath = join(configDir, 'linear-config.json');
16
+ this.tokensPath = join(configDir, 'linear-tokens.json');
17
+ }
18
+ /**
19
+ * Check if Linear integration is configured
20
+ */
21
+ isConfigured() {
22
+ return existsSync(this.configPath) && existsSync(this.tokensPath);
23
+ }
24
+ /**
25
+ * Save OAuth application configuration
26
+ */
27
+ saveConfig(config) {
28
+ writeFileSync(this.configPath, JSON.stringify(config, null, 2));
29
+ this.config = config;
30
+ logger.info('Linear OAuth configuration saved');
31
+ }
32
+ /**
33
+ * Load OAuth configuration
34
+ */
35
+ loadConfig() {
36
+ if (!existsSync(this.configPath)) {
37
+ return null;
38
+ }
39
+ try {
40
+ const configData = readFileSync(this.configPath, 'utf8');
41
+ this.config = JSON.parse(configData);
42
+ return this.config;
43
+ }
44
+ catch (error) {
45
+ logger.error('Failed to load Linear configuration:', error);
46
+ return null;
47
+ }
48
+ }
49
+ /**
50
+ * Generate OAuth authorization URL with PKCE
51
+ */
52
+ generateAuthUrl(state) {
53
+ if (!this.config) {
54
+ throw new Error('Linear OAuth configuration not loaded');
55
+ }
56
+ // Generate PKCE parameters
57
+ const codeVerifier = randomBytes(32).toString('base64url');
58
+ const codeChallenge = createHash('sha256')
59
+ .update(codeVerifier)
60
+ .digest('base64url');
61
+ const params = new URLSearchParams({
62
+ client_id: this.config.clientId,
63
+ redirect_uri: this.config.redirectUri,
64
+ response_type: 'code',
65
+ scope: this.config.scopes.join(' '),
66
+ code_challenge: codeChallenge,
67
+ code_challenge_method: 'S256',
68
+ actor: 'app', // Enable actor authorization for service accounts
69
+ });
70
+ if (state) {
71
+ params.set('state', state);
72
+ }
73
+ const authUrl = `https://linear.app/oauth/authorize?${params.toString()}`;
74
+ return { url: authUrl, codeVerifier };
75
+ }
76
+ /**
77
+ * Exchange authorization code for access token
78
+ */
79
+ async exchangeCodeForToken(authCode, codeVerifier) {
80
+ if (!this.config) {
81
+ throw new Error('Linear OAuth configuration not loaded');
82
+ }
83
+ const tokenUrl = 'https://api.linear.app/oauth/token';
84
+ const body = new URLSearchParams({
85
+ grant_type: 'authorization_code',
86
+ client_id: this.config.clientId,
87
+ client_secret: this.config.clientSecret,
88
+ redirect_uri: this.config.redirectUri,
89
+ code: authCode,
90
+ code_verifier: codeVerifier,
91
+ });
92
+ const response = await fetch(tokenUrl, {
93
+ method: 'POST',
94
+ headers: {
95
+ 'Content-Type': 'application/x-www-form-urlencoded',
96
+ Accept: 'application/json',
97
+ },
98
+ body: body.toString(),
99
+ });
100
+ if (!response.ok) {
101
+ const errorText = await response.text();
102
+ throw new Error(`Token exchange failed: ${response.status} ${errorText}`);
103
+ }
104
+ const result = (await response.json());
105
+ // Calculate expiration time (tokens expire in 24 hours)
106
+ const expiresAt = Date.now() + result.expiresIn * 1000;
107
+ const tokens = {
108
+ accessToken: result.accessToken,
109
+ refreshToken: result.refreshToken,
110
+ expiresAt,
111
+ scope: result.scope.split(' '),
112
+ };
113
+ this.saveTokens(tokens);
114
+ return tokens;
115
+ }
116
+ /**
117
+ * Refresh access token using refresh token
118
+ */
119
+ async refreshAccessToken() {
120
+ if (!this.config) {
121
+ throw new Error('Linear OAuth configuration not loaded');
122
+ }
123
+ const currentTokens = this.loadTokens();
124
+ if (!currentTokens?.refreshToken) {
125
+ throw new Error('No refresh token available');
126
+ }
127
+ const tokenUrl = 'https://api.linear.app/oauth/token';
128
+ const body = new URLSearchParams({
129
+ grant_type: 'refresh_token',
130
+ client_id: this.config.clientId,
131
+ client_secret: this.config.clientSecret,
132
+ refresh_token: currentTokens.refreshToken,
133
+ });
134
+ const response = await fetch(tokenUrl, {
135
+ method: 'POST',
136
+ headers: {
137
+ 'Content-Type': 'application/x-www-form-urlencoded',
138
+ Accept: 'application/json',
139
+ },
140
+ body: body.toString(),
141
+ });
142
+ if (!response.ok) {
143
+ const errorText = await response.text();
144
+ throw new Error(`Token refresh failed: ${response.status} ${errorText}`);
145
+ }
146
+ const result = (await response.json());
147
+ const tokens = {
148
+ accessToken: result.accessToken,
149
+ refreshToken: result.refreshToken || currentTokens.refreshToken,
150
+ expiresAt: Date.now() + result.expiresIn * 1000,
151
+ scope: result.scope.split(' '),
152
+ };
153
+ this.saveTokens(tokens);
154
+ return tokens;
155
+ }
156
+ /**
157
+ * Get valid access token (refresh if needed)
158
+ */
159
+ async getValidToken() {
160
+ const tokens = this.loadTokens();
161
+ if (!tokens) {
162
+ throw new Error('No Linear tokens found. Please complete OAuth setup.');
163
+ }
164
+ // Check if token expires in next 5 minutes
165
+ const fiveMinutes = 5 * 60 * 1000;
166
+ if (tokens.expiresAt - Date.now() < fiveMinutes) {
167
+ logger.info('Linear token expiring soon, refreshing...');
168
+ const newTokens = await this.refreshAccessToken();
169
+ return newTokens.accessToken;
170
+ }
171
+ return tokens.accessToken;
172
+ }
173
+ /**
174
+ * Save tokens to file
175
+ */
176
+ saveTokens(tokens) {
177
+ writeFileSync(this.tokensPath, JSON.stringify(tokens, null, 2));
178
+ logger.info('Linear tokens saved');
179
+ }
180
+ /**
181
+ * Load tokens from file
182
+ */
183
+ loadTokens() {
184
+ if (!existsSync(this.tokensPath)) {
185
+ return null;
186
+ }
187
+ try {
188
+ const tokensData = readFileSync(this.tokensPath, 'utf8');
189
+ return JSON.parse(tokensData);
190
+ }
191
+ catch (error) {
192
+ logger.error('Failed to load Linear tokens:', error);
193
+ return null;
194
+ }
195
+ }
196
+ /**
197
+ * Clear stored tokens and config
198
+ */
199
+ clearAuth() {
200
+ if (existsSync(this.tokensPath)) {
201
+ writeFileSync(this.tokensPath, '');
202
+ }
203
+ if (existsSync(this.configPath)) {
204
+ writeFileSync(this.configPath, '');
205
+ }
206
+ logger.info('Linear authentication cleared');
207
+ }
208
+ }
209
+ /**
210
+ * Default Linear OAuth scopes for task management
211
+ */
212
+ export const DEFAULT_LINEAR_SCOPES = [
213
+ 'read', // Read issues, projects, teams
214
+ 'write', // Create and update issues
215
+ 'admin', // Manage team settings and workflows
216
+ ];
217
+ /**
218
+ * Linear OAuth setup helper
219
+ */
220
+ export class LinearOAuthSetup {
221
+ authManager;
222
+ constructor(projectRoot) {
223
+ this.authManager = new LinearAuthManager(projectRoot);
224
+ }
225
+ /**
226
+ * Interactive setup for Linear OAuth
227
+ */
228
+ async setupInteractive() {
229
+ // For now, we'll provide manual setup instructions
230
+ // In a full implementation, this could open a browser or use a local server
231
+ const config = {
232
+ clientId: process.env.LINEAR_CLIENT_ID || '',
233
+ clientSecret: process.env.LINEAR_CLIENT_SECRET || '',
234
+ redirectUri: process.env.LINEAR_REDIRECT_URI ||
235
+ 'http://localhost:3456/auth/linear/callback',
236
+ scopes: DEFAULT_LINEAR_SCOPES,
237
+ };
238
+ if (!config.clientId || !config.clientSecret) {
239
+ return {
240
+ authUrl: '',
241
+ instructions: [
242
+ '1. Create a Linear OAuth application at https://linear.app/settings/api',
243
+ '2. Set redirect URI to: http://localhost:3456/auth/linear/callback',
244
+ '3. Copy your Client ID and Client Secret',
245
+ '4. Set environment variables:',
246
+ ' export LINEAR_CLIENT_ID="your_client_id"',
247
+ ' export LINEAR_CLIENT_SECRET="your_client_secret"',
248
+ '5. Re-run this setup command',
249
+ ],
250
+ };
251
+ }
252
+ this.authManager.saveConfig(config);
253
+ const { url, codeVerifier } = this.authManager.generateAuthUrl();
254
+ // Store code verifier temporarily (in a real app, this would be in a secure session store)
255
+ process.env._LINEAR_CODE_VERIFIER = codeVerifier;
256
+ return {
257
+ authUrl: url,
258
+ instructions: [
259
+ '1. Open this URL in your browser:',
260
+ url,
261
+ '',
262
+ '2. Approve the StackMemory integration',
263
+ '3. Copy the authorization code from the redirect URL',
264
+ '4. Run: stackmemory linear authorize <code>',
265
+ ],
266
+ };
267
+ }
268
+ /**
269
+ * Complete OAuth flow with authorization code
270
+ */
271
+ async completeAuth(authCode) {
272
+ try {
273
+ const codeVerifier = process.env._LINEAR_CODE_VERIFIER;
274
+ if (!codeVerifier) {
275
+ throw new Error('Code verifier not found. Please restart the setup process.');
276
+ }
277
+ await this.authManager.exchangeCodeForToken(authCode, codeVerifier);
278
+ delete process.env._LINEAR_CODE_VERIFIER;
279
+ logger.info('Linear OAuth setup completed successfully!');
280
+ return true;
281
+ }
282
+ catch (error) {
283
+ logger.error('Failed to complete Linear OAuth setup:', error);
284
+ return false;
285
+ }
286
+ }
287
+ /**
288
+ * Test the Linear connection
289
+ */
290
+ async testConnection() {
291
+ try {
292
+ const token = await this.authManager.getValidToken();
293
+ // Test with a simple GraphQL query
294
+ const response = await fetch('https://api.linear.app/graphql', {
295
+ method: 'POST',
296
+ headers: {
297
+ Authorization: `Bearer ${token}`,
298
+ 'Content-Type': 'application/json',
299
+ },
300
+ body: JSON.stringify({
301
+ query: 'query { viewer { id name email } }',
302
+ }),
303
+ });
304
+ if (response.ok) {
305
+ const result = (await response.json());
306
+ if (result.data?.viewer) {
307
+ logger.info(`Connected to Linear as: ${result.data.viewer.name} (${result.data.viewer.email})`);
308
+ return true;
309
+ }
310
+ }
311
+ return false;
312
+ }
313
+ catch (error) {
314
+ logger.error('Linear connection test failed:', error);
315
+ return false;
316
+ }
317
+ }
318
+ }
319
+ //# sourceMappingURL=auth.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auth.js","sourceRoot":"","sources":["../../../../src/integrations/linear/auth.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,UAAU,EAAE,WAAW,EAAE,MAAM,QAAQ,CAAC;AACjD,OAAO,EAAE,YAAY,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,IAAI,CAAC;AAC7D,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,EAAE,MAAM,EAAE,MAAM,iCAAiC,CAAC;AAwBzD,MAAM,OAAO,iBAAiB;IACpB,UAAU,CAAS;IACnB,UAAU,CAAS;IACnB,MAAM,CAAoB;IAElC,YAAY,WAAmB;QAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,cAAc,CAAC,CAAC;QACpD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,SAAS,EAAE,oBAAoB,CAAC,CAAC;QACxD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,SAAS,EAAE,oBAAoB,CAAC,CAAC;IAC1D,CAAC;IAED;;OAEG;IACH,YAAY;QACV,OAAO,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACpE,CAAC;IAED;;OAEG;IACH,UAAU,CAAC,MAAwB;QACjC,aAAa,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAChE,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;QACrB,MAAM,CAAC,IAAI,CAAC,kCAAkC,CAAC,CAAC;IAClD,CAAC;IAED;;OAEG;IACH,UAAU;QACR,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACjC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;YACzD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;YACrC,OAAO,IAAI,CAAC,MAAO,CAAC;QACtB,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,sCAAsC,EAAE,KAAc,CAAC,CAAC;YACrE,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACH,eAAe,CAAC,KAAc;QAC5B,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;QAC3D,CAAC;QAED,2BAA2B;QAC3B,MAAM,YAAY,GAAG,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;QAC3D,MAAM,aAAa,GAAG,UAAU,CAAC,QAAQ,CAAC;aACvC,MAAM,CAAC,YAAY,CAAC;aACpB,MAAM,CAAC,WAAW,CAAC,CAAC;QAEvB,MAAM,MAAM,GAAG,IAAI,eAAe,CAAC;YACjC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;YAC/B,YAAY,EAAE,IAAI,CAAC,MAAM,CAAC,WAAW;YACrC,aAAa,EAAE,MAAM;YACrB,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC;YACnC,cAAc,EAAE,aAAa;YAC7B,qBAAqB,EAAE,MAAM;YAC7B,KAAK,EAAE,KAAK,EAAE,kDAAkD;SACjE,CAAC,CAAC;QAEH,IAAI,KAAK,EAAE,CAAC;YACV,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAC7B,CAAC;QAED,MAAM,OAAO,GAAG,sCAAsC,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC;QAE1E,OAAO,EAAE,GAAG,EAAE,OAAO,EAAE,YAAY,EAAE,CAAC;IACxC,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,oBAAoB,CACxB,QAAgB,EAChB,YAAoB;QAEpB,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;QAC3D,CAAC;QAED,MAAM,QAAQ,GAAG,oCAAoC,CAAC;QAEtD,MAAM,IAAI,GAAG,IAAI,eAAe,CAAC;YAC/B,UAAU,EAAE,oBAAoB;YAChC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;YAC/B,aAAa,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY;YACvC,YAAY,EAAE,IAAI,CAAC,MAAM,CAAC,WAAW;YACrC,IAAI,EAAE,QAAQ;YACd,aAAa,EAAE,YAAY;SAC5B,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,QAAQ,EAAE;YACrC,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,mCAAmC;gBACnD,MAAM,EAAE,kBAAkB;aAC3B;YACD,IAAI,EAAE,IAAI,CAAC,QAAQ,EAAE;SACtB,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,SAAS,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACxC,MAAM,IAAI,KAAK,CAAC,0BAA0B,QAAQ,CAAC,MAAM,IAAI,SAAS,EAAE,CAAC,CAAC;QAC5E,CAAC;QAED,MAAM,MAAM,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAqB,CAAC;QAE3D,wDAAwD;QACxD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,MAAM,CAAC,SAAS,GAAG,IAAI,CAAC;QAEvD,MAAM,MAAM,GAAiB;YAC3B,WAAW,EAAE,MAAM,CAAC,WAAW;YAC/B,YAAY,EAAE,MAAM,CAAC,YAAY;YACjC,SAAS;YACT,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC;SAC/B,CAAC;QAEF,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QACxB,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,kBAAkB;QACtB,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;QAC3D,CAAC;QAED,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QACxC,IAAI,CAAC,aAAa,EAAE,YAAY,EAAE,CAAC;YACjC,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;QAED,MAAM,QAAQ,GAAG,oCAAoC,CAAC;QAEtD,MAAM,IAAI,GAAG,IAAI,eAAe,CAAC;YAC/B,UAAU,EAAE,eAAe;YAC3B,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;YAC/B,aAAa,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY;YACvC,aAAa,EAAE,aAAa,CAAC,YAAY;SAC1C,CAAC,CAAC;QAEH,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,QAAQ,EAAE;YACrC,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,mCAAmC;gBACnD,MAAM,EAAE,kBAAkB;aAC3B;YACD,IAAI,EAAE,IAAI,CAAC,QAAQ,EAAE;SACtB,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,SAAS,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACxC,MAAM,IAAI,KAAK,CAAC,yBAAyB,QAAQ,CAAC,MAAM,IAAI,SAAS,EAAE,CAAC,CAAC;QAC3E,CAAC;QAED,MAAM,MAAM,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAqB,CAAC;QAE3D,MAAM,MAAM,GAAiB;YAC3B,WAAW,EAAE,MAAM,CAAC,WAAW;YAC/B,YAAY,EAAE,MAAM,CAAC,YAAY,IAAI,aAAa,CAAC,YAAY;YAC/D,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,MAAM,CAAC,SAAS,GAAG,IAAI;YAC/C,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC;SAC/B,CAAC;QAEF,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QACxB,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,aAAa;QACjB,MAAM,MAAM,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QACjC,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,MAAM,IAAI,KAAK,CAAC,sDAAsD,CAAC,CAAC;QAC1E,CAAC;QAED,2CAA2C;QAC3C,MAAM,WAAW,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC;QAClC,IAAI,MAAM,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,WAAW,EAAE,CAAC;YAChD,MAAM,CAAC,IAAI,CAAC,2CAA2C,CAAC,CAAC;YACzD,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAClD,OAAO,SAAS,CAAC,WAAW,CAAC;QAC/B,CAAC;QAED,OAAO,MAAM,CAAC,WAAW,CAAC;IAC5B,CAAC;IAED;;OAEG;IACK,UAAU,CAAC,MAAoB;QACrC,aAAa,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAChE,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;IACrC,CAAC;IAED;;OAEG;IACH,UAAU;QACR,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YACjC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,YAAY,CAAC,IAAI,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;YACzD,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QAChC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,+BAA+B,EAAE,KAAc,CAAC,CAAC;YAC9D,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACH,SAAS;QACP,IAAI,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YAChC,aAAa,CAAC,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC;QACD,IAAI,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;YAChC,aAAa,CAAC,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC;QACD,MAAM,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;IAC/C,CAAC;CACF;AAED;;GAEG;AACH,MAAM,CAAC,MAAM,qBAAqB,GAAG;IACnC,MAAM,EAAE,+BAA+B;IACvC,OAAO,EAAE,2BAA2B;IACpC,OAAO,EAAE,qCAAqC;CAC/C,CAAC;AAEF;;GAEG;AACH,MAAM,OAAO,gBAAgB;IACnB,WAAW,CAAoB;IAEvC,YAAY,WAAmB;QAC7B,IAAI,CAAC,WAAW,GAAG,IAAI,iBAAiB,CAAC,WAAW,CAAC,CAAC;IACxD,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,gBAAgB;QAIpB,mDAAmD;QACnD,4EAA4E;QAE5E,MAAM,MAAM,GAAqB;YAC/B,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,gBAAgB,IAAI,EAAE;YAC5C,YAAY,EAAE,OAAO,CAAC,GAAG,CAAC,oBAAoB,IAAI,EAAE;YACpD,WAAW,EACT,OAAO,CAAC,GAAG,CAAC,mBAAmB;gBAC/B,4CAA4C;YAC9C,MAAM,EAAE,qBAAqB;SAC9B,CAAC;QAEF,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,CAAC;YAC7C,OAAO;gBACL,OAAO,EAAE,EAAE;gBACX,YAAY,EAAE;oBACZ,yEAAyE;oBACzE,oEAAoE;oBACpE,0CAA0C;oBAC1C,+BAA+B;oBAC/B,6CAA6C;oBAC7C,qDAAqD;oBACrD,8BAA8B;iBAC/B;aACF,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAEpC,MAAM,EAAE,GAAG,EAAE,YAAY,EAAE,GAAG,IAAI,CAAC,WAAW,CAAC,eAAe,EAAE,CAAC;QAEjE,2FAA2F;QAC3F,OAAO,CAAC,GAAG,CAAC,qBAAqB,GAAG,YAAY,CAAC;QAEjD,OAAO;YACL,OAAO,EAAE,GAAG;YACZ,YAAY,EAAE;gBACZ,mCAAmC;gBACnC,GAAG;gBACH,EAAE;gBACF,wCAAwC;gBACxC,sDAAsD;gBACtD,6CAA6C;aAC9C;SACF,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,YAAY,CAAC,QAAgB;QACjC,IAAI,CAAC;YACH,MAAM,YAAY,GAAG,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC;YACvD,IAAI,CAAC,YAAY,EAAE,CAAC;gBAClB,MAAM,IAAI,KAAK,CACb,4DAA4D,CAC7D,CAAC;YACJ,CAAC;YAED,MAAM,IAAI,CAAC,WAAW,CAAC,oBAAoB,CAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;YACpE,OAAO,OAAO,CAAC,GAAG,CAAC,qBAAqB,CAAC;YAEzC,MAAM,CAAC,IAAI,CAAC,4CAA4C,CAAC,CAAC;YAC1D,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,wCAAwC,EAAE,KAAc,CAAC,CAAC;YACvE,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,cAAc;QAClB,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,WAAW,CAAC,aAAa,EAAE,CAAC;YAErD,mCAAmC;YACnC,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,gCAAgC,EAAE;gBAC7D,MAAM,EAAE,MAAM;gBACd,OAAO,EAAE;oBACP,aAAa,EAAE,UAAU,KAAK,EAAE;oBAChC,cAAc,EAAE,kBAAkB;iBACnC;gBACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;oBACnB,KAAK,EAAE,oCAAoC;iBAC5C,CAAC;aACH,CAAC,CAAC;YAEH,IAAI,QAAQ,CAAC,EAAE,EAAE,CAAC;gBAChB,MAAM,MAAM,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAQ,CAAC;gBAC9C,IAAI,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,CAAC;oBACxB,MAAM,CAAC,IAAI,CACT,2BAA2B,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,KAAK,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,GAAG,CACnF,CAAC;oBACF,OAAO,IAAI,CAAC;gBACd,CAAC;YACH,CAAC;YAED,OAAO,KAAK,CAAC;QACf,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,MAAM,CAAC,KAAK,CAAC,gCAAgC,EAAE,KAAc,CAAC,CAAC;YAC/D,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;CACF"}
@@ -0,0 +1,77 @@
1
+ /**
2
+ * Linear Auto-Sync Service
3
+ * Background service for automatic bidirectional synchronization
4
+ */
5
+ import { SyncConfig } from './sync.js';
6
+ export interface AutoSyncConfig extends SyncConfig {
7
+ enabled: boolean;
8
+ interval: number;
9
+ retryAttempts: number;
10
+ retryDelay: number;
11
+ quietHours?: {
12
+ start: number;
13
+ end: number;
14
+ };
15
+ }
16
+ export declare class LinearAutoSyncService {
17
+ private config;
18
+ private projectRoot;
19
+ private configManager;
20
+ private syncEngine?;
21
+ private intervalId?;
22
+ private isRunning;
23
+ private lastSyncTime;
24
+ private retryCount;
25
+ constructor(projectRoot: string, config?: Partial<AutoSyncConfig>);
26
+ /**
27
+ * Start the auto-sync service
28
+ */
29
+ start(): Promise<void>;
30
+ /**
31
+ * Stop the auto-sync service
32
+ */
33
+ stop(): void;
34
+ /**
35
+ * Get service status
36
+ */
37
+ getStatus(): {
38
+ running: boolean;
39
+ lastSyncTime: number;
40
+ nextSyncTime?: number;
41
+ retryCount: number;
42
+ config: AutoSyncConfig;
43
+ };
44
+ /**
45
+ * Update configuration
46
+ */
47
+ updateConfig(newConfig: Partial<AutoSyncConfig>): void;
48
+ /**
49
+ * Force immediate sync
50
+ */
51
+ forceSync(): Promise<void>;
52
+ /**
53
+ * Schedule next sync based on configuration
54
+ */
55
+ private scheduleNextSync;
56
+ /**
57
+ * Perform synchronization with error handling and retries
58
+ */
59
+ private performSync;
60
+ /**
61
+ * Check if current time is within quiet hours
62
+ */
63
+ private isInQuietHours;
64
+ }
65
+ /**
66
+ * Initialize global auto-sync service
67
+ */
68
+ export declare function initializeAutoSync(projectRoot: string, config?: Partial<AutoSyncConfig>): LinearAutoSyncService;
69
+ /**
70
+ * Get global auto-sync service
71
+ */
72
+ export declare function getAutoSyncService(): LinearAutoSyncService | null;
73
+ /**
74
+ * Stop global auto-sync service
75
+ */
76
+ export declare function stopAutoSync(): void;
77
+ //# sourceMappingURL=auto-sync.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"auto-sync.d.ts","sourceRoot":"","sources":["../../../../src/integrations/linear/auto-sync.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAKH,OAAO,EAAyC,UAAU,EAAE,MAAM,WAAW,CAAC;AAM9E,MAAM,WAAW,cAAe,SAAQ,UAAU;IAChD,OAAO,EAAE,OAAO,CAAC;IACjB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,CAAC,EAAE;QACX,KAAK,EAAE,MAAM,CAAC;QACd,GAAG,EAAE,MAAM,CAAC;KACb,CAAC;CACH;AAED,qBAAa,qBAAqB;IAChC,OAAO,CAAC,MAAM,CAAiB;IAC/B,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,aAAa,CAAsB;IAC3C,OAAO,CAAC,UAAU,CAAC,CAAmB;IACtC,OAAO,CAAC,UAAU,CAAC,CAAiB;IACpC,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,YAAY,CAAK;IACzB,OAAO,CAAC,UAAU,CAAK;gBAEX,WAAW,EAAE,MAAM,EAAE,MAAM,CAAC,EAAE,OAAO,CAAC,cAAc,CAAC;IA4BjE;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAyD5B;;OAEG;IACH,IAAI,IAAI,IAAI;IAUZ;;OAEG;IACH,SAAS,IAAI;QACX,OAAO,EAAE,OAAO,CAAC;QACjB,YAAY,EAAE,MAAM,CAAC;QACrB,YAAY,CAAC,EAAE,MAAM,CAAC;QACtB,UAAU,EAAE,MAAM,CAAC;QACnB,MAAM,EAAE,cAAc,CAAC;KACxB;IAcD;;OAEG;IACH,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,IAAI;IAYtD;;OAEG;IACG,SAAS,IAAI,OAAO,CAAC,IAAI,CAAC;IAShC;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAYxB;;OAEG;YACW,WAAW;IAkFzB;;OAEG;IACH,OAAO,CAAC,cAAc;CAevB;AAOD;;GAEG;AACH,wBAAgB,kBAAkB,CAChC,WAAW,EAAE,MAAM,EACnB,MAAM,CAAC,EAAE,OAAO,CAAC,cAAc,CAAC,GAC/B,qBAAqB,CAOvB;AAED;;GAEG;AACH,wBAAgB,kBAAkB,IAAI,qBAAqB,GAAG,IAAI,CAEjE;AAED;;GAEG;AACH,wBAAgB,YAAY,IAAI,IAAI,CAKnC"}
@@ -0,0 +1,268 @@
1
+ /**
2
+ * Linear Auto-Sync Service
3
+ * Background service for automatic bidirectional synchronization
4
+ */
5
+ import { logger } from '../../core/monitoring/logger.js';
6
+ import { PebblesTaskStore } from '../../features/tasks/pebbles-task-store.js';
7
+ import { LinearAuthManager } from './auth.js';
8
+ import { LinearSyncEngine, DEFAULT_SYNC_CONFIG } from './sync.js';
9
+ import { LinearConfigManager } from './config.js';
10
+ import Database from 'better-sqlite3';
11
+ import { join } from 'path';
12
+ import { existsSync } from 'fs';
13
+ export class LinearAutoSyncService {
14
+ config;
15
+ projectRoot;
16
+ configManager;
17
+ syncEngine;
18
+ intervalId;
19
+ isRunning = false;
20
+ lastSyncTime = 0;
21
+ retryCount = 0;
22
+ constructor(projectRoot, config) {
23
+ this.projectRoot = projectRoot;
24
+ this.configManager = new LinearConfigManager(projectRoot);
25
+ // Load persisted config or use defaults
26
+ const persistedConfig = this.configManager.loadConfig();
27
+ const baseConfig = persistedConfig
28
+ ? this.configManager.toAutoSyncConfig(persistedConfig)
29
+ : {
30
+ ...DEFAULT_SYNC_CONFIG,
31
+ enabled: true,
32
+ interval: 5,
33
+ retryAttempts: 3,
34
+ retryDelay: 30000,
35
+ autoSync: true,
36
+ direction: 'bidirectional',
37
+ conflictResolution: 'newest_wins',
38
+ quietHours: { start: 22, end: 7 },
39
+ };
40
+ this.config = { ...baseConfig, ...config };
41
+ // Save any new config updates
42
+ if (config && Object.keys(config).length > 0) {
43
+ this.configManager.saveConfig(config);
44
+ }
45
+ }
46
+ /**
47
+ * Start the auto-sync service
48
+ */
49
+ async start() {
50
+ if (this.isRunning) {
51
+ logger.warn('Linear auto-sync service is already running');
52
+ return;
53
+ }
54
+ try {
55
+ // Verify Linear integration is configured
56
+ const authManager = new LinearAuthManager(this.projectRoot);
57
+ if (!authManager.isConfigured()) {
58
+ throw new Error('Linear integration not configured. Run "stackmemory linear setup" first.');
59
+ }
60
+ // Initialize sync engine
61
+ const dbPath = join(this.projectRoot, '.stackmemory', 'context.db');
62
+ if (!existsSync(dbPath)) {
63
+ throw new Error('StackMemory not initialized. Run "stackmemory init" first.');
64
+ }
65
+ const db = new Database(dbPath);
66
+ const taskStore = new PebblesTaskStore(this.projectRoot, db);
67
+ this.syncEngine = new LinearSyncEngine(taskStore, authManager, this.config);
68
+ // Test connection before starting
69
+ const token = await authManager.getValidToken();
70
+ if (!token) {
71
+ throw new Error('Unable to get valid Linear token. Check authentication.');
72
+ }
73
+ this.isRunning = true;
74
+ this.scheduleNextSync();
75
+ logger.info('Linear auto-sync service started', {
76
+ interval: this.config.interval,
77
+ direction: this.config.direction,
78
+ conflictResolution: this.config.conflictResolution,
79
+ });
80
+ // Perform initial sync
81
+ this.performSync();
82
+ }
83
+ catch (error) {
84
+ logger.error('Failed to start Linear auto-sync service:', error);
85
+ throw error;
86
+ }
87
+ }
88
+ /**
89
+ * Stop the auto-sync service
90
+ */
91
+ stop() {
92
+ if (this.intervalId) {
93
+ clearTimeout(this.intervalId);
94
+ this.intervalId = undefined;
95
+ }
96
+ this.isRunning = false;
97
+ logger.info('Linear auto-sync service stopped');
98
+ }
99
+ /**
100
+ * Get service status
101
+ */
102
+ getStatus() {
103
+ const nextSyncTime = this.intervalId
104
+ ? this.lastSyncTime + this.config.interval * 60 * 1000
105
+ : undefined;
106
+ return {
107
+ running: this.isRunning,
108
+ lastSyncTime: this.lastSyncTime,
109
+ nextSyncTime,
110
+ retryCount: this.retryCount,
111
+ config: this.config,
112
+ };
113
+ }
114
+ /**
115
+ * Update configuration
116
+ */
117
+ updateConfig(newConfig) {
118
+ this.config = { ...this.config, ...newConfig };
119
+ if (this.isRunning) {
120
+ // Restart with new config
121
+ this.stop();
122
+ this.start();
123
+ }
124
+ logger.info('Linear auto-sync config updated', newConfig);
125
+ }
126
+ /**
127
+ * Force immediate sync
128
+ */
129
+ async forceSync() {
130
+ if (!this.syncEngine) {
131
+ throw new Error('Sync engine not initialized');
132
+ }
133
+ logger.info('Forcing immediate Linear sync');
134
+ await this.performSync();
135
+ }
136
+ /**
137
+ * Schedule next sync based on configuration
138
+ */
139
+ scheduleNextSync() {
140
+ if (!this.isRunning)
141
+ return;
142
+ const delay = this.config.interval * 60 * 1000; // Convert minutes to milliseconds
143
+ this.intervalId = setTimeout(() => {
144
+ if (this.isRunning) {
145
+ this.performSync();
146
+ }
147
+ }, delay);
148
+ }
149
+ /**
150
+ * Perform synchronization with error handling and retries
151
+ */
152
+ async performSync() {
153
+ if (!this.syncEngine) {
154
+ logger.error('Sync engine not available');
155
+ return;
156
+ }
157
+ // Check quiet hours
158
+ if (this.isInQuietHours()) {
159
+ logger.debug('Skipping sync during quiet hours');
160
+ this.scheduleNextSync();
161
+ return;
162
+ }
163
+ try {
164
+ logger.debug('Starting Linear auto-sync');
165
+ const result = await this.syncEngine.sync();
166
+ if (result.success) {
167
+ this.lastSyncTime = Date.now();
168
+ this.retryCount = 0;
169
+ // Log sync results
170
+ const hasChanges = result.synced.toLinear > 0 ||
171
+ result.synced.fromLinear > 0 ||
172
+ result.synced.updated > 0;
173
+ if (hasChanges) {
174
+ logger.info('Linear auto-sync completed with changes', {
175
+ toLinear: result.synced.toLinear,
176
+ fromLinear: result.synced.fromLinear,
177
+ updated: result.synced.updated,
178
+ conflicts: result.conflicts.length,
179
+ });
180
+ }
181
+ else {
182
+ logger.debug('Linear auto-sync completed - no changes');
183
+ }
184
+ // Handle conflicts
185
+ if (result.conflicts.length > 0) {
186
+ logger.warn('Linear sync conflicts detected', {
187
+ count: result.conflicts.length,
188
+ conflicts: result.conflicts.map((c) => ({
189
+ taskId: c.taskId,
190
+ reason: c.reason,
191
+ })),
192
+ });
193
+ }
194
+ }
195
+ else {
196
+ throw new Error(`Sync failed: ${result.errors.join(', ')}`);
197
+ }
198
+ }
199
+ catch (error) {
200
+ logger.error('Linear auto-sync failed:', error);
201
+ this.retryCount++;
202
+ if (this.retryCount <= this.config.retryAttempts) {
203
+ logger.info(`Retrying Linear sync in ${this.config.retryDelay / 1000}s (attempt ${this.retryCount}/${this.config.retryAttempts})`);
204
+ // Schedule retry
205
+ setTimeout(() => {
206
+ if (this.isRunning) {
207
+ this.performSync();
208
+ }
209
+ }, this.config.retryDelay);
210
+ return; // Don't schedule next sync yet
211
+ }
212
+ else {
213
+ logger.error(`Linear auto-sync failed after ${this.config.retryAttempts} attempts, skipping until next interval`);
214
+ this.retryCount = 0;
215
+ }
216
+ }
217
+ // Schedule next sync
218
+ this.scheduleNextSync();
219
+ }
220
+ /**
221
+ * Check if current time is within quiet hours
222
+ */
223
+ isInQuietHours() {
224
+ if (!this.config.quietHours)
225
+ return false;
226
+ const now = new Date();
227
+ const currentHour = now.getHours();
228
+ const { start, end } = this.config.quietHours;
229
+ if (start < end) {
230
+ // Quiet hours within same day (e.g., 22:00 - 07:00 next day)
231
+ return currentHour >= start || currentHour < end;
232
+ }
233
+ else {
234
+ // Quiet hours span midnight (e.g., 10:00 - 18:00)
235
+ return currentHour >= start && currentHour < end;
236
+ }
237
+ }
238
+ }
239
+ /**
240
+ * Global auto-sync service instance
241
+ */
242
+ let autoSyncService = null;
243
+ /**
244
+ * Initialize global auto-sync service
245
+ */
246
+ export function initializeAutoSync(projectRoot, config) {
247
+ if (autoSyncService) {
248
+ autoSyncService.stop();
249
+ }
250
+ autoSyncService = new LinearAutoSyncService(projectRoot, config);
251
+ return autoSyncService;
252
+ }
253
+ /**
254
+ * Get global auto-sync service
255
+ */
256
+ export function getAutoSyncService() {
257
+ return autoSyncService;
258
+ }
259
+ /**
260
+ * Stop global auto-sync service
261
+ */
262
+ export function stopAutoSync() {
263
+ if (autoSyncService) {
264
+ autoSyncService.stop();
265
+ autoSyncService = null;
266
+ }
267
+ }
268
+ //# sourceMappingURL=auto-sync.js.map