@dollhousemcp/mcp-server 1.7.1 → 1.7.3
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/CHANGELOG.md +22 -0
- package/dist/auth/GitHubAuthManager.js +2 -2
- package/dist/config/ConfigManager.d.ts +158 -25
- package/dist/config/ConfigManager.d.ts.map +1 -1
- package/dist/config/ConfigManager.js +627 -88
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ConfigHandler.d.ts +32 -0
- package/dist/handlers/ConfigHandler.d.ts.map +1 -0
- package/dist/handlers/ConfigHandler.js +202 -0
- package/dist/handlers/SyncHandlerV2.d.ts +42 -0
- package/dist/handlers/SyncHandlerV2.d.ts.map +1 -0
- package/dist/handlers/SyncHandlerV2.js +231 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +19 -3
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts +0 -1
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -1
- package/dist/portfolio/GitHubPortfolioIndexer.js +36 -16
- package/dist/portfolio/PortfolioRepoManager.d.ts +2 -1
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioRepoManager.js +2 -1
- package/dist/portfolio/PortfolioSyncManager.d.ts +127 -0
- package/dist/portfolio/PortfolioSyncManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioSyncManager.js +818 -0
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +54 -2
- package/dist/security/secureYamlParser.d.ts +46 -2
- package/dist/security/secureYamlParser.d.ts.map +1 -1
- package/dist/security/secureYamlParser.js +47 -3
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +16 -10
- package/dist/server/tools/ConfigToolsV2.d.ts +10 -0
- package/dist/server/tools/ConfigToolsV2.d.ts.map +1 -0
- package/dist/server/tools/ConfigToolsV2.js +110 -0
- package/dist/server/types.d.ts +2 -0
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/utils/logger.d.ts +45 -0
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +202 -9
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"GitHubPortfolioIndexer.d.ts","sourceRoot":"","sources":["../../src/portfolio/GitHubPortfolioIndexer.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;
|
|
1
|
+
{"version":3,"file":"GitHubPortfolioIndexer.d.ts","sourceRoot":"","sources":["../../src/portfolio/GitHubPortfolioIndexer.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAIH,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAOzC,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,WAAW,CAAC;IACzB,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,EAAE,MAAM,CAAC;IACpB,YAAY,EAAE,IAAI,CAAC;IACnB,IAAI,EAAE,MAAM,CAAC;CACd;AAED,MAAM,WAAW,oBAAoB;IACnC,QAAQ,EAAE,MAAM,CAAC;IACjB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,IAAI,CAAC;IAClB,QAAQ,EAAE,GAAG,CAAC,WAAW,EAAE,gBAAgB,EAAE,CAAC,CAAC;IAC/C,aAAa,EAAE,MAAM,CAAC;IACtB,GAAG,EAAE,MAAM,CAAC;IACZ,aAAa,CAAC,EAAE;QACd,SAAS,EAAE,MAAM,CAAC;QAClB,SAAS,EAAE,IAAI,CAAC;KACjB,CAAC;CACH;AAED,MAAM,WAAW,kBAAkB;IACjC,KAAK,CAAC,EAAE,OAAO,CAAC;IAChB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,YAAY,CAAC,EAAE,WAAW,EAAE,CAAC;IAC7B,UAAU,CAAC,EAAE,OAAO,CAAC;CACtB;AAED,qBAAa,sBAAsB;IACjC,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAuC;IAC9D,OAAO,CAAC,MAAM,CAAC,YAAY,CAAS;IAEpC,OAAO,CAAC,KAAK,CAAqC;IAClD,OAAO,CAAC,SAAS,CAAqB;IACtC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAkB;IACtC,OAAO,CAAC,gBAAgB,CAAS;IACjC,OAAO,CAAC,eAAe,CAAqB;IAC5C,OAAO,CAAC,QAAQ,CAAC,iBAAiB,CAAiB;IAEnD,OAAO,CAAC,oBAAoB,CAAuB;IACnD,OAAO,CAAC,QAAQ,CAAW;IAC3B,OAAO,CAAC,gBAAgB,CAAwB;IAEhD,OAAO;IAQP;;OAEG;WACW,WAAW,IAAI,sBAAsB;IAgBnD;;OAEG;IACU,QAAQ,CAAC,KAAK,UAAQ,GAAG,OAAO,CAAC,oBAAoB,CAAC;IAkDnE;;OAEG;IACI,qBAAqB,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAgBlD;;OAEG;IACI,UAAU,IAAI,IAAI;IAUzB;;OAEG;IACI,aAAa,IAAI;QACtB,aAAa,EAAE,OAAO,CAAC;QACvB,SAAS,EAAE,IAAI,GAAG,IAAI,CAAC;QACvB,OAAO,EAAE,OAAO,CAAC;QACjB,gBAAgB,EAAE,OAAO,CAAC;QAC1B,aAAa,EAAE,MAAM,CAAC;KACvB;IAUD;;OAEG;YACW,UAAU;IAmDxB;;OAEG;YACW,sBAAsB;IAapC;;OAEG;YACW,gBAAgB;IA0D9B;;OAEG;YACW,aAAa;IAsD3B;;OAEG;YACW,uBAAuB;IA8DrC;;OAEG;YACW,sBAAsB;IAsDpC;;OAEG;IACH,OAAO,CAAC,wBAAwB;IA6BhC;;OAEG;YACW,iBAAiB;IAS/B;;OAEG;IACH,OAAO,CAAC,YAAY;IASpB;;OAEG;IACH,OAAO,CAAC,gBAAgB;IAuBxB;;OAEG;IACH,OAAO,CAAC,gBAAgB;CAiBzB"}
|
|
@@ -9,7 +9,6 @@
|
|
|
9
9
|
* - Fallback strategy for resilient operation
|
|
10
10
|
* - Performance optimized for 1000+ portfolio elements
|
|
11
11
|
*/
|
|
12
|
-
import { GitHubClient } from '../collection/GitHubClient.js';
|
|
13
12
|
import { PortfolioRepoManager } from './PortfolioRepoManager.js';
|
|
14
13
|
import { ElementType } from './types.js';
|
|
15
14
|
import { logger } from '../utils/logger.js';
|
|
@@ -26,14 +25,12 @@ export class GitHubPortfolioIndexer {
|
|
|
26
25
|
recentUserAction = false;
|
|
27
26
|
actionTimestamp = null;
|
|
28
27
|
actionGracePeriod = 2 * 60 * 1000; // 2 minutes after action
|
|
29
|
-
githubClient;
|
|
30
28
|
portfolioRepoManager;
|
|
31
29
|
apiCache;
|
|
32
30
|
rateLimitTracker;
|
|
33
31
|
constructor() {
|
|
34
32
|
this.apiCache = new APICache(); // Uses default settings
|
|
35
33
|
this.rateLimitTracker = new Map();
|
|
36
|
-
this.githubClient = new GitHubClient(this.apiCache, this.rateLimitTracker);
|
|
37
34
|
this.portfolioRepoManager = new PortfolioRepoManager();
|
|
38
35
|
logger.debug('GitHubPortfolioIndexer created');
|
|
39
36
|
}
|
|
@@ -252,7 +249,9 @@ export class GitHubPortfolioIndexer {
|
|
|
252
249
|
}
|
|
253
250
|
`;
|
|
254
251
|
const variables = { owner: username, name: repository };
|
|
255
|
-
|
|
252
|
+
// GraphQL endpoint not yet implemented with PortfolioRepoManager
|
|
253
|
+
// For now, just throw to fall back to REST
|
|
254
|
+
throw new Error('GraphQL not yet implemented for portfolio operations');
|
|
256
255
|
// Note: This is a simplified GraphQL implementation
|
|
257
256
|
// In a real implementation, you would send POST request with query and variables
|
|
258
257
|
throw new Error('GraphQL implementation not yet complete');
|
|
@@ -263,8 +262,8 @@ export class GitHubPortfolioIndexer {
|
|
|
263
262
|
async fetchWithREST(username, repository) {
|
|
264
263
|
const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;
|
|
265
264
|
// Get repository info and latest commit
|
|
266
|
-
const repoInfo = await this.
|
|
267
|
-
const latestCommit = await this.
|
|
265
|
+
const repoInfo = await this.portfolioRepoManager.githubRequest(`/repos/${normalizedUsername}/${repository}`);
|
|
266
|
+
const latestCommit = await this.portfolioRepoManager.githubRequest(`/repos/${normalizedUsername}/${repository}/commits/HEAD`);
|
|
268
267
|
// Initialize index
|
|
269
268
|
const index = {
|
|
270
269
|
username: normalizedUsername,
|
|
@@ -279,15 +278,24 @@ export class GitHubPortfolioIndexer {
|
|
|
279
278
|
index.elements.set(elementType, []);
|
|
280
279
|
}
|
|
281
280
|
// Fetch content for each element type
|
|
281
|
+
logger.info(`Fetching content for all element types from GitHub portfolio...`);
|
|
282
282
|
for (const elementType of Object.values(ElementType)) {
|
|
283
283
|
try {
|
|
284
|
+
logger.debug(`Fetching ${elementType} from GitHub...`);
|
|
284
285
|
const entries = await this.fetchElementTypeContent(normalizedUsername, repository, elementType);
|
|
285
286
|
index.elements.set(elementType, entries);
|
|
286
287
|
index.totalElements += entries.length;
|
|
288
|
+
if (entries.length > 0) {
|
|
289
|
+
logger.info(`✅ Found ${entries.length} ${elementType} in GitHub portfolio`);
|
|
290
|
+
}
|
|
291
|
+
else {
|
|
292
|
+
logger.debug(`📁 No ${elementType} found (directory may not exist yet)`);
|
|
293
|
+
}
|
|
287
294
|
}
|
|
288
295
|
catch (error) {
|
|
289
|
-
logger.warn(
|
|
290
|
-
error: error instanceof Error ? error.message : String(error)
|
|
296
|
+
logger.warn(`❌ Failed to fetch ${elementType} from GitHub portfolio`, {
|
|
297
|
+
error: error instanceof Error ? error.message : String(error),
|
|
298
|
+
elementType
|
|
291
299
|
});
|
|
292
300
|
// Continue with other element types
|
|
293
301
|
}
|
|
@@ -299,8 +307,8 @@ export class GitHubPortfolioIndexer {
|
|
|
299
307
|
*/
|
|
300
308
|
async fetchElementTypeContent(username, repository, elementType) {
|
|
301
309
|
try {
|
|
302
|
-
// Get directory listing
|
|
303
|
-
const contents = await this.
|
|
310
|
+
// Get directory listing using PortfolioRepoManager
|
|
311
|
+
const contents = await this.portfolioRepoManager.githubRequest(`/repos/${username}/${repository}/contents/${elementType}`);
|
|
304
312
|
if (!Array.isArray(contents)) {
|
|
305
313
|
return [];
|
|
306
314
|
}
|
|
@@ -326,10 +334,20 @@ export class GitHubPortfolioIndexer {
|
|
|
326
334
|
return entries;
|
|
327
335
|
}
|
|
328
336
|
catch (error) {
|
|
329
|
-
// Directory might not exist
|
|
330
|
-
|
|
331
|
-
|
|
337
|
+
// Directory might not exist - check for 404 errors
|
|
338
|
+
// PortfolioRepoManager will throw standard GitHub API errors
|
|
339
|
+
if (error instanceof Error) {
|
|
340
|
+
// Check for 404 Not Found errors from GitHub API
|
|
341
|
+
if (error.message.includes('404') ||
|
|
342
|
+
error.message.includes('Not Found')) {
|
|
343
|
+
logger.debug(`Directory ${elementType} not found in GitHub repository (this is normal if not yet created)`);
|
|
344
|
+
return [];
|
|
345
|
+
}
|
|
346
|
+
// Log the actual error for debugging
|
|
347
|
+
logger.debug(`Error fetching ${elementType}: ${error.message}`);
|
|
332
348
|
}
|
|
349
|
+
// Re-throw other errors
|
|
350
|
+
logger.warn(`Unexpected error fetching ${elementType}:`, error);
|
|
333
351
|
throw error;
|
|
334
352
|
}
|
|
335
353
|
}
|
|
@@ -354,7 +372,9 @@ export class GitHubPortfolioIndexer {
|
|
|
354
372
|
// This is expensive, so only do it for small files or when specifically needed
|
|
355
373
|
if (fileInfo.size && fileInfo.size < 10000) { // Only for files < 10KB
|
|
356
374
|
try {
|
|
357
|
-
|
|
375
|
+
// For download URLs, we need to fetch directly, not through API
|
|
376
|
+
const response = await fetch(fileInfo.download_url);
|
|
377
|
+
const content = await response.text();
|
|
358
378
|
const metadata = this.parseMetadataFromContent(content);
|
|
359
379
|
if (metadata.name)
|
|
360
380
|
entry.name = metadata.name;
|
|
@@ -412,7 +432,7 @@ export class GitHubPortfolioIndexer {
|
|
|
412
432
|
*/
|
|
413
433
|
async getGitHubUsername() {
|
|
414
434
|
try {
|
|
415
|
-
const userInfo = await this.
|
|
435
|
+
const userInfo = await this.portfolioRepoManager.githubRequest('/user');
|
|
416
436
|
return userInfo.login;
|
|
417
437
|
}
|
|
418
438
|
catch (error) {
|
|
@@ -472,4 +492,4 @@ export class GitHubPortfolioIndexer {
|
|
|
472
492
|
return index;
|
|
473
493
|
}
|
|
474
494
|
}
|
|
475
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"GitHubPortfolioIndexer.js","sourceRoot":"","sources":["../../src/portfolio/GitHubPortfolioIndexer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * GitHub Portfolio Indexer - Fetches and indexes user's GitHub portfolio for fast searching\n * \n * Features:\n * - Singleton pattern for efficient resource usage\n * - Smart caching with TTL and invalidation after user actions\n * - GraphQL/REST API integration for efficient fetching\n * - Rate limiting and authentication handling\n * - Fallback strategy for resilient operation\n * - Performance optimized for 1000+ portfolio elements\n */\n\nimport { GitHubClient } from '../collection/GitHubClient.js';\nimport { PortfolioRepoManager } from './PortfolioRepoManager.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { ElementType } from './types.js';\nimport { logger } from '../utils/logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\nimport { APICache } from '../cache/APICache.js';\n\nexport interface GitHubIndexEntry {\n  path: string;\n  name: string;\n  description?: string;\n  version?: string;\n  author?: string;\n  elementType: ElementType;\n  sha: string; // File SHA for change detection\n  htmlUrl: string; // Link to GitHub\n  downloadUrl: string;\n  lastModified: Date;\n  size: number;\n}\n\nexport interface GitHubPortfolioIndex {\n  username: string;\n  repository: string;\n  lastUpdated: Date;\n  elements: Map<ElementType, GitHubIndexEntry[]>;\n  totalElements: number;\n  sha: string; // Latest commit SHA\n  rateLimitInfo?: {\n    remaining: number;\n    resetTime: Date;\n  };\n}\n\nexport interface GitHubFetchOptions {\n  force?: boolean;\n  maxElements?: number;\n  elementTypes?: ElementType[];\n  useGraphQL?: boolean;\n}\n\nexport class GitHubPortfolioIndexer {\n  private static instance: GitHubPortfolioIndexer | null = null;\n  private static instanceLock = false;\n  \n  private cache: GitHubPortfolioIndex | null = null;\n  private lastFetch: Date | null = null;\n  private readonly ttl = 15 * 60 * 1000; // 15 minutes\n  private recentUserAction = false;\n  private actionTimestamp: Date | null = null;\n  private readonly actionGracePeriod = 2 * 60 * 1000; // 2 minutes after action\n  \n  private githubClient: GitHubClient;\n  private portfolioRepoManager: PortfolioRepoManager;\n  private apiCache: APICache;\n  private rateLimitTracker: Map<string, number[]>;\n  \n  private constructor() {\n    this.apiCache = new APICache(); // Uses default settings\n    this.rateLimitTracker = new Map();\n    this.githubClient = new GitHubClient(this.apiCache, this.rateLimitTracker);\n    this.portfolioRepoManager = new PortfolioRepoManager();\n    \n    logger.debug('GitHubPortfolioIndexer created');\n  }\n\n  /**\n   * Singleton pattern with thread safety\n   */\n  public static getInstance(): GitHubPortfolioIndexer {\n    if (!this.instance) {\n      if (this.instanceLock) {\n        throw new Error('GitHubPortfolioIndexer instance is being created by another thread');\n      }\n      \n      try {\n        this.instanceLock = true;\n        this.instance = new GitHubPortfolioIndexer();\n      } finally {\n        this.instanceLock = false;\n      }\n    }\n    return this.instance;\n  }\n\n  /**\n   * Main method to get GitHub portfolio index\n   */\n  public async getIndex(force = false): Promise<GitHubPortfolioIndex> {\n    try {\n      // Check if we need fresh data\n      if (force || this.shouldFetchFresh()) {\n        return await this.fetchFresh();\n      }\n      \n      // Return cached data if available and valid\n      if (this.cache && this.isCacheValid()) {\n        logger.debug('Returning cached GitHub portfolio index', {\n          username: this.cache.username,\n          totalElements: this.cache.totalElements,\n          age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'\n        });\n        return this.cache;\n      }\n      \n      // Try to fetch fresh, fall back to stale cache on failure\n      try {\n        return await this.fetchFresh();\n      } catch (error) {\n        logger.warn('Failed to fetch fresh GitHub portfolio index, checking for stale cache', {\n          error: error instanceof Error ? error.message : String(error)\n        });\n        \n        // Return stale cache if available\n        if (this.cache) {\n          logger.info('Returning stale GitHub portfolio cache as fallback', {\n            username: this.cache.username,\n            age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'\n          });\n          return this.cache;\n        }\n        \n        // Return empty index as last resort\n        return this.createEmptyIndex();\n      }\n      \n    } catch (error) {\n      ErrorHandler.logError('GitHubPortfolioIndexer.getIndex', error);\n      \n      // Return stale cache or empty index\n      if (this.cache) {\n        return this.cache;\n      }\n      \n      return this.createEmptyIndex();\n    }\n  }\n\n  /**\n   * Invalidate cache after user actions\n   */\n  public invalidateAfterAction(action: string): void {\n    logger.info('Invalidating GitHub portfolio cache after user action', { action });\n    \n    this.recentUserAction = true;\n    this.actionTimestamp = new Date();\n    \n    // Log security event for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_CACHE_INVALIDATION',\n      severity: 'LOW',\n      source: 'GitHubPortfolioIndexer.invalidateAfterAction',\n      details: `Cache invalidated after user action: ${action}`,\n      metadata: { action }\n    });\n  }\n\n  /**\n   * Clear all cached data\n   */\n  public clearCache(): void {\n    this.cache = null;\n    this.lastFetch = null;\n    this.recentUserAction = false;\n    this.actionTimestamp = null;\n    this.apiCache.clear();\n    \n    logger.info('GitHub portfolio cache cleared');\n  }\n\n  /**\n   * Get cache statistics\n   */\n  public getCacheStats(): {\n    hasCachedData: boolean;\n    lastFetch: Date | null;\n    isStale: boolean;\n    recentUserAction: boolean;\n    totalElements: number;\n  } {\n    return {\n      hasCachedData: this.cache !== null,\n      lastFetch: this.lastFetch,\n      isStale: !this.isCacheValid(),\n      recentUserAction: this.recentUserAction,\n      totalElements: this.cache?.totalElements || 0\n    };\n  }\n\n  /**\n   * Fetch fresh data from GitHub\n   */\n  private async fetchFresh(): Promise<GitHubPortfolioIndex> {\n    const startTime = Date.now();\n    logger.info('Fetching fresh GitHub portfolio index...');\n    \n    try {\n      // Get GitHub username from token\n      const username = await this.getGitHubUsername();\n      const repository = 'dollhouse-portfolio';\n      \n      // Check if portfolio repository exists\n      const repoExists = await this.portfolioRepoManager.checkPortfolioExists(username);\n      if (!repoExists) {\n        logger.info('GitHub portfolio repository does not exist', { username });\n        return this.createEmptyIndex(username, repository);\n      }\n      \n      // Fetch repository content using GitHub API\n      const index = await this.fetchRepositoryContent(username, repository);\n      \n      // Update cache\n      this.cache = index;\n      this.lastFetch = new Date();\n      this.recentUserAction = false;\n      this.actionTimestamp = null;\n      \n      const duration = Date.now() - startTime;\n      logger.info('GitHub portfolio index fetched successfully', {\n        username,\n        totalElements: index.totalElements,\n        duration: `${duration}ms`,\n        rateLimitRemaining: index.rateLimitInfo?.remaining\n      });\n      \n      // Log security event\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_FETCH_SUCCESS',\n        severity: 'LOW',\n        source: 'GitHubPortfolioIndexer.fetchFresh',\n        details: `Fetched GitHub portfolio with ${index.totalElements} elements in ${duration}ms`,\n        metadata: { username, duration, totalElements: index.totalElements }\n      });\n      \n      return index;\n      \n    } catch (error) {\n      const duration = Date.now() - startTime;\n      ErrorHandler.logError('GitHubPortfolioIndexer.fetchFresh', error, { duration });\n      throw ErrorHandler.wrapError(error, 'Failed to fetch GitHub portfolio index', ErrorCategory.NETWORK_ERROR);\n    }\n  }\n\n  /**\n   * Fetch repository content from GitHub API\n   */\n  private async fetchRepositoryContent(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    // Try GraphQL first for better performance, fallback to REST\n    try {\n      return await this.fetchWithGraphQL(username, repository);\n    } catch (graphqlError) {\n      logger.debug('GraphQL fetch failed, falling back to REST API', {\n        error: graphqlError instanceof Error ? graphqlError.message : String(graphqlError)\n      });\n      \n      return await this.fetchWithREST(username, repository);\n    }\n  }\n\n  /**\n   * Fetch using GraphQL for better performance\n   */\n  private async fetchWithGraphQL(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    const query = `\n      query GetPortfolioContent($owner: String!, $name: String!) {\n        repository(owner: $owner, name: $name) {\n          defaultBranchRef {\n            target {\n              ... on Commit {\n                oid\n                history(first: 1) {\n                  nodes {\n                    committedDate\n                  }\n                }\n              }\n            }\n          }\n          object(expression: \"HEAD:\") {\n            ... on Tree {\n              entries {\n                name\n                type\n                object {\n                  ... on Tree {\n                    entries {\n                      name\n                      type\n                      oid\n                      object {\n                        ... on Blob {\n                          byteSize\n                          text\n                        }\n                      }\n                    }\n                  }\n                }\n              }\n            }\n          }\n        }\n        rateLimit {\n          remaining\n          resetAt\n        }\n      }\n    `;\n    \n    const variables = { owner: username, name: repository };\n    \n    const response = await this.githubClient.fetchFromGitHub('https://api.github.com/graphql', true);\n    \n    // Note: This is a simplified GraphQL implementation\n    // In a real implementation, you would send POST request with query and variables\n    throw new Error('GraphQL implementation not yet complete');\n  }\n\n  /**\n   * Fetch using REST API with pagination\n   */\n  private async fetchWithREST(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    \n    // Get repository info and latest commit\n    const repoInfo = await this.githubClient.fetchFromGitHub(\n      `https://api.github.com/repos/${normalizedUsername}/${repository}`\n    );\n    \n    const latestCommit = await this.githubClient.fetchFromGitHub(\n      `https://api.github.com/repos/${normalizedUsername}/${repository}/commits/HEAD`\n    );\n    \n    // Initialize index\n    const index: GitHubPortfolioIndex = {\n      username: normalizedUsername,\n      repository,\n      lastUpdated: new Date(latestCommit.commit.committer.date),\n      elements: new Map(),\n      totalElements: 0,\n      sha: latestCommit.sha\n    };\n    \n    // Initialize element type maps\n    for (const elementType of Object.values(ElementType)) {\n      index.elements.set(elementType, []);\n    }\n    \n    // Fetch content for each element type\n    for (const elementType of Object.values(ElementType)) {\n      try {\n        const entries = await this.fetchElementTypeContent(normalizedUsername, repository, elementType);\n        index.elements.set(elementType, entries);\n        index.totalElements += entries.length;\n      } catch (error) {\n        logger.warn(`Failed to fetch ${elementType} from GitHub portfolio`, {\n          error: error instanceof Error ? error.message : String(error)\n        });\n        // Continue with other element types\n      }\n    }\n    \n    return index;\n  }\n\n  /**\n   * Fetch content for a specific element type\n   */\n  private async fetchElementTypeContent(\n    username: string,\n    repository: string,\n    elementType: ElementType\n  ): Promise<GitHubIndexEntry[]> {\n    try {\n      // Get directory listing\n      const contents = await this.githubClient.fetchFromGitHub(\n        `https://api.github.com/repos/${username}/${repository}/contents/${elementType}`\n      );\n      \n      if (!Array.isArray(contents)) {\n        return [];\n      }\n      \n      const entries: GitHubIndexEntry[] = [];\n      const maxConcurrent = 5; // Limit concurrent requests\n      \n      // Process files in batches to avoid rate limiting\n      for (let i = 0; i < contents.length; i += maxConcurrent) {\n        const batch = contents.slice(i, i + maxConcurrent);\n        const batchPromises = batch\n          .filter(item => item.type === 'file' && item.name.endsWith('.md'))\n          .map(item => this.createGitHubIndexEntry(username, repository, elementType, item));\n        \n        const batchResults = await Promise.allSettled(batchPromises);\n        \n        for (const result of batchResults) {\n          if (result.status === 'fulfilled' && result.value) {\n            entries.push(result.value);\n          }\n        }\n        \n        // Add delay between batches to respect rate limits\n        if (i + maxConcurrent < contents.length) {\n          await new Promise(resolve => setTimeout(resolve, 100));\n        }\n      }\n      \n      return entries;\n      \n    } catch (error) {\n      // Directory might not exist\n      if (error instanceof Error && error.message.includes('404')) {\n        return [];\n      }\n      throw error;\n    }\n  }\n\n  /**\n   * Create GitHub index entry from API response\n   */\n  private async createGitHubIndexEntry(\n    username: string,\n    repository: string,\n    elementType: ElementType,\n    fileInfo: any\n  ): Promise<GitHubIndexEntry | null> {\n    try {\n      // Parse metadata from filename or fetch content if needed\n      const name = fileInfo.name.replace('.md', '').replace(/-/g, ' ');\n      \n      const entry: GitHubIndexEntry = {\n        path: fileInfo.path,\n        name,\n        elementType,\n        sha: fileInfo.sha,\n        htmlUrl: fileInfo.html_url,\n        downloadUrl: fileInfo.download_url,\n        lastModified: new Date(), // GitHub API doesn't provide file modification time directly\n        size: fileInfo.size || 0\n      };\n      \n      // Optionally fetch content to extract metadata\n      // This is expensive, so only do it for small files or when specifically needed\n      if (fileInfo.size && fileInfo.size < 10000) { // Only for files < 10KB\n        try {\n          const content = await this.githubClient.fetchFromGitHub(fileInfo.download_url);\n          const metadata = this.parseMetadataFromContent(content);\n          \n          if (metadata.name) entry.name = metadata.name;\n          if (metadata.description) entry.description = metadata.description;\n          if (metadata.version) entry.version = metadata.version;\n          if (metadata.author) entry.author = metadata.author;\n        } catch (metadataError) {\n          // Non-critical error, continue without metadata\n          logger.debug('Failed to fetch metadata for file', {\n            path: fileInfo.path,\n            error: metadataError instanceof Error ? metadataError.message : String(metadataError)\n          });\n        }\n      }\n      \n      return entry;\n      \n    } catch (error) {\n      logger.debug('Failed to create GitHub index entry', {\n        path: fileInfo.path,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Parse metadata from file content (frontmatter)\n   */\n  private parseMetadataFromContent(content: string): {\n    name?: string;\n    description?: string;\n    version?: string;\n    author?: string;\n  } {\n    const metadata: any = {};\n    \n    // Simple frontmatter parsing (could use a proper YAML parser)\n    const frontmatterMatch = content.match(/^---\\n([\\s\\S]*?)\\n---/);\n    if (frontmatterMatch) {\n      const frontmatter = frontmatterMatch[1];\n      \n      const nameMatch = frontmatter.match(/^name:\\s*(.+)$/m);\n      if (nameMatch) metadata.name = nameMatch[1].trim();\n      \n      const descMatch = frontmatter.match(/^description:\\s*(.+)$/m);\n      if (descMatch) metadata.description = descMatch[1].trim();\n      \n      const versionMatch = frontmatter.match(/^version:\\s*(.+)$/m);\n      if (versionMatch) metadata.version = versionMatch[1].trim();\n      \n      const authorMatch = frontmatter.match(/^author:\\s*(.+)$/m);\n      if (authorMatch) metadata.author = authorMatch[1].trim();\n    }\n    \n    return metadata;\n  }\n\n  /**\n   * Get GitHub username from authenticated token\n   */\n  private async getGitHubUsername(): Promise<string> {\n    try {\n      const userInfo = await this.githubClient.fetchFromGitHub('https://api.github.com/user', true);\n      return userInfo.login;\n    } catch (error) {\n      throw new Error('Failed to get GitHub username. Please ensure you are authenticated with GitHub.');\n    }\n  }\n\n  /**\n   * Check if cache is valid\n   */\n  private isCacheValid(): boolean {\n    if (!this.cache || !this.lastFetch) {\n      return false;\n    }\n    \n    const age = Date.now() - this.lastFetch.getTime();\n    return age < this.ttl;\n  }\n\n  /**\n   * Determine if we should fetch fresh data\n   */\n  private shouldFetchFresh(): boolean {\n    // Always fetch if no cache\n    if (!this.cache || !this.lastFetch) {\n      return true;\n    }\n    \n    // Check for recent user actions\n    if (this.recentUserAction && this.actionTimestamp) {\n      const actionAge = Date.now() - this.actionTimestamp.getTime();\n      if (actionAge < this.actionGracePeriod) {\n        logger.debug('Fetching fresh due to recent user action', { actionAge });\n        return true;\n      } else {\n        // Grace period expired, clear action flag\n        this.recentUserAction = false;\n        this.actionTimestamp = null;\n      }\n    }\n    \n    // Check TTL\n    return !this.isCacheValid();\n  }\n\n  /**\n   * Create empty index when no portfolio exists\n   */\n  private createEmptyIndex(username?: string, repository?: string): GitHubPortfolioIndex {\n    const index: GitHubPortfolioIndex = {\n      username: username || 'unknown',\n      repository: repository || 'dollhouse-portfolio',\n      lastUpdated: new Date(),\n      elements: new Map(),\n      totalElements: 0,\n      sha: ''\n    };\n    \n    // Initialize empty element type maps\n    for (const elementType of Object.values(ElementType)) {\n      index.elements.set(elementType, []);\n    }\n    \n    return index;\n  }\n}"]}
|
|
495
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"GitHubPortfolioIndexer.js","sourceRoot":"","sources":["../../src/portfolio/GitHubPortfolioIndexer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * GitHub Portfolio Indexer - Fetches and indexes user's GitHub portfolio for fast searching\n * \n * Features:\n * - Singleton pattern for efficient resource usage\n * - Smart caching with TTL and invalidation after user actions\n * - GraphQL/REST API integration for efficient fetching\n * - Rate limiting and authentication handling\n * - Fallback strategy for resilient operation\n * - Performance optimized for 1000+ portfolio elements\n */\n\nimport { PortfolioRepoManager } from './PortfolioRepoManager.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { ElementType } from './types.js';\nimport { logger } from '../utils/logger.js';\nimport { SecurityMonitor } from '../security/securityMonitor.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { ErrorHandler, ErrorCategory } from '../utils/ErrorHandler.js';\nimport { APICache } from '../cache/APICache.js';\n\nexport interface GitHubIndexEntry {\n  path: string;\n  name: string;\n  description?: string;\n  version?: string;\n  author?: string;\n  elementType: ElementType;\n  sha: string; // File SHA for change detection\n  htmlUrl: string; // Link to GitHub\n  downloadUrl: string;\n  lastModified: Date;\n  size: number;\n}\n\nexport interface GitHubPortfolioIndex {\n  username: string;\n  repository: string;\n  lastUpdated: Date;\n  elements: Map<ElementType, GitHubIndexEntry[]>;\n  totalElements: number;\n  sha: string; // Latest commit SHA\n  rateLimitInfo?: {\n    remaining: number;\n    resetTime: Date;\n  };\n}\n\nexport interface GitHubFetchOptions {\n  force?: boolean;\n  maxElements?: number;\n  elementTypes?: ElementType[];\n  useGraphQL?: boolean;\n}\n\nexport class GitHubPortfolioIndexer {\n  private static instance: GitHubPortfolioIndexer | null = null;\n  private static instanceLock = false;\n  \n  private cache: GitHubPortfolioIndex | null = null;\n  private lastFetch: Date | null = null;\n  private readonly ttl = 15 * 60 * 1000; // 15 minutes\n  private recentUserAction = false;\n  private actionTimestamp: Date | null = null;\n  private readonly actionGracePeriod = 2 * 60 * 1000; // 2 minutes after action\n  \n  private portfolioRepoManager: PortfolioRepoManager;\n  private apiCache: APICache;\n  private rateLimitTracker: Map<string, number[]>;\n  \n  private constructor() {\n    this.apiCache = new APICache(); // Uses default settings\n    this.rateLimitTracker = new Map();\n    this.portfolioRepoManager = new PortfolioRepoManager();\n    \n    logger.debug('GitHubPortfolioIndexer created');\n  }\n\n  /**\n   * Singleton pattern with thread safety\n   */\n  public static getInstance(): GitHubPortfolioIndexer {\n    if (!this.instance) {\n      if (this.instanceLock) {\n        throw new Error('GitHubPortfolioIndexer instance is being created by another thread');\n      }\n      \n      try {\n        this.instanceLock = true;\n        this.instance = new GitHubPortfolioIndexer();\n      } finally {\n        this.instanceLock = false;\n      }\n    }\n    return this.instance;\n  }\n\n  /**\n   * Main method to get GitHub portfolio index\n   */\n  public async getIndex(force = false): Promise<GitHubPortfolioIndex> {\n    try {\n      // Check if we need fresh data\n      if (force || this.shouldFetchFresh()) {\n        return await this.fetchFresh();\n      }\n      \n      // Return cached data if available and valid\n      if (this.cache && this.isCacheValid()) {\n        logger.debug('Returning cached GitHub portfolio index', {\n          username: this.cache.username,\n          totalElements: this.cache.totalElements,\n          age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'\n        });\n        return this.cache;\n      }\n      \n      // Try to fetch fresh, fall back to stale cache on failure\n      try {\n        return await this.fetchFresh();\n      } catch (error) {\n        logger.warn('Failed to fetch fresh GitHub portfolio index, checking for stale cache', {\n          error: error instanceof Error ? error.message : String(error)\n        });\n        \n        // Return stale cache if available\n        if (this.cache) {\n          logger.info('Returning stale GitHub portfolio cache as fallback', {\n            username: this.cache.username,\n            age: this.lastFetch ? Date.now() - this.lastFetch.getTime() : 'unknown'\n          });\n          return this.cache;\n        }\n        \n        // Return empty index as last resort\n        return this.createEmptyIndex();\n      }\n      \n    } catch (error) {\n      ErrorHandler.logError('GitHubPortfolioIndexer.getIndex', error);\n      \n      // Return stale cache or empty index\n      if (this.cache) {\n        return this.cache;\n      }\n      \n      return this.createEmptyIndex();\n    }\n  }\n\n  /**\n   * Invalidate cache after user actions\n   */\n  public invalidateAfterAction(action: string): void {\n    logger.info('Invalidating GitHub portfolio cache after user action', { action });\n    \n    this.recentUserAction = true;\n    this.actionTimestamp = new Date();\n    \n    // Log security event for audit trail\n    SecurityMonitor.logSecurityEvent({\n      type: 'PORTFOLIO_CACHE_INVALIDATION',\n      severity: 'LOW',\n      source: 'GitHubPortfolioIndexer.invalidateAfterAction',\n      details: `Cache invalidated after user action: ${action}`,\n      metadata: { action }\n    });\n  }\n\n  /**\n   * Clear all cached data\n   */\n  public clearCache(): void {\n    this.cache = null;\n    this.lastFetch = null;\n    this.recentUserAction = false;\n    this.actionTimestamp = null;\n    this.apiCache.clear();\n    \n    logger.info('GitHub portfolio cache cleared');\n  }\n\n  /**\n   * Get cache statistics\n   */\n  public getCacheStats(): {\n    hasCachedData: boolean;\n    lastFetch: Date | null;\n    isStale: boolean;\n    recentUserAction: boolean;\n    totalElements: number;\n  } {\n    return {\n      hasCachedData: this.cache !== null,\n      lastFetch: this.lastFetch,\n      isStale: !this.isCacheValid(),\n      recentUserAction: this.recentUserAction,\n      totalElements: this.cache?.totalElements || 0\n    };\n  }\n\n  /**\n   * Fetch fresh data from GitHub\n   */\n  private async fetchFresh(): Promise<GitHubPortfolioIndex> {\n    const startTime = Date.now();\n    logger.info('Fetching fresh GitHub portfolio index...');\n    \n    try {\n      // Get GitHub username from token\n      const username = await this.getGitHubUsername();\n      const repository = 'dollhouse-portfolio';\n      \n      // Check if portfolio repository exists\n      const repoExists = await this.portfolioRepoManager.checkPortfolioExists(username);\n      if (!repoExists) {\n        logger.info('GitHub portfolio repository does not exist', { username });\n        return this.createEmptyIndex(username, repository);\n      }\n      \n      // Fetch repository content using GitHub API\n      const index = await this.fetchRepositoryContent(username, repository);\n      \n      // Update cache\n      this.cache = index;\n      this.lastFetch = new Date();\n      this.recentUserAction = false;\n      this.actionTimestamp = null;\n      \n      const duration = Date.now() - startTime;\n      logger.info('GitHub portfolio index fetched successfully', {\n        username,\n        totalElements: index.totalElements,\n        duration: `${duration}ms`,\n        rateLimitRemaining: index.rateLimitInfo?.remaining\n      });\n      \n      // Log security event\n      SecurityMonitor.logSecurityEvent({\n        type: 'PORTFOLIO_FETCH_SUCCESS',\n        severity: 'LOW',\n        source: 'GitHubPortfolioIndexer.fetchFresh',\n        details: `Fetched GitHub portfolio with ${index.totalElements} elements in ${duration}ms`,\n        metadata: { username, duration, totalElements: index.totalElements }\n      });\n      \n      return index;\n      \n    } catch (error) {\n      const duration = Date.now() - startTime;\n      ErrorHandler.logError('GitHubPortfolioIndexer.fetchFresh', error, { duration });\n      throw ErrorHandler.wrapError(error, 'Failed to fetch GitHub portfolio index', ErrorCategory.NETWORK_ERROR);\n    }\n  }\n\n  /**\n   * Fetch repository content from GitHub API\n   */\n  private async fetchRepositoryContent(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    // Try GraphQL first for better performance, fallback to REST\n    try {\n      return await this.fetchWithGraphQL(username, repository);\n    } catch (graphqlError) {\n      logger.debug('GraphQL fetch failed, falling back to REST API', {\n        error: graphqlError instanceof Error ? graphqlError.message : String(graphqlError)\n      });\n      \n      return await this.fetchWithREST(username, repository);\n    }\n  }\n\n  /**\n   * Fetch using GraphQL for better performance\n   */\n  private async fetchWithGraphQL(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    const query = `\n      query GetPortfolioContent($owner: String!, $name: String!) {\n        repository(owner: $owner, name: $name) {\n          defaultBranchRef {\n            target {\n              ... on Commit {\n                oid\n                history(first: 1) {\n                  nodes {\n                    committedDate\n                  }\n                }\n              }\n            }\n          }\n          object(expression: \"HEAD:\") {\n            ... on Tree {\n              entries {\n                name\n                type\n                object {\n                  ... on Tree {\n                    entries {\n                      name\n                      type\n                      oid\n                      object {\n                        ... on Blob {\n                          byteSize\n                          text\n                        }\n                      }\n                    }\n                  }\n                }\n              }\n            }\n          }\n        }\n        rateLimit {\n          remaining\n          resetAt\n        }\n      }\n    `;\n    \n    const variables = { owner: username, name: repository };\n    \n    // GraphQL endpoint not yet implemented with PortfolioRepoManager\n    // For now, just throw to fall back to REST\n    throw new Error('GraphQL not yet implemented for portfolio operations');\n    \n    // Note: This is a simplified GraphQL implementation\n    // In a real implementation, you would send POST request with query and variables\n    throw new Error('GraphQL implementation not yet complete');\n  }\n\n  /**\n   * Fetch using REST API with pagination\n   */\n  private async fetchWithREST(username: string, repository: string): Promise<GitHubPortfolioIndex> {\n    const normalizedUsername = UnicodeValidator.normalize(username).normalizedContent;\n    \n    // Get repository info and latest commit\n    const repoInfo = await this.portfolioRepoManager.githubRequest(\n      `/repos/${normalizedUsername}/${repository}`\n    );\n    \n    const latestCommit = await this.portfolioRepoManager.githubRequest(\n      `/repos/${normalizedUsername}/${repository}/commits/HEAD`\n    );\n    \n    // Initialize index\n    const index: GitHubPortfolioIndex = {\n      username: normalizedUsername,\n      repository,\n      lastUpdated: new Date(latestCommit.commit.committer.date),\n      elements: new Map(),\n      totalElements: 0,\n      sha: latestCommit.sha\n    };\n    \n    // Initialize element type maps\n    for (const elementType of Object.values(ElementType)) {\n      index.elements.set(elementType, []);\n    }\n    \n    // Fetch content for each element type\n    logger.info(`Fetching content for all element types from GitHub portfolio...`);\n    \n    for (const elementType of Object.values(ElementType)) {\n      try {\n        logger.debug(`Fetching ${elementType} from GitHub...`);\n        const entries = await this.fetchElementTypeContent(normalizedUsername, repository, elementType);\n        index.elements.set(elementType, entries);\n        index.totalElements += entries.length;\n        \n        if (entries.length > 0) {\n          logger.info(`✅ Found ${entries.length} ${elementType} in GitHub portfolio`);\n        } else {\n          logger.debug(`📁 No ${elementType} found (directory may not exist yet)`);\n        }\n      } catch (error) {\n        logger.warn(`❌ Failed to fetch ${elementType} from GitHub portfolio`, {\n          error: error instanceof Error ? error.message : String(error),\n          elementType\n        });\n        // Continue with other element types\n      }\n    }\n    \n    return index;\n  }\n\n  /**\n   * Fetch content for a specific element type\n   */\n  private async fetchElementTypeContent(\n    username: string,\n    repository: string,\n    elementType: ElementType\n  ): Promise<GitHubIndexEntry[]> {\n    try {\n      // Get directory listing using PortfolioRepoManager\n      const contents = await this.portfolioRepoManager.githubRequest(\n        `/repos/${username}/${repository}/contents/${elementType}`\n      );\n      \n      if (!Array.isArray(contents)) {\n        return [];\n      }\n      \n      const entries: GitHubIndexEntry[] = [];\n      const maxConcurrent = 5; // Limit concurrent requests\n      \n      // Process files in batches to avoid rate limiting\n      for (let i = 0; i < contents.length; i += maxConcurrent) {\n        const batch = contents.slice(i, i + maxConcurrent);\n        const batchPromises = batch\n          .filter(item => item.type === 'file' && item.name.endsWith('.md'))\n          .map(item => this.createGitHubIndexEntry(username, repository, elementType, item));\n        \n        const batchResults = await Promise.allSettled(batchPromises);\n        \n        for (const result of batchResults) {\n          if (result.status === 'fulfilled' && result.value) {\n            entries.push(result.value);\n          }\n        }\n        \n        // Add delay between batches to respect rate limits\n        if (i + maxConcurrent < contents.length) {\n          await new Promise(resolve => setTimeout(resolve, 100));\n        }\n      }\n      \n      return entries;\n      \n    } catch (error) {\n      // Directory might not exist - check for 404 errors\n      // PortfolioRepoManager will throw standard GitHub API errors\n      if (error instanceof Error) {\n        // Check for 404 Not Found errors from GitHub API\n        if (error.message.includes('404') || \n            error.message.includes('Not Found')) {\n          logger.debug(`Directory ${elementType} not found in GitHub repository (this is normal if not yet created)`);\n          return [];\n        }\n        \n        // Log the actual error for debugging\n        logger.debug(`Error fetching ${elementType}: ${error.message}`);\n      }\n      \n      // Re-throw other errors\n      logger.warn(`Unexpected error fetching ${elementType}:`, error);\n      throw error;\n    }\n  }\n\n  /**\n   * Create GitHub index entry from API response\n   */\n  private async createGitHubIndexEntry(\n    username: string,\n    repository: string,\n    elementType: ElementType,\n    fileInfo: any\n  ): Promise<GitHubIndexEntry | null> {\n    try {\n      // Parse metadata from filename or fetch content if needed\n      const name = fileInfo.name.replace('.md', '').replace(/-/g, ' ');\n      \n      const entry: GitHubIndexEntry = {\n        path: fileInfo.path,\n        name,\n        elementType,\n        sha: fileInfo.sha,\n        htmlUrl: fileInfo.html_url,\n        downloadUrl: fileInfo.download_url,\n        lastModified: new Date(), // GitHub API doesn't provide file modification time directly\n        size: fileInfo.size || 0\n      };\n      \n      // Optionally fetch content to extract metadata\n      // This is expensive, so only do it for small files or when specifically needed\n      if (fileInfo.size && fileInfo.size < 10000) { // Only for files < 10KB\n        try {\n          // For download URLs, we need to fetch directly, not through API\n          const response = await fetch(fileInfo.download_url);\n          const content = await response.text();\n          const metadata = this.parseMetadataFromContent(content);\n          \n          if (metadata.name) entry.name = metadata.name;\n          if (metadata.description) entry.description = metadata.description;\n          if (metadata.version) entry.version = metadata.version;\n          if (metadata.author) entry.author = metadata.author;\n        } catch (metadataError) {\n          // Non-critical error, continue without metadata\n          logger.debug('Failed to fetch metadata for file', {\n            path: fileInfo.path,\n            error: metadataError instanceof Error ? metadataError.message : String(metadataError)\n          });\n        }\n      }\n      \n      return entry;\n      \n    } catch (error) {\n      logger.debug('Failed to create GitHub index entry', {\n        path: fileInfo.path,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      return null;\n    }\n  }\n\n  /**\n   * Parse metadata from file content (frontmatter)\n   */\n  private parseMetadataFromContent(content: string): {\n    name?: string;\n    description?: string;\n    version?: string;\n    author?: string;\n  } {\n    const metadata: any = {};\n    \n    // Simple frontmatter parsing (could use a proper YAML parser)\n    const frontmatterMatch = content.match(/^---\\n([\\s\\S]*?)\\n---/);\n    if (frontmatterMatch) {\n      const frontmatter = frontmatterMatch[1];\n      \n      const nameMatch = frontmatter.match(/^name:\\s*(.+)$/m);\n      if (nameMatch) metadata.name = nameMatch[1].trim();\n      \n      const descMatch = frontmatter.match(/^description:\\s*(.+)$/m);\n      if (descMatch) metadata.description = descMatch[1].trim();\n      \n      const versionMatch = frontmatter.match(/^version:\\s*(.+)$/m);\n      if (versionMatch) metadata.version = versionMatch[1].trim();\n      \n      const authorMatch = frontmatter.match(/^author:\\s*(.+)$/m);\n      if (authorMatch) metadata.author = authorMatch[1].trim();\n    }\n    \n    return metadata;\n  }\n\n  /**\n   * Get GitHub username from authenticated token\n   */\n  private async getGitHubUsername(): Promise<string> {\n    try {\n      const userInfo = await this.portfolioRepoManager.githubRequest('/user');\n      return userInfo.login;\n    } catch (error) {\n      throw new Error('Failed to get GitHub username. Please ensure you are authenticated with GitHub.');\n    }\n  }\n\n  /**\n   * Check if cache is valid\n   */\n  private isCacheValid(): boolean {\n    if (!this.cache || !this.lastFetch) {\n      return false;\n    }\n    \n    const age = Date.now() - this.lastFetch.getTime();\n    return age < this.ttl;\n  }\n\n  /**\n   * Determine if we should fetch fresh data\n   */\n  private shouldFetchFresh(): boolean {\n    // Always fetch if no cache\n    if (!this.cache || !this.lastFetch) {\n      return true;\n    }\n    \n    // Check for recent user actions\n    if (this.recentUserAction && this.actionTimestamp) {\n      const actionAge = Date.now() - this.actionTimestamp.getTime();\n      if (actionAge < this.actionGracePeriod) {\n        logger.debug('Fetching fresh due to recent user action', { actionAge });\n        return true;\n      } else {\n        // Grace period expired, clear action flag\n        this.recentUserAction = false;\n        this.actionTimestamp = null;\n      }\n    }\n    \n    // Check TTL\n    return !this.isCacheValid();\n  }\n\n  /**\n   * Create empty index when no portfolio exists\n   */\n  private createEmptyIndex(username?: string, repository?: string): GitHubPortfolioIndex {\n    const index: GitHubPortfolioIndex = {\n      username: username || 'unknown',\n      repository: repository || 'dollhouse-portfolio',\n      lastUpdated: new Date(),\n      elements: new Map(),\n      totalElements: 0,\n      sha: ''\n    };\n    \n    // Initialize empty element type maps\n    for (const elementType of Object.values(ElementType)) {\n      index.elements.set(elementType, []);\n    }\n    \n    return index;\n  }\n}"]}
|
|
@@ -33,8 +33,9 @@ export declare class PortfolioRepoManager {
|
|
|
33
33
|
private getTokenAndValidate;
|
|
34
34
|
/**
|
|
35
35
|
* Make authenticated GitHub API request
|
|
36
|
+
* Made public to support GitHubPortfolioIndexer operations
|
|
36
37
|
*/
|
|
37
|
-
|
|
38
|
+
githubRequest(path: string, method?: string, body?: any): Promise<any>;
|
|
38
39
|
/**
|
|
39
40
|
* Check if portfolio repository exists for a user
|
|
40
41
|
* No consent required - this is a read-only operation
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"PortfolioRepoManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioRepoManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,+BAA+B,CAAC;AAOzD,MAAM,WAAW,oBAAoB;IACnC,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAyB;IACpE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAuC;IAClF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAA4B;IAEnE,OAAO,CAAC,KAAK,CAAuB;;IAMpC;;;OAGG;IACI,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAIpC;;;;OAIG;YACW,mBAAmB;IA6BjC
|
|
1
|
+
{"version":3,"file":"PortfolioRepoManager.d.ts","sourceRoot":"","sources":["../../src/portfolio/PortfolioRepoManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAEH,OAAO,EAAE,QAAQ,EAAE,MAAM,+BAA+B,CAAC;AAOzD,MAAM,WAAW,oBAAoB;IACnC,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,SAAS,CAAC,EAAE,OAAO,CAAC;CACrB;AAED,qBAAa,oBAAoB;IAC/B,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAyB;IACpE,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,CAAuC;IAClF,OAAO,CAAC,MAAM,CAAC,QAAQ,CAAC,eAAe,CAA4B;IAEnE,OAAO,CAAC,KAAK,CAAuB;;IAMpC;;;OAGG;IACI,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAIpC;;;;OAIG;YACW,mBAAmB;IA6BjC;;;OAGG;IACU,aAAa,CACxB,IAAI,EAAE,MAAM,EACZ,MAAM,GAAE,MAAc,EACtB,IAAI,CAAC,EAAE,GAAG,GACT,OAAO,CAAC,GAAG,CAAC;IA2Ef;;;;OAIG;IACG,oBAAoB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAe9D;;;OAGG;IACG,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;IA+EtF;;;OAGG;IACG,WAAW,CAAC,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;IA2OnF;;;OAGG;IACG,0BAA0B,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA+CjE;;;OAGG;IACH,OAAO,CAAC,eAAe;IAcvB;;;;OAIG;WACW,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM;IAwBpD;;OAEG;IACH,OAAO,CAAC,oBAAoB;CAQ7B"}
|