diff --git a/package-lock.json b/package-lock.json index bdd19b5..51f4388 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "twinny", - "version": "3.19.7", + "version": "3.19.8", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "twinny", - "version": "3.19.7", + "version": "3.19.8", "cpu": [ "x64", "arm64" @@ -50,7 +50,7 @@ "rehype-raw": "^7.0.0", "remark-gfm": "^4.0.0", "string_score": "^0.1.22", - "symmetry-core": "^1.0.13", + "symmetry-core": "^1.0.14", "tippy.js": "^6.3.7", "tiptap-markdown": "^0.8.10", "toxe": "^1.1.0", @@ -15968,9 +15968,9 @@ } }, "node_modules/symmetry-core": { - "version": "1.0.13", - "resolved": "https://registry.npmjs.org/symmetry-core/-/symmetry-core-1.0.13.tgz", - "integrity": "sha512-szkqvTvzkcZzDbdYcqaoMNwjRqXtjyw2OBtzCjX2xwZbROy5tlVA8Trsi71Xfenybrzj/SgdHzvDhWuTYumiYg==", + "version": "1.0.14", + "resolved": "https://registry.npmjs.org/symmetry-core/-/symmetry-core-1.0.14.tgz", + "integrity": "sha512-1KW9Mpgx8ZPgW77Jwv1X/qf7eaVjI8pJ/nCo8F+9QPWfy7BSTX0Bcx1iZhxy+iR7Dm1uyExiTzvp/G0/UlhlGw==", "dependencies": { "chalk": "^4.1.2", "commander": "^12.1.0", diff --git a/package.json b/package.json index 488412f..8194f35 100644 --- a/package.json +++ b/package.json @@ -2,7 +2,7 @@ "name": "twinny", "displayName": "twinny - AI Code Completion and Chat", "description": "Locally hosted AI code completion plugin for vscode", - "version": "3.19.7", + "version": "3.19.8", "icon": "assets/icon.png", "keywords": [ "code-inference", @@ -512,7 +512,7 @@ "rehype-raw": "^7.0.0", "remark-gfm": "^4.0.0", "string_score": "^0.1.22", - "symmetry-core": "^1.0.13", + "symmetry-core": "^1.0.14", "tippy.js": "^6.3.7", "tiptap-markdown": "^0.8.10", "toxe": "^1.1.0", diff --git a/src/common/constants.ts b/src/common/constants.ts index fbdac2b..201fcbd 100644 --- a/src/common/constants.ts +++ b/src/common/constants.ts @@ -25,7 +25,7 @@ export const SKIP_IMPORT_KEYWORDS_AFTER = ["from", "as", "import"] export const MIN_COMPLETION_CHUNKS = 2 export const MAX_EMPTY_COMPLETION_CHARS = 250 export const DEFAULT_RERANK_THRESHOLD = 0.5 -export const URL_SYMMETRY_WS = "wss://twinny.dev/ws" +export const URL_SYMMETRY_WS = "https://twinny.dev/ws" export const defaultChunkOptions = { maxSize: 500, diff --git a/src/extension/symmetry-service.ts b/src/extension/symmetry-service.ts index c7d9d1c..2055549 100644 --- a/src/extension/symmetry-service.ts +++ b/src/extension/symmetry-service.ts @@ -8,8 +8,7 @@ import os from "os" import path from "path" import { EventEmitter } from "stream" import { serverMessageKeys, SymmetryProvider } from "symmetry-core" -import * as vscode from "vscode" -import { commands, ExtensionContext, Webview, window, workspace } from "vscode" +import { commands, ExtensionContext, Webview, workspace } from "vscode" import { ACTIVE_CHAT_PROVIDER_STORAGE_KEY, @@ -45,7 +44,6 @@ export class SymmetryService extends EventEmitter { private _config = workspace.getConfiguration("twinny") private _completion = "" private _context: ExtensionContext - private _emitterKey = "" private _provider: SymmetryProvider | undefined private _providerPeer: undefined | Peer private _providerSwarm: undefined | typeof Hyperswarm @@ -107,38 +105,40 @@ export class SymmetryService extends EventEmitter { this._providerTopic = discoveryKey this._serverSwarm.join(this._providerTopic, { client: true, server: false }) this._serverSwarm.flush() - this._serverSwarm.on("connection", (peer: Peer) => { - this._serverPeer = peer - peer.write( - createSymmetryMessage(serverMessageKeys.requestProvider, { - modelName: model - }) - ) - peer.on("data", (message: Buffer) => { - const data = safeParseJson>( - message.toString() - ) - if (data && data.key) { - switch (data?.key) { - case serverMessageKeys.ping: - this._providerPeer?.write( - createSymmetryMessage(serverMessageKeys.pong) - ) - break - case serverMessageKeys.providerDetails: - peer.write( - createSymmetryMessage( - serverMessageKeys.verifySession, - data.data?.sessionToken - ) - ) - break - case serverMessageKeys.sessionValid: - this.connectToProvider(data.data) - } - } + this._serverSwarm.on("connection", (peer: Peer) => + this.handleServerConnection(peer, model) + ) + } + + private handleServerConnection = (peer: Peer, model: string | undefined) => { + this._serverPeer = peer + peer.write( + createSymmetryMessage(serverMessageKeys.requestProvider, { + modelName: model }) - }) + ) + peer.on("data", this.handleServerData) + } + + private handleServerData = (message: Buffer) => { + const data = safeParseJson>( + message.toString() + ) + if (!data || !data.key) return + + switch (data.key) { + case serverMessageKeys.providerDetails: + this._serverPeer?.write( + createSymmetryMessage( + serverMessageKeys.verifySession, + data.data?.sessionToken + ) + ) + break + case serverMessageKeys.sessionValid: + this.connectToProvider(data.data) + break + } } public disconnect = async () => { @@ -160,59 +160,51 @@ export class SymmetryService extends EventEmitter { server: false }) this._providerSwarm.flush() - this._providerSwarm.on("connection", (peer: any) => { - this._providerPeer = peer - this.providerListeners(peer) - this._webView?.postMessage({ - type: EVENT_NAME.twinnyConnectedToSymmetry, - value: { - data: { - modelName: connection.modelName, - name: connection.name, - provider: connection.provider - } - } - }) - this._webView?.postMessage({ - type: EVENT_NAME.twinnySetTab, - value: { - data: WEBUI_TABS.chat - } - }) - this._sessionManager?.set( - EXTENSION_SESSION_NAME.twinnySymmetryConnection, - connection - ) - commands.executeCommand( - "setContext", - EXTENSION_CONTEXT_NAME.twinnySymmetryTab, - false - ) + this._providerSwarm.on("connection", (peer: any) => + this.handleProviderConnection(peer, connection) + ) + } + + private handleProviderConnection(peer: Peer, connection: SymmetryConnection) { + this._providerPeer = peer + this.setupProviderListeners(peer) + this.notifyWebView(EVENT_NAME.twinnyConnectedToSymmetry, { + data: { + modelName: connection.modelName, + name: connection.name, + provider: connection.provider + } }) - return this + this.notifyWebView(EVENT_NAME.twinnySetTab, { data: WEBUI_TABS.chat }) + this._sessionManager?.set( + EXTENSION_SESSION_NAME.twinnySymmetryConnection, + connection + ) + commands.executeCommand( + "setContext", + EXTENSION_CONTEXT_NAME.twinnySymmetryTab, + false + ) } - private providerListeners = (peer: any) => { + private setupProviderListeners(peer: Peer) { peer.on("data", (chunk: Buffer) => { const str = chunk.toString() - if (str.includes("symmetryEmitterKey")) - this._emitterKey = JSON.parse(str).symmetryEmitterKey - - if (!this._emitterKey) return - if (str.includes(serverMessageKeys.inferenceEnded)) this.handleInferenceEnd() - - this.handleIncomingData(chunk, (response: StreamResponse) => { - if (!this._symmetryProvider) return - const data = getChatDataFromProvider(this._symmetryProvider, response) - this._completion = this._completion + data - if (!data) return - this.emit(this._emitterKey, this._completion) - }) + this.handleIncomingData(chunk, (response: StreamResponse) => + this.processResponseData(response) + ) }) } + private processResponseData(response: StreamResponse) { + if (!this._symmetryProvider) return + const data = getChatDataFromProvider(this._symmetryProvider, response) + this._completion += data + if (data) this.emit(SYMMETRY_EMITTER_KEY.inference, this._completion) + } + private handleInferenceEnd() { commands.executeCommand( "setContext", @@ -221,14 +213,12 @@ export class SymmetryService extends EventEmitter { ) if (!this._completion) return - if (this._emitterKey === SYMMETRY_EMITTER_KEY.inference) { - this._webView?.postMessage({ - type: EVENT_NAME.twinnyOnEnd, - value: { - completion: this._completion.trimStart() - } - } as ServerMessage) - } + this._webView?.postMessage({ + type: EVENT_NAME.twinnyOnEnd, + value: { + completion: this._completion.trimStart() + } + } as ServerMessage) this._completion = "" } @@ -251,13 +241,6 @@ export class SymmetryService extends EventEmitter { } } - public getChatProvider() { - const provider = this._context?.globalState.get( - ACTIVE_CHAT_PROVIDER_STORAGE_KEY - ) - return provider - } - private getSymmetryConfigPath(): string { const homeDir = os.homedir() return path.join(homeDir, ".config", "symmetry", "provider.yaml") @@ -288,7 +271,7 @@ export class SymmetryService extends EventEmitter { systemMessage: "" }) - return fs.promises.writeFile(configPath, symmetryConfiguration, "utf8"); + return fs.promises.writeFile(configPath, symmetryConfiguration, "utf8") } public startSymmetryProvider = async () => { @@ -325,6 +308,17 @@ export class SymmetryService extends EventEmitter { }) } + private notifyWebView(type: string, value: any = {}) { + this._webView?.postMessage({ type, value }) + } + + public getChatProvider() { + const provider = this._context?.globalState.get( + ACTIVE_CHAT_PROVIDER_STORAGE_KEY + ) + return provider + } + public stopSymmetryProvider = async () => { await this._provider?.destroySwarms() updateSymmetryStatus(this._webView, "disconnected")