diff --git a/vscode/src/chat/chat-view/ChatController.ts b/vscode/src/chat/chat-view/ChatController.ts index de853295ecbd..6931f677834d 100644 --- a/vscode/src/chat/chat-view/ChatController.ts +++ b/vscode/src/chat/chat-view/ChatController.ts @@ -361,6 +361,9 @@ export class ChatController implements vscode.Disposable, vscode.WebviewViewProv viewColumn: vscode.ViewColumn.Beside, }) break + case 'openRemoteFile': + this.openRemoteFile(message.uri) + break case 'newFile': await handleCodeFromSaveToNewFile(message.text, this.editor) break @@ -910,6 +913,16 @@ export class ChatController implements vscode.Disposable, vscode.WebviewViewProv return } + private async openRemoteFile(uri: vscode.Uri) { + const json = uri.toJSON() + json.scheme = 'sourcegraph' + const sourcegraphSchemaURI = vscode.Uri.from(json) + + vscode.workspace + .openTextDocument(sourcegraphSchemaURI) + .then(doc => vscode.window.showTextDocument(doc)) + } + private submitOrEditOperation: AbortController | undefined public startNewSubmitOrEditOperation(): AbortSignal { this.submitOrEditOperation?.abort() diff --git a/vscode/src/chat/chat-view/sourcegraphRemoteFile.ts b/vscode/src/chat/chat-view/sourcegraphRemoteFile.ts new file mode 100644 index 000000000000..375dbe207184 --- /dev/null +++ b/vscode/src/chat/chat-view/sourcegraphRemoteFile.ts @@ -0,0 +1,57 @@ +import { graphqlClient, isError } from '@sourcegraph/cody-shared' +import { LRUCache } from 'lru-cache' +import * as vscode from 'vscode' + +export class SourcegraphRemoteFileProvider + implements vscode.TextDocumentContentProvider, vscode.Disposable +{ + private cache = new LRUCache({ max: 128 }) + private disposables: vscode.Disposable[] = [] + + constructor() { + this.disposables.push(vscode.workspace.registerTextDocumentContentProvider('sourcegraph', this)) + } + + async provideTextDocumentContent(uri: vscode.Uri): Promise { + const content = + this.cache.get(uri.toString()) || + (await SourcegraphRemoteFileProvider.getFileContentsFromURL(uri)) + + this.cache.set(uri.toString(), content) + + return content + } + + public dispose(): void { + this.cache.clear() + for (const disposable of this.disposables) { + disposable.dispose() + } + this.disposables = [] + } + + private static async getFileContentsFromURL(URL: vscode.Uri): Promise { + const path = URL.path + const [repoRev = '', filePath] = path.split('/-/blob/') + let [repoName, rev = 'HEAD'] = repoRev.split('@') + repoName = repoName.replace(/^\/+/, '') + + if (!repoName || !filePath) { + throw new Error('Invalid URI') + } + + const dataOrError = await graphqlClient.getFileContents(repoName, filePath, rev) + + if (isError(dataOrError)) { + throw new Error(dataOrError.message) + } + + const content = dataOrError.repository?.commit?.file?.content + + if (!content) { + throw new Error('File not found') + } + + return content + } +} diff --git a/vscode/src/chat/protocol.ts b/vscode/src/chat/protocol.ts index f7004c609cb2..57274b8302b5 100644 --- a/vscode/src/chat/protocol.ts +++ b/vscode/src/chat/protocol.ts @@ -74,6 +74,7 @@ export type WebviewMessage = | { command: 'restoreHistory'; chatID: string } | { command: 'links'; value: string } | { command: 'openURI'; uri: Uri } + | { command: 'openRemoteFile'; uri: Uri } | { command: 'openFileLink' uri: Uri diff --git a/vscode/src/main.ts b/vscode/src/main.ts index 5ba73e55b4e5..b694c3882758 100644 --- a/vscode/src/main.ts +++ b/vscode/src/main.ts @@ -51,6 +51,7 @@ import { registerAutoEditTestRenderCommand } from './autoedits/renderer/mock-ren import type { MessageProviderOptions } from './chat/MessageProvider' import { ChatsController, CodyChatEditorViewType } from './chat/chat-view/ChatsController' import { ContextRetriever } from './chat/chat-view/ContextRetriever' +import { SourcegraphRemoteFileProvider } from './chat/chat-view/sourcegraphRemoteFile' import type { ChatIntentAPIClient } from './chat/context/chatIntentAPIClient' import { ACCOUNT_LIMITS_INFO_URL, @@ -905,7 +906,15 @@ function registerChat( extensionClient: platform.extensionClient, }) const promptsManager = new PromptsManager({ chatsController }) - disposables.push(ghostHintDecorator, editorManager, new CodeActionProvider(), promptsManager) + const sourcegraphRemoteFileProvider = new SourcegraphRemoteFileProvider() + + disposables.push( + ghostHintDecorator, + editorManager, + new CodeActionProvider(), + promptsManager, + sourcegraphRemoteFileProvider + ) // Register a serializer for reviving the chat panel on reload if (vscode.window.registerWebviewPanelSerializer) { diff --git a/vscode/webviews/components/codeSnippet/CodeSnippet.tsx b/vscode/webviews/components/codeSnippet/CodeSnippet.tsx index 9cd8b0674ba7..4eca110fae87 100644 --- a/vscode/webviews/components/codeSnippet/CodeSnippet.tsx +++ b/vscode/webviews/components/codeSnippet/CodeSnippet.tsx @@ -24,12 +24,16 @@ import { pluralize, } from './utils' +import { CodyIDE } from '@sourcegraph/cody-shared' import type { NLSSearchFileMatch, NLSSearchResult, } from '@sourcegraph/cody-shared/src/sourcegraph-api/graphql/client' import type { Observable } from 'observable-fns' import { useInView } from 'react-intersection-observer' +import { URI } from 'vscode-uri' +import { getVSCodeAPI } from '../../utils/VSCodeApi' +import { useConfig } from '../../utils/useConfig' import styles from './CodeSnippet.module.css' const DEFAULT_VISIBILITY_OFFSET = '500px' @@ -123,6 +127,20 @@ export const FileMatchSearchResult: FC collapsedHighlightCount useEffect(() => setExpanded(allExpanded || defaultExpanded), [allExpanded, defaultExpanded]) + const { + clientCapabilities: { agentIDE }, + } = useConfig() + const openRemoteFile = useCallback(() => { + if (agentIDE !== CodyIDE.VSCode) { + return + } + + const uri = URI.parse(fileURL) + getVSCodeAPI().postMessage({ + command: 'openRemoteFile', + uri, + }) + }, [fileURL, agentIDE]) const handleVisibility = useCallback( (inView: boolean, entry: IntersectionObserverEntry) => { @@ -183,6 +201,7 @@ export const FileMatchSearchResult: FC {expandable && (