@eclipse-lyra/extension-rag-system 0.7.5 → 0.7.7
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/dist/index.js
CHANGED
|
@@ -14,7 +14,7 @@ extensionRegistry.registerExtension({
|
|
|
14
14
|
id: pkg.name,
|
|
15
15
|
name: t("EXT_RAG_SYSTEM_NAME"),
|
|
16
16
|
description: t("EXT_RAG_SYSTEM_DESC"),
|
|
17
|
-
loader: () => import("./rag-system-extension-
|
|
17
|
+
loader: () => import("./rag-system-extension-CE4P1Thg.js"),
|
|
18
18
|
icon: "database",
|
|
19
19
|
experimental: true,
|
|
20
20
|
dependencies: ["@eclipse-lyra/extension-ai-system", "@eclipse-lyra/extension-in-browser-ml"]
|
|
@@ -1087,7 +1087,7 @@ function ragSystemExtension(context) {
|
|
|
1087
1087
|
return input.key === ".system.rag-system";
|
|
1088
1088
|
},
|
|
1089
1089
|
handle: async (input) => {
|
|
1090
|
-
input.
|
|
1090
|
+
input.component = () => html`
|
|
1091
1091
|
<lyra-rag-system-manager .input=${input}></lyra-rag-system-manager>
|
|
1092
1092
|
`;
|
|
1093
1093
|
return input;
|
|
@@ -1139,4 +1139,4 @@ function ragSystemExtension(context) {
|
|
|
1139
1139
|
export {
|
|
1140
1140
|
ragSystemExtension as default
|
|
1141
1141
|
};
|
|
1142
|
-
//# sourceMappingURL=rag-system-extension-
|
|
1142
|
+
//# sourceMappingURL=rag-system-extension-CE4P1Thg.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"rag-system-extension-DfD6H8Vr.js","sources":["../src/rag-system-manager.ts","../src/rag-system-extension.ts"],"sourcesContent":["import { customElement, property, state } from 'lit/decorators.js';\nimport { css, html, TemplateResult, nothing } from 'lit';\nimport { createRef, ref } from 'lit/directives/ref.js';\nimport { LyraPart } from '@eclipse-lyra/core';\nimport { EditorInput } from '@eclipse-lyra/core';\nimport { documentIndexService, IndexedDocument } from './document-index-service';\nimport { searchWorkspaceDocuments, RAGSearchResult } from './rag-service';\nimport { workspaceService, File } from '@eclipse-lyra/core';\nimport { createLogger } from '@eclipse-lyra/core';\nimport { getWorkspacePath } from './utils/workspace-utils';\nimport { SnippetExtractor } from './utils/snippet-extractor';\nimport { CONTENT_PREVIEW_LENGTHS, SNIPPET_LENGTHS } from './utils/constants';\nimport { toastError, toastInfo } from '@eclipse-lyra/core';\nimport { taskService } from '@eclipse-lyra/core';\nimport { editorRegistry } from '@eclipse-lyra/core';\n\nconst logger = createLogger('RAGSystemManager');\nconst snippetExtractor = new SnippetExtractor();\n\n@customElement('lyra-rag-system-manager')\nexport class LyraRAGSystemManager extends LyraPart {\n @property({ attribute: false })\n public input?: EditorInput;\n\n @state()\n private documents: IndexedDocument[] = [];\n\n\n @state()\n private stats: { totalDocuments: number; byWorkspace: Record<string, number> } = {\n totalDocuments: 0,\n byWorkspace: {}\n };\n\n @state()\n private loading = false;\n\n @state()\n private selectedDocument: IndexedDocument | null = null;\n\n @state()\n private searchQuery = '';\n\n @state()\n private filterWorkspace: string | null = null;\n\n @state()\n private filterByActiveWorkspace = true;\n\n @state()\n private filteredDocuments: IndexedDocument[] = [];\n \n @state()\n private searchResults: Map<string, RAGSearchResult> = new Map();\n\n @state()\n private reindexing = false;\n\n private treeRef = createRef<HTMLElement>();\n private searchDebounceTimer?: number;\n private searchInputValue = '';\n\n protected async doInitUI() {\n try {\n await documentIndexService.initialize();\n await Promise.all([\n this.loadDocuments(),\n this.loadStats()\n ]);\n } catch (error) {\n logger.error(`Failed to initialize document index manager: ${error}`);\n toastError(`Failed to initialize: ${error}`);\n }\n }\n\n private async loadDocuments() {\n this.loading = true;\n this.requestUpdate();\n try {\n let workspacePath: string | undefined;\n if (this.filterByActiveWorkspace) {\n const workspaceResult = await getWorkspacePath();\n workspacePath = workspaceResult?.workspacePath;\n }\n this.documents = await documentIndexService.listDocuments(workspacePath);\n await this.updateFilteredDocuments();\n } catch (error) {\n logger.error(`Failed to load documents: ${error}`);\n toastError(`Failed to load documents: ${error}`);\n } finally {\n this.loading = false;\n }\n }\n\n private async updateFilteredDocuments() {\n this.filteredDocuments = await this.getFilteredDocuments();\n this.requestUpdate();\n }\n\n private async loadStats() {\n try {\n this.stats = await documentIndexService.getStats();\n this.requestUpdate();\n } catch (error) {\n logger.error(`Failed to load stats: ${error}`);\n }\n }\n\n private handleTreeSelection(e: CustomEvent) {\n // Try getting from event detail first (as per extensions pattern)\n let selection = e.detail?.selection || [];\n\n // If no selection in detail, try getting from tree element directly (as per lyra-filebrowser pattern)\n if (selection.length === 0 && this.treeRef.value) {\n // @ts-ignore\n selection = this.treeRef.value.selectedItems || [];\n }\n\n if (selection.length > 0) {\n const selectedItem = selection[0];\n if (selectedItem?.model) {\n this.selectedDocument = selectedItem.model;\n } else {\n this.selectedDocument = null;\n }\n } else {\n this.selectedDocument = null;\n }\n }\n\n private async getFilteredDocuments(): Promise<IndexedDocument[]> {\n if (!this.documents || this.documents.length === 0) {\n return [];\n }\n\n if (this.searchQuery.trim()) {\n try {\n const searchResults = await searchWorkspaceDocuments(this.searchQuery, {\n limit: 50,\n workspacePath: this.filterWorkspace || undefined\n });\n\n this.searchResults.clear();\n const uniqueDocs = new Map<string, IndexedDocument>();\n const aggregatedResults = new Map<string, RAGSearchResult>();\n \n for (const result of searchResults) {\n const docId = result.document.id;\n uniqueDocs.set(docId, result.document);\n \n const existing = aggregatedResults.get(docId);\n if (!existing) {\n aggregatedResults.set(docId, {\n document: result.document,\n relevance: result.relevance,\n matchedSnippets: [...result.matchedSnippets]\n });\n } else {\n if (result.relevance > existing.relevance) {\n existing.relevance = result.relevance;\n }\n const seenSnippets = new Set(existing.matchedSnippets);\n for (const snippet of result.matchedSnippets) {\n if (!seenSnippets.has(snippet)) {\n existing.matchedSnippets.push(snippet);\n seenSnippets.add(snippet);\n }\n }\n }\n }\n \n for (const [docId, result] of aggregatedResults) {\n this.searchResults.set(docId, result);\n }\n\n const resultDocs = Array.from(uniqueDocs.values());\n \n if (this.filterWorkspace) {\n return resultDocs.filter(doc => doc.workspacePath === this.filterWorkspace);\n }\n \n return resultDocs;\n } catch (error) {\n logger.debug(`RAG search failed in document manager: ${error}`);\n this.searchResults.clear();\n return [];\n }\n } else {\n this.searchResults.clear();\n }\n\n let filtered = [...this.documents];\n\n if (this.filterWorkspace) {\n filtered = filtered.filter(doc => doc.workspacePath === this.filterWorkspace);\n }\n\n return filtered;\n }\n\n private highlightMatches(text: string, query: string): TemplateResult {\n if (!query || !query.trim()) {\n return html`${text}`;\n }\n\n const queryLower = query.toLowerCase();\n const textLower = text.toLowerCase();\n const parts: Array<string | TemplateResult> = [];\n let lastIndex = 0;\n let index = textLower.indexOf(queryLower, lastIndex);\n\n while (index !== -1) {\n if (index > lastIndex) {\n parts.push(text.substring(lastIndex, index));\n }\n parts.push(html`<mark class=\"search-match\">${text.substring(index, index + query.length)}</mark>`);\n lastIndex = index + query.length;\n index = textLower.indexOf(queryLower, lastIndex);\n }\n\n if (lastIndex < text.length) {\n parts.push(text.substring(lastIndex));\n }\n\n return html`${parts}`;\n }\n\n private getContentPreview(doc: IndexedDocument): TemplateResult {\n const searchResult = this.searchResults.get(doc.id);\n \n if (searchResult && searchResult.matchedSnippets.length > 0) {\n return html`\n <table class=\"snippets-table\">\n <thead>\n <tr>\n <th class=\"snippet-number-col\">#</th>\n <th class=\"snippet-content-col\">Content</th>\n </tr>\n </thead>\n <tbody>\n ${searchResult.matchedSnippets.map((snippet, idx) => html`\n <tr>\n <td class=\"snippet-number\">${idx + 1}</td>\n <td class=\"snippet-content\">${this.highlightMatches(snippet, this.searchQuery)}</td>\n </tr>\n `)}\n </tbody>\n </table>\n `;\n }\n\n if (this.searchQuery && this.searchQuery.trim()) {\n const snippets = snippetExtractor.extractContextSnippets(\n doc.content,\n this.searchQuery,\n SNIPPET_LENGTHS.CONTEXT\n );\n\n if (snippets.length > 0) {\n return html`\n <table class=\"snippets-table\">\n <thead>\n <tr>\n <th class=\"snippet-number-col\">#</th>\n <th class=\"snippet-content-col\">Content</th>\n </tr>\n </thead>\n <tbody>\n ${snippets.map((snippet, idx) => html`\n <tr>\n <td class=\"snippet-number\">${idx + 1}</td>\n <td class=\"snippet-content\">${this.highlightMatches(doc.content.substring(snippet.start, snippet.end), this.searchQuery)}</td>\n </tr>\n `)}\n </tbody>\n </table>\n `;\n }\n }\n\n const preview = snippetExtractor.extractSimpleSnippet(doc.content, CONTENT_PREVIEW_LENGTHS.LONG);\n return html`\n <div class=\"snippet-preview\">${preview}</div>\n `;\n }\n\n private async deleteDocument(doc: IndexedDocument) {\n try {\n await documentIndexService.deleteDocument(doc.id);\n toastInfo(`Deleted: ${doc.fileName}`);\n await this.loadDocuments();\n await this.loadStats();\n if (this.selectedDocument?.id === doc.id) {\n this.selectedDocument = null;\n }\n } catch (error) {\n toastError(`Failed to delete document: ${error}`);\n }\n }\n\n private async reindexDocument(doc: IndexedDocument) {\n try {\n const workspaceResult = await getWorkspacePath();\n if (!workspaceResult) {\n toastError('No workspace connected');\n return;\n }\n\n const resource = await workspaceResult.workspace.getResource(doc.filePath);\n if (!resource) {\n toastError(`File not found: ${doc.filePath}`);\n return;\n }\n\n if (!(resource instanceof File)) {\n toastError(`Resource is not a file: ${doc.filePath}`);\n return;\n }\n\n const file: File = resource;\n\n await taskService.runAsync('Reindexing document', async progress => {\n progress.message = `Reindexing ${doc.fileName}...`;\n await documentIndexService.reindexDocument(file);\n progress.progress = 100;\n });\n\n toastInfo(`Reindexed: ${doc.fileName}`);\n await this.loadDocuments();\n if (this.selectedDocument?.id === doc.id) {\n this.selectedDocument = await documentIndexService.getDocument(doc.id);\n }\n } catch (error) {\n toastError(`Failed to reindex document: ${error}`);\n }\n }\n\n private async reindexAllDocuments() {\n if (this.reindexing) {\n return;\n }\n\n const stats = await documentIndexService.getStats();\n if (stats.totalDocuments === 0) {\n toastInfo('No documents to reindex');\n return;\n }\n\n this.reindexing = true;\n this.requestUpdate();\n\n try {\n const result = await taskService.runAsync('Reindexing all documents', async progress => {\n progress.message = 'Starting reindexing...';\n \n const total = stats.totalDocuments;\n\n const reindexResult = await documentIndexService.reindexAllDocuments();\n \n const processed = reindexResult.succeeded + reindexResult.failed;\n progress.progress = total > 0 ? (processed / total) * 100 : 100;\n progress.message = `Reindexed ${reindexResult.succeeded}/${total} documents${reindexResult.failed > 0 ? ` (${reindexResult.failed} failed)` : ''}`;\n \n return reindexResult;\n });\n\n await this.loadDocuments();\n await this.loadStats();\n toastInfo(`Reindexing completed: ${result.succeeded} succeeded, ${result.failed} failed`);\n } catch (error) {\n logger.error(`Failed to reindex all documents: ${error}`);\n toastError(`Failed to reindex all documents: ${error}`);\n } finally {\n this.reindexing = false;\n this.requestUpdate();\n }\n }\n\n private formatFileSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} LyraB`;\n return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n }\n\n private formatDate(timestamp: number): string {\n return new Date(timestamp).toLocaleString();\n }\n\n private getFileIcon(fileType: string): string {\n return editorRegistry.getFileIcon(fileType);\n }\n\n protected renderToolbar(): TemplateResult {\n const workspaces = Object.keys(this.stats?.byWorkspace || {});\n\n return html`\n <wa-input\n type=\"search\"\n placeholder=\"Search documents...\"\n .value=${this.searchInputValue}\n @input=${(e: any) => {\n this.searchInputValue = e.target.value;\n \n if (this.searchDebounceTimer) {\n clearTimeout(this.searchDebounceTimer);\n }\n \n this.searchDebounceTimer = window.setTimeout(async () => {\n this.searchQuery = this.searchInputValue;\n await this.updateFilteredDocuments();\n }, 200);\n }}\n @wa-clear=${async () => {\n if (this.searchDebounceTimer) {\n clearTimeout(this.searchDebounceTimer);\n }\n this.searchInputValue = '';\n this.searchQuery = '';\n await this.updateFilteredDocuments();\n }}\n size=\"small\"\n with-clear\n autocomplete=\"off\"\n style=\"flex: 1; max-width: 400px;\">\n <wa-icon name=\"magnifying-glass\" slot=\"start\"></wa-icon>\n </wa-input>\n \n <wa-switch\n .checked=${this.filterByActiveWorkspace}\n @change=${async (e: any) => {\n this.filterByActiveWorkspace = e.target.checked;\n await this.loadDocuments();\n }}\n size=\"small\">\n Active workspace only\n </wa-switch>\n \n ${workspaces.length > 1 ? html`\n <wa-select \n .value=${this.filterWorkspace || ''}\n @change=${async (e: any) => {\n this.filterWorkspace = e.target.value || null;\n await this.updateFilteredDocuments();\n }}\n size=\"small\"\n style=\"width: 200px;\">\n <wa-option value=\"\">All Workspaces</wa-option>\n ${workspaces.map(ws => html`\n <wa-option value=\"${ws}\">${ws} (${this.stats.byWorkspace[ws]})</wa-option>\n `)}\n </wa-select>\n ` : nothing}\n \n <lyra-command \n size=\"small\" \n icon=\"arrow-rotate-right\"\n title=\"Refresh document list\"\n .action=${() => this.loadDocuments()}\n ?disabled=${this.reindexing}>\n Refresh\n </lyra-command>\n \n <lyra-command \n size=\"small\" \n icon=\"database\"\n title=\"Re-index all documents\"\n .action=${() => this.reindexAllDocuments()}\n ?disabled=${this.reindexing || this.loading}>\n ${this.reindexing ? 'Reindexing...' : 'Re-index All'}\n </lyra-command>\n `;\n }\n\n protected render(): TemplateResult {\n if (!this.stats) {\n this.stats = { totalDocuments: 0, byWorkspace: {} };\n }\n\n const filteredDocs = this.filteredDocuments;\n const workspaces = Object.keys(this.stats.byWorkspace || {});\n\n return html`\n <div class=\"rag-system-manager\">\n <div class=\"header\">\n <div class=\"header-content\">\n <div class=\"stats\">\n <span>Total: ${this.stats.totalDocuments} documents</span>\n ${workspaces.length > 0 ? html`\n <span>Workspaces: ${workspaces.length}</span>\n ` : nothing}\n </div>\n </div>\n </div>\n\n <wa-split-panel position=\"40\" style=\"height: 100%;\">\n <div class=\"document-list\" slot=\"start\">\n ${this.loading ? html`\n <div class=\"loading\">\n <wa-spinner></wa-spinner>\n <span>Loading documents...</span>\n </div>\n ` : filteredDocs.length === 0 ? html`\n <div class=\"empty\">\n <wa-icon name=\"inbox\" style=\"font-size: 3rem; opacity: 0.3;\"></wa-icon>\n <p>${this.searchQuery || this.filterWorkspace ? 'No documents match your filters' : 'No documents indexed yet'}</p>\n </div>\n ` : html`\n <wa-tree \n ${ref(this.treeRef)}\n selection=\"leaf\"\n style=\"--indent-guide-width: 1px;\"\n @wa-selection-change=${(e: CustomEvent) => {\n this.handleTreeSelection(e);\n }}>\n ${filteredDocs.map(doc => html`\n <wa-tree-item \n .model=${doc}\n ?selected=${this.selectedDocument?.id === doc.id}>\n <wa-icon name=\"${this.getFileIcon(doc.fileType)}\"></wa-icon>\n <div class=\"tree-item-info\">\n <strong class=\"tree-item-path\">${doc.filePath}</strong>\n <div class=\"tree-item-meta\">\n <small class=\"meta-size\">${this.formatFileSize(doc.metadata.size)}</small>\n <small class=\"meta-date\">${this.formatDate(doc.indexedAt)}</small>\n </div>\n </div>\n <div class=\"tree-item-actions\" @click=${(e: Event) => e.stopPropagation()}>\n <wa-button\n variant=\"neutral\"\n appearance=\"plain\"\n size=\"small\"\n title=\"Reindex\"\n @click=${() => this.reindexDocument(doc)}>\n <wa-icon name=\"arrow-rotate-right\"></wa-icon>\n </wa-button>\n <wa-button\n variant=\"danger\"\n appearance=\"plain\"\n size=\"small\"\n title=\"Delete\"\n @click=${() => this.deleteDocument(doc)}>\n <wa-icon name=\"trash\"></wa-icon>\n </wa-button>\n </div>\n </wa-tree-item>\n `)}\n </wa-tree>\n `}\n </div>\n\n <div slot=\"end\">\n ${this.selectedDocument ? html`\n <div class=\"document-details\">\n <div class=\"details-content\">\n <div class=\"metadata-grid\">\n <wa-input\n label=\"File Path\"\n .value=${this.selectedDocument.filePath}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.filePath}\n size=\"small\"\n label=\"Copy file path\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Workspace\"\n .value=${this.selectedDocument.workspacePath}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.workspacePath}\n size=\"small\"\n label=\"Copy workspace\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"File Type\"\n .value=${this.selectedDocument.fileType}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.fileType}\n size=\"small\"\n label=\"Copy file type\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Size\"\n .value=${this.formatFileSize(this.selectedDocument.metadata.size)}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.formatFileSize(this.selectedDocument.metadata.size)}\n size=\"small\"\n label=\"Copy size\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Indexed At\"\n .value=${this.formatDate(this.selectedDocument.indexedAt)}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.formatDate(this.selectedDocument.indexedAt)}\n size=\"small\"\n label=\"Copy indexed date\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Last Updated\"\n .value=${this.formatDate(this.selectedDocument.updatedAt)}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.formatDate(this.selectedDocument.updatedAt)}\n size=\"small\"\n label=\"Copy updated date\">\n </wa-copy-button>\n </wa-input>\n </div>\n \n ${this.selectedDocument.metadata.tags && this.selectedDocument.metadata.tags.length > 0 ? html`\n <div class=\"tags-section\">\n <wa-input\n label=\"Tags\"\n .value=${this.selectedDocument.metadata.tags.join(', ')}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.metadata.tags.join(', ')}\n size=\"small\"\n label=\"Copy tags\">\n </wa-copy-button>\n </wa-input>\n </div>\n ` : nothing}\n \n <div class=\"detail-section\">\n <label>Content Preview${this.searchQuery ? html` <span class=\"search-hint\">(showing matches for \"${this.searchQuery}\")</span>` : nothing}</label>\n <wa-scroller class=\"content-preview\" orientation=\"vertical\">\n <div class=\"content-preview-inner\">\n ${this.getContentPreview(this.selectedDocument)}\n </div>\n </wa-scroller>\n </div>\n </div>\n </div>\n ` : html`\n <div class=\"document-details empty\">\n <wa-icon name=\"file-lines\" style=\"font-size: 3rem; opacity: 0.3;\"></wa-icon>\n <p>Select a document to view details</p>\n </div>\n `}\n </div>\n </wa-split-panel>\n </div>\n `;\n }\n\n static styles = css`\n :host {\n display: flex;\n flex-direction: column;\n height: 100%;\n overflow: hidden;\n min-height: 0;\n }\n\n .rag-system-manager {\n display: flex;\n flex-direction: column;\n height: 100%;\n min-height: 0;\n overflow: hidden;\n }\n\n wa-split-panel {\n flex: 1;\n min-height: 0;\n overflow: hidden;\n }\n\n .document-list {\n height: 100%;\n overflow-y: auto;\n }\n\n .tree-item-info {\n flex: 1;\n min-width: 0;\n }\n\n .tree-item-path {\n overflow: hidden;\n text-overflow: ellipsis;\n white-space: nowrap;\n }\n\n .tree-item-meta {\n display: flex;\n align-items: center;\n gap: var(--wa-space-xs);\n flex-wrap: wrap;\n margin-top: var(--wa-space-xs);\n }\n\n .tree-item-actions {\n opacity: 0;\n }\n\n wa-tree-item:hover .tree-item-actions {\n opacity: 1;\n }\n\n .document-details {\n height: 100%;\n display: flex;\n flex-direction: column;\n min-height: 0;\n overflow: hidden;\n }\n\n .details-content {\n flex: 1;\n display: flex;\n flex-direction: column;\n padding: var(--wa-space-s);\n min-height: 0;\n overflow: hidden;\n }\n\n .metadata-grid {\n display: grid;\n grid-template-columns: 1fr 1fr;\n gap: var(--wa-space-s);\n flex-shrink: 0;\n }\n\n .tags-section {\n margin-top: var(--wa-space-s);\n flex-shrink: 0;\n margin-bottom: var(--wa-space-s);\n }\n\n .detail-section {\n flex: 1;\n display: flex;\n flex-direction: column;\n min-height: 0;\n position: relative;\n }\n\n .detail-section label {\n flex-shrink: 0;\n margin-bottom: var(--wa-space-xs);\n }\n\n .content-preview {\n position: absolute;\n top: 0;\n right: 0;\n left: 0;\n bottom: 0;\n }\n\n .content-preview-inner {\n width: 100%;\n }\n\n .snippets-table {\n width: 100%;\n border-collapse: collapse;\n background-color: var(--wa-color-surface-raised);\n }\n\n .snippets-table thead {\n background-color: var(--wa-color-neutral-fill-quiet);\n }\n\n .snippets-table th {\n padding: var(--wa-space-xs) var(--wa-space-s);\n text-align: left;\n font-size: 0.75rem;\n font-weight: 600;\n color: var(--wa-color-text-quiet);\n border-bottom: 1px solid var(--wa-color-surface-border);\n }\n\n .snippets-table td {\n padding: var(--wa-space-s);\n border-bottom: 1px solid var(--wa-color-surface-border);\n vertical-align: top;\n }\n\n .snippets-table tbody tr:last-child td {\n border-bottom: none;\n }\n\n .snippets-table tbody tr:hover {\n background-color: var(--wa-color-neutral-fill-quiet);\n }\n\n .snippet-number-col {\n width: 3rem;\n text-align: center;\n }\n\n .snippet-content-col {\n width: auto;\n }\n\n .snippet-number {\n font-size: 0.75rem;\n font-weight: 600;\n color: var(--wa-color-text-quiet);\n text-align: center;\n }\n\n .snippet-content {\n white-space: pre-wrap;\n word-wrap: breaword;\n overflow-wrap: breaword;\n font-family: monospace;\n font-size: 0.875rem;\n line-height: 1.5;\n color: var(--wa-color-text-normal);\n }\n\n .snippet-preview {\n white-space: pre-wrap;\n word-wrap: breaword;\n overflow-wrap: breaword;\n font-family: monospace;\n font-size: 0.875rem;\n line-height: 1.5;\n color: var(--wa-color-text-normal);\n padding: var(--wa-space-s);\n background-color: var(--wa-color-surface-raised);\n border-radius: var(--wa-border-radius-medium);\n }\n\n .snippet-content mark.search-match {\n background: var(--wa-color-warning-fill-loud);\n color: var(--wa-color-warning-text-loud);\n padding: 0 2px;\n border-radius: 2px;\n font-weight: 600;\n }\n `;\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'lyra-rag-system-manager': LyraRAGSystemManager;\n }\n}\n\n","import { documentIndexService } from './document-index-service';\nimport { registerAll } from '@eclipse-lyra/core';\nimport { workspaceService, File } from '@eclipse-lyra/core';\nimport { toastError, toastInfo } from '@eclipse-lyra/core';\nimport { taskService } from '@eclipse-lyra/core';\nimport { activeSelectionSignal } from '@eclipse-lyra/core';\nimport { createLogger } from '@eclipse-lyra/core';\nimport { contributionRegistry } from '@eclipse-lyra/core';\nimport { editorRegistry, EditorInput } from '@eclipse-lyra/core';\nimport { html } from 'lit';\nimport { TOOLBAR_MAIN_RIGHT } from '@eclipse-lyra/core';\n// Import the component to ensure it's registered\nimport './rag-system-manager';\n\nconst logger = createLogger('RAGSystemExtension');\n\nexport default function ragSystemExtension(context: any) {\n documentIndexService.initialize().catch(err => {\n logger.error(`Failed to initialize document index service: ${err}`);\n });\n\n registerAll({\n command: {\n id: 'rag-system.index-file',\n name: 'Index Document',\n description: 'Index the currently selected file for search and retrieval',\n parameters: [\n {\n name: 'includeContent',\n description: 'Whether to include full content in index (default: true)',\n required: false\n }\n ]\n },\n handler: {\n canExecute: context => {\n const selection = activeSelectionSignal.get();\n return selection instanceof File;\n },\n execute: async context => {\n const selection = activeSelectionSignal.get();\n \n if (!(selection instanceof File)) {\n toastError('Please select a file to index');\n return;\n }\n\n const includeContent = context.params?.includeContent !== false;\n\n await taskService.runAsync('Indexing document', async progress => {\n progress.message = `Indexing ${selection.getName()}...`;\n \n try {\n const document = await documentIndexService.indexDocument(selection, {\n includeContent\n });\n \n progress.progress = 100;\n toastInfo(`Document indexed: ${document.fileName}`);\n } catch (error) {\n toastError(`Failed to index document: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.index-workspace',\n name: 'Index Workspace',\n description: 'Index all indexable files in the current workspace',\n parameters: [\n {\n name: 'includeContent',\n description: 'Whether to include full content in index (default: true)',\n required: false\n },\n {\n name: 'maxFileSize',\n description: 'Maximum file size in bytes to index (default: 5MB)',\n required: false\n }\n ]\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n const workspace = await workspaceService.getWorkspace();\n \n if (!workspace) {\n toastError('No workspace selected');\n return;\n }\n\n const includeContent = context.params?.includeContent !== false;\n const maxFileSize = context.params?.maxFileSize \n ? parseInt(context.params.maxFileSize) \n : undefined;\n\n await taskService.runAsync('Indexing workspace', async progress => {\n progress.message = 'Collecting files...';\n progress.progress = 0;\n\n try {\n const result = await documentIndexService.indexWorkspace(workspace, {\n includeContent,\n maxFileSize\n });\n\n progress.progress = 100;\n \n if (result.failed > 0) {\n toastError(\n `Indexing complete: ${result.indexed} indexed, ${result.failed} failed. ` +\n `Check console for details.`\n );\n } else {\n toastInfo(`Workspace indexed: ${result.indexed} documents`);\n }\n } catch (error) {\n toastError(`Failed to index workspace: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.list-documents',\n name: 'List Indexed Documents',\n description: 'List all indexed documents in the current workspace',\n parameters: []\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n const workspace = await workspaceService.getWorkspace();\n const workspacePath = workspace?.getName();\n\n await taskService.runAsync('Loading indexed documents', async progress => {\n try {\n const documents = await documentIndexService.listDocuments(workspacePath);\n \n progress.progress = 100;\n \n if (documents.length === 0) {\n toastInfo('No documents indexed in this workspace');\n } else {\n logger.info(`Found ${documents.length} indexed documents`);\n toastInfo(`Found ${documents.length} indexed documents (check console for details)`);\n }\n } catch (error) {\n toastError(`Failed to list documents: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.delete-document',\n name: 'Delete Document from Index',\n description: 'Remove the selected file from the document index',\n parameters: []\n },\n handler: {\n canExecute: context => {\n const selection = activeSelectionSignal.get();\n return selection instanceof File;\n },\n execute: async context => {\n const selection = activeSelectionSignal.get();\n \n if (!(selection instanceof File)) {\n toastError('Please select a file to remove from index');\n return;\n }\n\n const workspace = selection.getWorkspace();\n const workspacePath = workspace.getName();\n const filePath = selection.getWorkspacePath();\n\n await taskService.runAsync('Deleting document from index', async progress => {\n try {\n const deleted = await documentIndexService.deleteDocumentByPath(\n workspacePath,\n filePath\n );\n\n progress.progress = 100;\n\n if (deleted) {\n toastInfo(`Document removed from index: ${selection.getName()}`);\n } else {\n toastInfo(`Document not found in index: ${selection.getName()}`);\n }\n } catch (error) {\n toastError(`Failed to delete document from index: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.clear-workspace',\n name: 'Clear Workspace Index',\n description: 'Remove all indexed documents from the current workspace',\n parameters: []\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n const workspace = await workspaceService.getWorkspace();\n \n if (!workspace) {\n toastError('No workspace selected');\n return;\n }\n\n const workspacePath = workspace.getName();\n\n await taskService.runAsync('Clearing workspace index', async progress => {\n try {\n const deleted = await documentIndexService.deleteWorkspace(workspacePath);\n\n progress.progress = 100;\n toastInfo(`Removed ${deleted} documents from index`);\n } catch (error) {\n toastError(`Failed to clear workspace index: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.get-stats',\n name: 'Document Index Statistics',\n description: 'Get statistics about the document index',\n parameters: []\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n await taskService.runAsync('Loading statistics', async progress => {\n try {\n const stats = await documentIndexService.getStats();\n \n progress.progress = 100;\n \n logger.info(`Document index statistics: ${JSON.stringify(stats)}`);\n toastInfo(\n `Index statistics: ${stats.totalDocuments} total documents. ` +\n `Check console for details.`\n );\n } catch (error) {\n toastError(`Failed to get statistics: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.reindex-file',\n name: 'Reindex Document',\n description: 'Reindex the selected file (useful after file changes)',\n parameters: []\n },\n handler: {\n canExecute: context => {\n const selection = activeSelectionSignal.get();\n return selection instanceof File;\n },\n execute: async context => {\n const selection = activeSelectionSignal.get();\n \n if (!(selection instanceof File)) {\n toastError('Please select a file to reindex');\n return;\n }\n\n await taskService.runAsync('Reindexing document', async progress => {\n progress.message = `Reindexing ${selection.getName()}...`;\n \n try {\n const document = await documentIndexService.reindexDocument(selection);\n \n progress.progress = 100;\n toastInfo(`Document reindexed: ${document.fileName}`);\n } catch (error) {\n toastError(`Failed to reindex document: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n logger.info('RAG system extension loaded');\n\n // Register editor for document index manager\n editorRegistry.registerEditorInputHandler({\n editorId: 'system.rag-system-manager',\n label: 'RAG System Manager',\n ranking: 1000,\n canHandle: (input: EditorInput) => {\n return input.key === '.system.rag-system';\n },\n handle: async (input: EditorInput) => {\n input.widgetFactory = () => html`\n <lyra-rag-system-manager .input=${input}></lyra-rag-system-manager>\n `;\n return input;\n }\n });\n\n // Register command and toolbar button to open document index manager\n registerAll({\n command: {\n id: 'open-rag-system-manager',\n name: 'Open RAG System Manager',\n description: 'Opens the RAG system manager to view and manage indexed documents',\n parameters: []\n },\n handler: {\n execute: _context => {\n const editorInput: Partial<EditorInput> = {\n title: 'RAG System Manager',\n data: {},\n key: '.system.rag-system',\n icon: 'database',\n state: {},\n };\n editorRegistry.loadEditor(editorInput as EditorInput).catch(err => {\n logger.error(`Failed to open document index manager: ${err}`);\n });\n }\n },\n contribution: {\n target: TOOLBAR_MAIN_RIGHT,\n icon: 'database',\n label: 'RAG System',\n }\n });\n\n // Register context menu entry for indexing documents\n contributionRegistry.registerContribution('contextmenu:filebrowser', {\n command: 'rag-system.index-file',\n icon: 'database',\n label: 'Index Document',\n disabled: () => {\n const selection = activeSelectionSignal.get();\n return !(selection instanceof File);\n }\n });\n\n // Integrate RAG with AI system\n import('./rag-integration').then(rag => {\n rag.integrateRAGWithAI();\n logger.info('RAG integration enabled');\n }).catch(err => {\n logger.warn(`Failed to load RAG integration: ${err}`);\n });\n}\n\n"],"names":["logger","context"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"rag-system-extension-CE4P1Thg.js","sources":["../src/rag-system-manager.ts","../src/rag-system-extension.ts"],"sourcesContent":["import { customElement, property, state } from 'lit/decorators.js';\nimport { css, html, TemplateResult, nothing } from 'lit';\nimport { createRef, ref } from 'lit/directives/ref.js';\nimport { LyraPart } from '@eclipse-lyra/core';\nimport { EditorInput } from '@eclipse-lyra/core';\nimport { documentIndexService, IndexedDocument } from './document-index-service';\nimport { searchWorkspaceDocuments, RAGSearchResult } from './rag-service';\nimport { workspaceService, File } from '@eclipse-lyra/core';\nimport { createLogger } from '@eclipse-lyra/core';\nimport { getWorkspacePath } from './utils/workspace-utils';\nimport { SnippetExtractor } from './utils/snippet-extractor';\nimport { CONTENT_PREVIEW_LENGTHS, SNIPPET_LENGTHS } from './utils/constants';\nimport { toastError, toastInfo } from '@eclipse-lyra/core';\nimport { taskService } from '@eclipse-lyra/core';\nimport { editorRegistry } from '@eclipse-lyra/core';\n\nconst logger = createLogger('RAGSystemManager');\nconst snippetExtractor = new SnippetExtractor();\n\n@customElement('lyra-rag-system-manager')\nexport class LyraRAGSystemManager extends LyraPart {\n @property({ attribute: false })\n public input?: EditorInput;\n\n @state()\n private documents: IndexedDocument[] = [];\n\n\n @state()\n private stats: { totalDocuments: number; byWorkspace: Record<string, number> } = {\n totalDocuments: 0,\n byWorkspace: {}\n };\n\n @state()\n private loading = false;\n\n @state()\n private selectedDocument: IndexedDocument | null = null;\n\n @state()\n private searchQuery = '';\n\n @state()\n private filterWorkspace: string | null = null;\n\n @state()\n private filterByActiveWorkspace = true;\n\n @state()\n private filteredDocuments: IndexedDocument[] = [];\n \n @state()\n private searchResults: Map<string, RAGSearchResult> = new Map();\n\n @state()\n private reindexing = false;\n\n private treeRef = createRef<HTMLElement>();\n private searchDebounceTimer?: number;\n private searchInputValue = '';\n\n protected async doInitUI() {\n try {\n await documentIndexService.initialize();\n await Promise.all([\n this.loadDocuments(),\n this.loadStats()\n ]);\n } catch (error) {\n logger.error(`Failed to initialize document index manager: ${error}`);\n toastError(`Failed to initialize: ${error}`);\n }\n }\n\n private async loadDocuments() {\n this.loading = true;\n this.requestUpdate();\n try {\n let workspacePath: string | undefined;\n if (this.filterByActiveWorkspace) {\n const workspaceResult = await getWorkspacePath();\n workspacePath = workspaceResult?.workspacePath;\n }\n this.documents = await documentIndexService.listDocuments(workspacePath);\n await this.updateFilteredDocuments();\n } catch (error) {\n logger.error(`Failed to load documents: ${error}`);\n toastError(`Failed to load documents: ${error}`);\n } finally {\n this.loading = false;\n }\n }\n\n private async updateFilteredDocuments() {\n this.filteredDocuments = await this.getFilteredDocuments();\n this.requestUpdate();\n }\n\n private async loadStats() {\n try {\n this.stats = await documentIndexService.getStats();\n this.requestUpdate();\n } catch (error) {\n logger.error(`Failed to load stats: ${error}`);\n }\n }\n\n private handleTreeSelection(e: CustomEvent) {\n // Try getting from event detail first (as per extensions pattern)\n let selection = e.detail?.selection || [];\n\n // If no selection in detail, try getting from tree element directly (as per lyra-filebrowser pattern)\n if (selection.length === 0 && this.treeRef.value) {\n // @ts-ignore\n selection = this.treeRef.value.selectedItems || [];\n }\n\n if (selection.length > 0) {\n const selectedItem = selection[0];\n if (selectedItem?.model) {\n this.selectedDocument = selectedItem.model;\n } else {\n this.selectedDocument = null;\n }\n } else {\n this.selectedDocument = null;\n }\n }\n\n private async getFilteredDocuments(): Promise<IndexedDocument[]> {\n if (!this.documents || this.documents.length === 0) {\n return [];\n }\n\n if (this.searchQuery.trim()) {\n try {\n const searchResults = await searchWorkspaceDocuments(this.searchQuery, {\n limit: 50,\n workspacePath: this.filterWorkspace || undefined\n });\n\n this.searchResults.clear();\n const uniqueDocs = new Map<string, IndexedDocument>();\n const aggregatedResults = new Map<string, RAGSearchResult>();\n \n for (const result of searchResults) {\n const docId = result.document.id;\n uniqueDocs.set(docId, result.document);\n \n const existing = aggregatedResults.get(docId);\n if (!existing) {\n aggregatedResults.set(docId, {\n document: result.document,\n relevance: result.relevance,\n matchedSnippets: [...result.matchedSnippets]\n });\n } else {\n if (result.relevance > existing.relevance) {\n existing.relevance = result.relevance;\n }\n const seenSnippets = new Set(existing.matchedSnippets);\n for (const snippet of result.matchedSnippets) {\n if (!seenSnippets.has(snippet)) {\n existing.matchedSnippets.push(snippet);\n seenSnippets.add(snippet);\n }\n }\n }\n }\n \n for (const [docId, result] of aggregatedResults) {\n this.searchResults.set(docId, result);\n }\n\n const resultDocs = Array.from(uniqueDocs.values());\n \n if (this.filterWorkspace) {\n return resultDocs.filter(doc => doc.workspacePath === this.filterWorkspace);\n }\n \n return resultDocs;\n } catch (error) {\n logger.debug(`RAG search failed in document manager: ${error}`);\n this.searchResults.clear();\n return [];\n }\n } else {\n this.searchResults.clear();\n }\n\n let filtered = [...this.documents];\n\n if (this.filterWorkspace) {\n filtered = filtered.filter(doc => doc.workspacePath === this.filterWorkspace);\n }\n\n return filtered;\n }\n\n private highlightMatches(text: string, query: string): TemplateResult {\n if (!query || !query.trim()) {\n return html`${text}`;\n }\n\n const queryLower = query.toLowerCase();\n const textLower = text.toLowerCase();\n const parts: Array<string | TemplateResult> = [];\n let lastIndex = 0;\n let index = textLower.indexOf(queryLower, lastIndex);\n\n while (index !== -1) {\n if (index > lastIndex) {\n parts.push(text.substring(lastIndex, index));\n }\n parts.push(html`<mark class=\"search-match\">${text.substring(index, index + query.length)}</mark>`);\n lastIndex = index + query.length;\n index = textLower.indexOf(queryLower, lastIndex);\n }\n\n if (lastIndex < text.length) {\n parts.push(text.substring(lastIndex));\n }\n\n return html`${parts}`;\n }\n\n private getContentPreview(doc: IndexedDocument): TemplateResult {\n const searchResult = this.searchResults.get(doc.id);\n \n if (searchResult && searchResult.matchedSnippets.length > 0) {\n return html`\n <table class=\"snippets-table\">\n <thead>\n <tr>\n <th class=\"snippet-number-col\">#</th>\n <th class=\"snippet-content-col\">Content</th>\n </tr>\n </thead>\n <tbody>\n ${searchResult.matchedSnippets.map((snippet, idx) => html`\n <tr>\n <td class=\"snippet-number\">${idx + 1}</td>\n <td class=\"snippet-content\">${this.highlightMatches(snippet, this.searchQuery)}</td>\n </tr>\n `)}\n </tbody>\n </table>\n `;\n }\n\n if (this.searchQuery && this.searchQuery.trim()) {\n const snippets = snippetExtractor.extractContextSnippets(\n doc.content,\n this.searchQuery,\n SNIPPET_LENGTHS.CONTEXT\n );\n\n if (snippets.length > 0) {\n return html`\n <table class=\"snippets-table\">\n <thead>\n <tr>\n <th class=\"snippet-number-col\">#</th>\n <th class=\"snippet-content-col\">Content</th>\n </tr>\n </thead>\n <tbody>\n ${snippets.map((snippet, idx) => html`\n <tr>\n <td class=\"snippet-number\">${idx + 1}</td>\n <td class=\"snippet-content\">${this.highlightMatches(doc.content.substring(snippet.start, snippet.end), this.searchQuery)}</td>\n </tr>\n `)}\n </tbody>\n </table>\n `;\n }\n }\n\n const preview = snippetExtractor.extractSimpleSnippet(doc.content, CONTENT_PREVIEW_LENGTHS.LONG);\n return html`\n <div class=\"snippet-preview\">${preview}</div>\n `;\n }\n\n private async deleteDocument(doc: IndexedDocument) {\n try {\n await documentIndexService.deleteDocument(doc.id);\n toastInfo(`Deleted: ${doc.fileName}`);\n await this.loadDocuments();\n await this.loadStats();\n if (this.selectedDocument?.id === doc.id) {\n this.selectedDocument = null;\n }\n } catch (error) {\n toastError(`Failed to delete document: ${error}`);\n }\n }\n\n private async reindexDocument(doc: IndexedDocument) {\n try {\n const workspaceResult = await getWorkspacePath();\n if (!workspaceResult) {\n toastError('No workspace connected');\n return;\n }\n\n const resource = await workspaceResult.workspace.getResource(doc.filePath);\n if (!resource) {\n toastError(`File not found: ${doc.filePath}`);\n return;\n }\n\n if (!(resource instanceof File)) {\n toastError(`Resource is not a file: ${doc.filePath}`);\n return;\n }\n\n const file: File = resource;\n\n await taskService.runAsync('Reindexing document', async progress => {\n progress.message = `Reindexing ${doc.fileName}...`;\n await documentIndexService.reindexDocument(file);\n progress.progress = 100;\n });\n\n toastInfo(`Reindexed: ${doc.fileName}`);\n await this.loadDocuments();\n if (this.selectedDocument?.id === doc.id) {\n this.selectedDocument = await documentIndexService.getDocument(doc.id);\n }\n } catch (error) {\n toastError(`Failed to reindex document: ${error}`);\n }\n }\n\n private async reindexAllDocuments() {\n if (this.reindexing) {\n return;\n }\n\n const stats = await documentIndexService.getStats();\n if (stats.totalDocuments === 0) {\n toastInfo('No documents to reindex');\n return;\n }\n\n this.reindexing = true;\n this.requestUpdate();\n\n try {\n const result = await taskService.runAsync('Reindexing all documents', async progress => {\n progress.message = 'Starting reindexing...';\n \n const total = stats.totalDocuments;\n\n const reindexResult = await documentIndexService.reindexAllDocuments();\n \n const processed = reindexResult.succeeded + reindexResult.failed;\n progress.progress = total > 0 ? (processed / total) * 100 : 100;\n progress.message = `Reindexed ${reindexResult.succeeded}/${total} documents${reindexResult.failed > 0 ? ` (${reindexResult.failed} failed)` : ''}`;\n \n return reindexResult;\n });\n\n await this.loadDocuments();\n await this.loadStats();\n toastInfo(`Reindexing completed: ${result.succeeded} succeeded, ${result.failed} failed`);\n } catch (error) {\n logger.error(`Failed to reindex all documents: ${error}`);\n toastError(`Failed to reindex all documents: ${error}`);\n } finally {\n this.reindexing = false;\n this.requestUpdate();\n }\n }\n\n private formatFileSize(bytes: number): string {\n if (bytes < 1024) return `${bytes} B`;\n if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} LyraB`;\n return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;\n }\n\n private formatDate(timestamp: number): string {\n return new Date(timestamp).toLocaleString();\n }\n\n private getFileIcon(fileType: string): string {\n return editorRegistry.getFileIcon(fileType);\n }\n\n protected renderToolbar(): TemplateResult {\n const workspaces = Object.keys(this.stats?.byWorkspace || {});\n\n return html`\n <wa-input\n type=\"search\"\n placeholder=\"Search documents...\"\n .value=${this.searchInputValue}\n @input=${(e: any) => {\n this.searchInputValue = e.target.value;\n \n if (this.searchDebounceTimer) {\n clearTimeout(this.searchDebounceTimer);\n }\n \n this.searchDebounceTimer = window.setTimeout(async () => {\n this.searchQuery = this.searchInputValue;\n await this.updateFilteredDocuments();\n }, 200);\n }}\n @wa-clear=${async () => {\n if (this.searchDebounceTimer) {\n clearTimeout(this.searchDebounceTimer);\n }\n this.searchInputValue = '';\n this.searchQuery = '';\n await this.updateFilteredDocuments();\n }}\n size=\"small\"\n with-clear\n autocomplete=\"off\"\n style=\"flex: 1; max-width: 400px;\">\n <wa-icon name=\"magnifying-glass\" slot=\"start\"></wa-icon>\n </wa-input>\n \n <wa-switch\n .checked=${this.filterByActiveWorkspace}\n @change=${async (e: any) => {\n this.filterByActiveWorkspace = e.target.checked;\n await this.loadDocuments();\n }}\n size=\"small\">\n Active workspace only\n </wa-switch>\n \n ${workspaces.length > 1 ? html`\n <wa-select \n .value=${this.filterWorkspace || ''}\n @change=${async (e: any) => {\n this.filterWorkspace = e.target.value || null;\n await this.updateFilteredDocuments();\n }}\n size=\"small\"\n style=\"width: 200px;\">\n <wa-option value=\"\">All Workspaces</wa-option>\n ${workspaces.map(ws => html`\n <wa-option value=\"${ws}\">${ws} (${this.stats.byWorkspace[ws]})</wa-option>\n `)}\n </wa-select>\n ` : nothing}\n \n <lyra-command \n size=\"small\" \n icon=\"arrow-rotate-right\"\n title=\"Refresh document list\"\n .action=${() => this.loadDocuments()}\n ?disabled=${this.reindexing}>\n Refresh\n </lyra-command>\n \n <lyra-command \n size=\"small\" \n icon=\"database\"\n title=\"Re-index all documents\"\n .action=${() => this.reindexAllDocuments()}\n ?disabled=${this.reindexing || this.loading}>\n ${this.reindexing ? 'Reindexing...' : 'Re-index All'}\n </lyra-command>\n `;\n }\n\n protected render(): TemplateResult {\n if (!this.stats) {\n this.stats = { totalDocuments: 0, byWorkspace: {} };\n }\n\n const filteredDocs = this.filteredDocuments;\n const workspaces = Object.keys(this.stats.byWorkspace || {});\n\n return html`\n <div class=\"rag-system-manager\">\n <div class=\"header\">\n <div class=\"header-content\">\n <div class=\"stats\">\n <span>Total: ${this.stats.totalDocuments} documents</span>\n ${workspaces.length > 0 ? html`\n <span>Workspaces: ${workspaces.length}</span>\n ` : nothing}\n </div>\n </div>\n </div>\n\n <wa-split-panel position=\"40\" style=\"height: 100%;\">\n <div class=\"document-list\" slot=\"start\">\n ${this.loading ? html`\n <div class=\"loading\">\n <wa-spinner></wa-spinner>\n <span>Loading documents...</span>\n </div>\n ` : filteredDocs.length === 0 ? html`\n <div class=\"empty\">\n <wa-icon name=\"inbox\" style=\"font-size: 3rem; opacity: 0.3;\"></wa-icon>\n <p>${this.searchQuery || this.filterWorkspace ? 'No documents match your filters' : 'No documents indexed yet'}</p>\n </div>\n ` : html`\n <wa-tree \n ${ref(this.treeRef)}\n selection=\"leaf\"\n style=\"--indent-guide-width: 1px;\"\n @wa-selection-change=${(e: CustomEvent) => {\n this.handleTreeSelection(e);\n }}>\n ${filteredDocs.map(doc => html`\n <wa-tree-item \n .model=${doc}\n ?selected=${this.selectedDocument?.id === doc.id}>\n <wa-icon name=\"${this.getFileIcon(doc.fileType)}\"></wa-icon>\n <div class=\"tree-item-info\">\n <strong class=\"tree-item-path\">${doc.filePath}</strong>\n <div class=\"tree-item-meta\">\n <small class=\"meta-size\">${this.formatFileSize(doc.metadata.size)}</small>\n <small class=\"meta-date\">${this.formatDate(doc.indexedAt)}</small>\n </div>\n </div>\n <div class=\"tree-item-actions\" @click=${(e: Event) => e.stopPropagation()}>\n <wa-button\n variant=\"neutral\"\n appearance=\"plain\"\n size=\"small\"\n title=\"Reindex\"\n @click=${() => this.reindexDocument(doc)}>\n <wa-icon name=\"arrow-rotate-right\"></wa-icon>\n </wa-button>\n <wa-button\n variant=\"danger\"\n appearance=\"plain\"\n size=\"small\"\n title=\"Delete\"\n @click=${() => this.deleteDocument(doc)}>\n <wa-icon name=\"trash\"></wa-icon>\n </wa-button>\n </div>\n </wa-tree-item>\n `)}\n </wa-tree>\n `}\n </div>\n\n <div slot=\"end\">\n ${this.selectedDocument ? html`\n <div class=\"document-details\">\n <div class=\"details-content\">\n <div class=\"metadata-grid\">\n <wa-input\n label=\"File Path\"\n .value=${this.selectedDocument.filePath}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.filePath}\n size=\"small\"\n label=\"Copy file path\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Workspace\"\n .value=${this.selectedDocument.workspacePath}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.workspacePath}\n size=\"small\"\n label=\"Copy workspace\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"File Type\"\n .value=${this.selectedDocument.fileType}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.fileType}\n size=\"small\"\n label=\"Copy file type\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Size\"\n .value=${this.formatFileSize(this.selectedDocument.metadata.size)}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.formatFileSize(this.selectedDocument.metadata.size)}\n size=\"small\"\n label=\"Copy size\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Indexed At\"\n .value=${this.formatDate(this.selectedDocument.indexedAt)}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.formatDate(this.selectedDocument.indexedAt)}\n size=\"small\"\n label=\"Copy indexed date\">\n </wa-copy-button>\n </wa-input>\n \n <wa-input\n label=\"Last Updated\"\n .value=${this.formatDate(this.selectedDocument.updatedAt)}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.formatDate(this.selectedDocument.updatedAt)}\n size=\"small\"\n label=\"Copy updated date\">\n </wa-copy-button>\n </wa-input>\n </div>\n \n ${this.selectedDocument.metadata.tags && this.selectedDocument.metadata.tags.length > 0 ? html`\n <div class=\"tags-section\">\n <wa-input\n label=\"Tags\"\n .value=${this.selectedDocument.metadata.tags.join(', ')}\n readonly\n size=\"small\">\n <wa-copy-button\n slot=\"end\"\n .value=${this.selectedDocument.metadata.tags.join(', ')}\n size=\"small\"\n label=\"Copy tags\">\n </wa-copy-button>\n </wa-input>\n </div>\n ` : nothing}\n \n <div class=\"detail-section\">\n <label>Content Preview${this.searchQuery ? html` <span class=\"search-hint\">(showing matches for \"${this.searchQuery}\")</span>` : nothing}</label>\n <wa-scroller class=\"content-preview\" orientation=\"vertical\">\n <div class=\"content-preview-inner\">\n ${this.getContentPreview(this.selectedDocument)}\n </div>\n </wa-scroller>\n </div>\n </div>\n </div>\n ` : html`\n <div class=\"document-details empty\">\n <wa-icon name=\"file-lines\" style=\"font-size: 3rem; opacity: 0.3;\"></wa-icon>\n <p>Select a document to view details</p>\n </div>\n `}\n </div>\n </wa-split-panel>\n </div>\n `;\n }\n\n static styles = css`\n :host {\n display: flex;\n flex-direction: column;\n height: 100%;\n overflow: hidden;\n min-height: 0;\n }\n\n .rag-system-manager {\n display: flex;\n flex-direction: column;\n height: 100%;\n min-height: 0;\n overflow: hidden;\n }\n\n wa-split-panel {\n flex: 1;\n min-height: 0;\n overflow: hidden;\n }\n\n .document-list {\n height: 100%;\n overflow-y: auto;\n }\n\n .tree-item-info {\n flex: 1;\n min-width: 0;\n }\n\n .tree-item-path {\n overflow: hidden;\n text-overflow: ellipsis;\n white-space: nowrap;\n }\n\n .tree-item-meta {\n display: flex;\n align-items: center;\n gap: var(--wa-space-xs);\n flex-wrap: wrap;\n margin-top: var(--wa-space-xs);\n }\n\n .tree-item-actions {\n opacity: 0;\n }\n\n wa-tree-item:hover .tree-item-actions {\n opacity: 1;\n }\n\n .document-details {\n height: 100%;\n display: flex;\n flex-direction: column;\n min-height: 0;\n overflow: hidden;\n }\n\n .details-content {\n flex: 1;\n display: flex;\n flex-direction: column;\n padding: var(--wa-space-s);\n min-height: 0;\n overflow: hidden;\n }\n\n .metadata-grid {\n display: grid;\n grid-template-columns: 1fr 1fr;\n gap: var(--wa-space-s);\n flex-shrink: 0;\n }\n\n .tags-section {\n margin-top: var(--wa-space-s);\n flex-shrink: 0;\n margin-bottom: var(--wa-space-s);\n }\n\n .detail-section {\n flex: 1;\n display: flex;\n flex-direction: column;\n min-height: 0;\n position: relative;\n }\n\n .detail-section label {\n flex-shrink: 0;\n margin-bottom: var(--wa-space-xs);\n }\n\n .content-preview {\n position: absolute;\n top: 0;\n right: 0;\n left: 0;\n bottom: 0;\n }\n\n .content-preview-inner {\n width: 100%;\n }\n\n .snippets-table {\n width: 100%;\n border-collapse: collapse;\n background-color: var(--wa-color-surface-raised);\n }\n\n .snippets-table thead {\n background-color: var(--wa-color-neutral-fill-quiet);\n }\n\n .snippets-table th {\n padding: var(--wa-space-xs) var(--wa-space-s);\n text-align: left;\n font-size: 0.75rem;\n font-weight: 600;\n color: var(--wa-color-text-quiet);\n border-bottom: 1px solid var(--wa-color-surface-border);\n }\n\n .snippets-table td {\n padding: var(--wa-space-s);\n border-bottom: 1px solid var(--wa-color-surface-border);\n vertical-align: top;\n }\n\n .snippets-table tbody tr:last-child td {\n border-bottom: none;\n }\n\n .snippets-table tbody tr:hover {\n background-color: var(--wa-color-neutral-fill-quiet);\n }\n\n .snippet-number-col {\n width: 3rem;\n text-align: center;\n }\n\n .snippet-content-col {\n width: auto;\n }\n\n .snippet-number {\n font-size: 0.75rem;\n font-weight: 600;\n color: var(--wa-color-text-quiet);\n text-align: center;\n }\n\n .snippet-content {\n white-space: pre-wrap;\n word-wrap: breaword;\n overflow-wrap: breaword;\n font-family: monospace;\n font-size: 0.875rem;\n line-height: 1.5;\n color: var(--wa-color-text-normal);\n }\n\n .snippet-preview {\n white-space: pre-wrap;\n word-wrap: breaword;\n overflow-wrap: breaword;\n font-family: monospace;\n font-size: 0.875rem;\n line-height: 1.5;\n color: var(--wa-color-text-normal);\n padding: var(--wa-space-s);\n background-color: var(--wa-color-surface-raised);\n border-radius: var(--wa-border-radius-medium);\n }\n\n .snippet-content mark.search-match {\n background: var(--wa-color-warning-fill-loud);\n color: var(--wa-color-warning-text-loud);\n padding: 0 2px;\n border-radius: 2px;\n font-weight: 600;\n }\n `;\n}\n\ndeclare global {\n interface HTMLElementTagNameMap {\n 'lyra-rag-system-manager': LyraRAGSystemManager;\n }\n}\n\n","import { documentIndexService } from './document-index-service';\nimport { registerAll } from '@eclipse-lyra/core';\nimport { workspaceService, File } from '@eclipse-lyra/core';\nimport { toastError, toastInfo } from '@eclipse-lyra/core';\nimport { taskService } from '@eclipse-lyra/core';\nimport { activeSelectionSignal } from '@eclipse-lyra/core';\nimport { createLogger } from '@eclipse-lyra/core';\nimport { contributionRegistry } from '@eclipse-lyra/core';\nimport { editorRegistry, EditorInput } from '@eclipse-lyra/core';\nimport { html } from 'lit';\nimport { TOOLBAR_MAIN_RIGHT } from '@eclipse-lyra/core';\n// Import the component to ensure it's registered\nimport './rag-system-manager';\n\nconst logger = createLogger('RAGSystemExtension');\n\nexport default function ragSystemExtension(context: any) {\n documentIndexService.initialize().catch(err => {\n logger.error(`Failed to initialize document index service: ${err}`);\n });\n\n registerAll({\n command: {\n id: 'rag-system.index-file',\n name: 'Index Document',\n description: 'Index the currently selected file for search and retrieval',\n parameters: [\n {\n name: 'includeContent',\n description: 'Whether to include full content in index (default: true)',\n required: false\n }\n ]\n },\n handler: {\n canExecute: context => {\n const selection = activeSelectionSignal.get();\n return selection instanceof File;\n },\n execute: async context => {\n const selection = activeSelectionSignal.get();\n \n if (!(selection instanceof File)) {\n toastError('Please select a file to index');\n return;\n }\n\n const includeContent = context.params?.includeContent !== false;\n\n await taskService.runAsync('Indexing document', async progress => {\n progress.message = `Indexing ${selection.getName()}...`;\n \n try {\n const document = await documentIndexService.indexDocument(selection, {\n includeContent\n });\n \n progress.progress = 100;\n toastInfo(`Document indexed: ${document.fileName}`);\n } catch (error) {\n toastError(`Failed to index document: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.index-workspace',\n name: 'Index Workspace',\n description: 'Index all indexable files in the current workspace',\n parameters: [\n {\n name: 'includeContent',\n description: 'Whether to include full content in index (default: true)',\n required: false\n },\n {\n name: 'maxFileSize',\n description: 'Maximum file size in bytes to index (default: 5MB)',\n required: false\n }\n ]\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n const workspace = await workspaceService.getWorkspace();\n \n if (!workspace) {\n toastError('No workspace selected');\n return;\n }\n\n const includeContent = context.params?.includeContent !== false;\n const maxFileSize = context.params?.maxFileSize \n ? parseInt(context.params.maxFileSize) \n : undefined;\n\n await taskService.runAsync('Indexing workspace', async progress => {\n progress.message = 'Collecting files...';\n progress.progress = 0;\n\n try {\n const result = await documentIndexService.indexWorkspace(workspace, {\n includeContent,\n maxFileSize\n });\n\n progress.progress = 100;\n \n if (result.failed > 0) {\n toastError(\n `Indexing complete: ${result.indexed} indexed, ${result.failed} failed. ` +\n `Check console for details.`\n );\n } else {\n toastInfo(`Workspace indexed: ${result.indexed} documents`);\n }\n } catch (error) {\n toastError(`Failed to index workspace: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.list-documents',\n name: 'List Indexed Documents',\n description: 'List all indexed documents in the current workspace',\n parameters: []\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n const workspace = await workspaceService.getWorkspace();\n const workspacePath = workspace?.getName();\n\n await taskService.runAsync('Loading indexed documents', async progress => {\n try {\n const documents = await documentIndexService.listDocuments(workspacePath);\n \n progress.progress = 100;\n \n if (documents.length === 0) {\n toastInfo('No documents indexed in this workspace');\n } else {\n logger.info(`Found ${documents.length} indexed documents`);\n toastInfo(`Found ${documents.length} indexed documents (check console for details)`);\n }\n } catch (error) {\n toastError(`Failed to list documents: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.delete-document',\n name: 'Delete Document from Index',\n description: 'Remove the selected file from the document index',\n parameters: []\n },\n handler: {\n canExecute: context => {\n const selection = activeSelectionSignal.get();\n return selection instanceof File;\n },\n execute: async context => {\n const selection = activeSelectionSignal.get();\n \n if (!(selection instanceof File)) {\n toastError('Please select a file to remove from index');\n return;\n }\n\n const workspace = selection.getWorkspace();\n const workspacePath = workspace.getName();\n const filePath = selection.getWorkspacePath();\n\n await taskService.runAsync('Deleting document from index', async progress => {\n try {\n const deleted = await documentIndexService.deleteDocumentByPath(\n workspacePath,\n filePath\n );\n\n progress.progress = 100;\n\n if (deleted) {\n toastInfo(`Document removed from index: ${selection.getName()}`);\n } else {\n toastInfo(`Document not found in index: ${selection.getName()}`);\n }\n } catch (error) {\n toastError(`Failed to delete document from index: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.clear-workspace',\n name: 'Clear Workspace Index',\n description: 'Remove all indexed documents from the current workspace',\n parameters: []\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n const workspace = await workspaceService.getWorkspace();\n \n if (!workspace) {\n toastError('No workspace selected');\n return;\n }\n\n const workspacePath = workspace.getName();\n\n await taskService.runAsync('Clearing workspace index', async progress => {\n try {\n const deleted = await documentIndexService.deleteWorkspace(workspacePath);\n\n progress.progress = 100;\n toastInfo(`Removed ${deleted} documents from index`);\n } catch (error) {\n toastError(`Failed to clear workspace index: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.get-stats',\n name: 'Document Index Statistics',\n description: 'Get statistics about the document index',\n parameters: []\n },\n handler: {\n canExecute: context => {\n return true;\n },\n execute: async context => {\n await taskService.runAsync('Loading statistics', async progress => {\n try {\n const stats = await documentIndexService.getStats();\n \n progress.progress = 100;\n \n logger.info(`Document index statistics: ${JSON.stringify(stats)}`);\n toastInfo(\n `Index statistics: ${stats.totalDocuments} total documents. ` +\n `Check console for details.`\n );\n } catch (error) {\n toastError(`Failed to get statistics: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n registerAll({\n command: {\n id: 'rag-system.reindex-file',\n name: 'Reindex Document',\n description: 'Reindex the selected file (useful after file changes)',\n parameters: []\n },\n handler: {\n canExecute: context => {\n const selection = activeSelectionSignal.get();\n return selection instanceof File;\n },\n execute: async context => {\n const selection = activeSelectionSignal.get();\n \n if (!(selection instanceof File)) {\n toastError('Please select a file to reindex');\n return;\n }\n\n await taskService.runAsync('Reindexing document', async progress => {\n progress.message = `Reindexing ${selection.getName()}...`;\n \n try {\n const document = await documentIndexService.reindexDocument(selection);\n \n progress.progress = 100;\n toastInfo(`Document reindexed: ${document.fileName}`);\n } catch (error) {\n toastError(`Failed to reindex document: ${error}`);\n throw error;\n }\n });\n }\n }\n });\n\n logger.info('RAG system extension loaded');\n\n // Register editor for document index manager\n editorRegistry.registerEditorInputHandler({\n editorId: 'system.rag-system-manager',\n label: 'RAG System Manager',\n ranking: 1000,\n canHandle: (input: EditorInput) => {\n return input.key === '.system.rag-system';\n },\n handle: async (input: EditorInput) => {\n input.component = () => html`\n <lyra-rag-system-manager .input=${input}></lyra-rag-system-manager>\n `;\n return input;\n }\n });\n\n // Register command and toolbar button to open document index manager\n registerAll({\n command: {\n id: 'open-rag-system-manager',\n name: 'Open RAG System Manager',\n description: 'Opens the RAG system manager to view and manage indexed documents',\n parameters: []\n },\n handler: {\n execute: _context => {\n const editorInput: Partial<EditorInput> = {\n title: 'RAG System Manager',\n data: {},\n key: '.system.rag-system',\n icon: 'database',\n state: {},\n };\n editorRegistry.loadEditor(editorInput as EditorInput).catch(err => {\n logger.error(`Failed to open document index manager: ${err}`);\n });\n }\n },\n contribution: {\n target: TOOLBAR_MAIN_RIGHT,\n icon: 'database',\n label: 'RAG System',\n }\n });\n\n // Register context menu entry for indexing documents\n contributionRegistry.registerContribution('contextmenu:filebrowser', {\n command: 'rag-system.index-file',\n icon: 'database',\n label: 'Index Document',\n disabled: () => {\n const selection = activeSelectionSignal.get();\n return !(selection instanceof File);\n }\n });\n\n // Integrate RAG with AI system\n import('./rag-integration').then(rag => {\n rag.integrateRAGWithAI();\n logger.info('RAG integration enabled');\n }).catch(err => {\n logger.warn(`Failed to load RAG integration: ${err}`);\n });\n}\n\n"],"names":["logger","context"],"mappings":"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"}
|