diff --git a/.gitignore b/.gitignore index cdfd567..e060237 100644 --- a/.gitignore +++ b/.gitignore @@ -29,6 +29,7 @@ release *.pyc .ruff_cache .mypy_cache +__pycache__ # python *.pyc diff --git a/backend/__pycache__/io_models.cpython-311.pyc b/backend/__pycache__/io_models.cpython-311.pyc index 00fa152..4f5330d 100644 Binary files a/backend/__pycache__/io_models.cpython-311.pyc and b/backend/__pycache__/io_models.cpython-311.pyc differ diff --git a/backend/io_models.py b/backend/io_models.py index d5c68f2..262798b 100644 --- a/backend/io_models.py +++ b/backend/io_models.py @@ -1,7 +1,8 @@ -from typing import List, Optional +from typing import List, Optional, Dict from preql.core.models import DataType, Purpose from pydantic import BaseModel, Field +from preql_nlp.enums import Provider class LineageItem(BaseModel): @@ -27,3 +28,23 @@ class Model(BaseModel): class ListModelResponse(BaseModel): models: List[Model] + + +class GenAIConnectionInSchema(BaseModel): + name: str + provider: Provider + api_key: str = Field(alias="apiKey") + extra: Dict | None = Field(default_factory=dict) + +class QueryInSchema(BaseModel): + connection: str + query: str + # chart_type: ChartType | None = None + +class GenAIQueryInSchema(BaseModel): + connection: str + text: str + genai_connection:str + +class GenAIQueryOutSchema(BaseModel): + text:str \ No newline at end of file diff --git a/backend/main.py b/backend/main.py index 78548d9..e2c57c5 100644 --- a/backend/main.py +++ b/backend/main.py @@ -41,11 +41,20 @@ from trilogy_public_models.inventory import parse_initial_models from sqlalchemy import create_engine -from backend.io_models import ListModelResponse, Model, UIConcept +from backend.io_models import ( + ListModelResponse, + Model, + UIConcept, + GenAIConnectionInSchema, + QueryInSchema, + GenAIQueryInSchema, + GenAIQueryOutSchema, +) from backend.models.helpers import flatten_lineage from duckdb_engine import * # this is for pyinstaller from sqlalchemy_bigquery import * # this is for pyinstaller from preql.executor import generate_result_set +from preql_nlp.core import NLPEngine PORT = 5678 @@ -111,6 +120,8 @@ class InstanceSettings: CONNECTIONS: Dict[str, Executor] = {} +GENAI_CONNECTIONS: Dict[str, NLPEngine] = {} + ## BEGIN REQUESTS @@ -152,10 +163,7 @@ class ConnectionListOutput(BaseModel): connections: List[ConnectionListItem] -class QueryInSchema(BaseModel): - connection: str - query: str - # chart_type: ChartType | None = None + class QueryOutColumn(BaseModel): @@ -334,6 +342,24 @@ def create_connection(connection: ConnectionInSchema): CONNECTIONS[connection.name] = executor +@router.post("/gen_ai_connection") +def create_connection(connection: GenAIConnectionInSchema): + engine = NLPEngine( + # name=connection.name, + provider=connection.provider, + model=connection.extra.get("model", None), + api_key=connection.api_key, + ) + try: + engine.test_connection() + except Exception as e: + raise HTTPException( + status_code=400, + detail=f"Error validating connection: {str(e)}", + ) + GENAI_CONNECTIONS[connection.name] = engine + + @router.post("/raw_query") def run_raw_query(query: QueryInSchema): start = datetime.now() @@ -379,6 +405,23 @@ def run_raw_query(query: QueryInSchema): return output +@router.post("/genai_query") +def run_genai_query(query: GenAIQueryInSchema): + from preql_nlp.main_v2 import build_query as build_query_v2 + start = datetime.now() + executor = CONNECTIONS.get(query.connection) + gen_ai = GENAI_CONNECTIONS.get(query.genai_connection) + + try: + processed_query_v2 = build_query_v2( + query.text, executor.environment, debug=True, llm=gen_ai.llm + ) + + generated = executor.generator.compile_statement(processed_query_v2) + return GenAIQueryOutSchema(text = generated) + except Exception as e: + raise HTTPException(status_code=500, detail=str(e)) + @router.post("/query") def run_query(query: QueryInSchema): start = datetime.now() diff --git a/frontend/src/ vuex.d.ts b/frontend/src/ vuex.d.ts new file mode 100644 index 0000000..baf37a9 --- /dev/null +++ b/frontend/src/ vuex.d.ts @@ -0,0 +1,16 @@ +import { Store } from "vuex"; +import { GenAiConnection } from "./models/GenAIConnection"; + +declare module "@vue/runtime-core" { + // declare your own store states + interface State { + getters: { + genAIConnections: Array; + }; + } + + // provide typings for `this.$store` + interface ComponentCustomProperties { + $store: Store; + } +} diff --git a/frontend/src/components/editor/EditorEditor.vue b/frontend/src/components/editor/EditorEditor.vue index bb805b0..ec3c872 100644 --- a/frontend/src/components/editor/EditorEditor.vue +++ b/frontend/src/components/editor/EditorEditor.vue @@ -90,6 +90,11 @@ export default defineComponent({ self.generatingPrompt = false; }) }, + async submitGenAI(selection:String) { + this.info = 'Generating PreQL query from prompt...' + let response = await this.editorData.runGenAIQuery(selection); + return response + }, async submit(retry = false) { // this.loading = true; this.info = 'Executing query...' @@ -192,6 +197,30 @@ export default defineComponent({ } }); + editor.addCommand(monaco.KeyMod.CtrlCmd | monaco.KeyCode.KeyG, () => { + + if (!this.loading && this.$store.getters.hasGenAIConnection) { + // this.submit(); + var selected: monaco.Selection | monaco.Range | null = editor.getSelection(); + if (!selected) { + var line = editor.getPosition(); + if (!line) { + return + } + selected = new monaco.Range(line.lineNumber, 1, line.lineNumber, 1); + + } + console.log(selected) + this.submitGenAI(selected).then((response) => { + var op = {range: selected, text: response, forceMoveMarkers: true}; + editor.executeEdits("gen-ai-prompt-shortcut", [op]); + }).catch((error) => { + this.error = axiosHelpers.getErrorMessage(error); + } + ) + } + }); + editor.addCommand(monaco.KeyMod.CtrlCmd | monaco.KeyCode.KeyS, () => { this.saveEditorText({ contents: editor.getValue(), name: this.editorData.name, connection:this.editorData.connection }).then( ()=> { diff --git a/frontend/src/components/editor/HintsComponent.vue b/frontend/src/components/editor/HintsComponent.vue index f5fd22e..0da4a19 100644 --- a/frontend/src/components/editor/HintsComponent.vue +++ b/frontend/src/components/editor/HintsComponent.vue @@ -68,7 +68,7 @@ export default { data() { return { - shortcuts: [ + staticShortcuts: [ { name: 'Run', keys: ['Enter'] @@ -103,13 +103,29 @@ export default { required: false, }, }, + mounted() { + console.log(this.$store.getters.genAIConnections) + }, computed: { + + shortcuts():Array { + if (this.$store.getters.genAIConnections.length > 0) { + return this.staticShortcuts.concat([ + { + name: 'Run GenAI', + keys: ['G'] + } + ]) + } + return this.staticShortcuts + }, + icon() { if (this.sysType == 'mac') { return '⌘' } return 'Ctrl' - } + }, } } \ No newline at end of file diff --git a/frontend/src/components/sidebar/GenAIManager.vue b/frontend/src/components/sidebar/GenAIManager.vue index d4da3d9..0ce301d 100644 --- a/frontend/src/components/sidebar/GenAIManager.vue +++ b/frontend/src/components/sidebar/GenAIManager.vue @@ -3,10 +3,47 @@ +
+ + + + + +
{{ connection.name }}
+ +
+ + + + + + + + + + + +
+
+
+ + diff --git a/frontend/src/components/sidebar/SidebarSelector.vue b/frontend/src/components/sidebar/SidebarSelector.vue index 7644a19..28227e4 100644 --- a/frontend/src/components/sidebar/SidebarSelector.vue +++ b/frontend/src/components/sidebar/SidebarSelector.vue @@ -38,7 +38,7 @@ export default { links: [{ 'name': 'Connections', 'address': 'connections', 'icon': 'mdi-database' }, { 'name': 'History', 'address': 'history', 'icon': 'mdi-history' }, { 'name': "Models", 'address': 'models', 'icon': 'mdi-table' }, - { 'name': 'Other', 'address': 'other', 'icon': 'mdi-dots-horizontal' }, + // { 'name': 'Other', 'address': 'other', 'icon': 'mdi-dots-horizontal' }, { 'name': 'GenAI', 'address': 'genai', 'icon': 'mdi-brain' }, ], diff --git a/frontend/src/components/sidebar/genai/NewConnectionPopup.vue b/frontend/src/components/sidebar/genai/NewConnectionPopup.vue new file mode 100644 index 0000000..336eb8e --- /dev/null +++ b/frontend/src/components/sidebar/genai/NewConnectionPopup.vue @@ -0,0 +1,110 @@ + + + \ No newline at end of file diff --git a/frontend/src/components/sidebar/genai/utility.ts b/frontend/src/components/sidebar/genai/utility.ts new file mode 100644 index 0000000..a1a13d2 --- /dev/null +++ b/frontend/src/components/sidebar/genai/utility.ts @@ -0,0 +1,9 @@ +import { GenAIType } from "/src/models/GenAIConnection"; + +export function getConnectionExtras(type: GenAIType): Array { + if (type === GenAIType.OPENAI) { + return ["model"]; + } else { + return []; + } +} diff --git a/frontend/src/models/ChatConnection.ts b/frontend/src/models/ChatConnection.ts deleted file mode 100644 index e261393..0000000 --- a/frontend/src/models/ChatConnection.ts +++ /dev/null @@ -1,19 +0,0 @@ - - -export enum ChatType { - OPENAI = 'OPENAI', -} - -export class ChatConnection { - type: ChatType, - api_key: string; - api_secret: string; - name: string; - - constructor( type: ChatType, api_key: string, api_secret: string, name: string) { - this.type = type - this.api_key = api_key - this.api_secret = api_secret - this.name = name - } -} \ No newline at end of file diff --git a/frontend/src/models/Connection.ts b/frontend/src/models/Connection.ts index ce12c5e..66bbd22 100644 --- a/frontend/src/models/Connection.ts +++ b/frontend/src/models/Connection.ts @@ -21,9 +21,8 @@ export class Connection implements ConnectionInterface { this.model = model this.extra = extra } - - // @ts-ignore - static fromJSON({ name, type, active, model, extra }) { + + public static fromJSON({ name, type, active, model, extra }) { return new Connection(name, type, false, model, extra) } } \ No newline at end of file diff --git a/frontend/src/models/Editor.ts b/frontend/src/models/Editor.ts index 36c889e..0d9e7f2 100644 --- a/frontend/src/models/Editor.ts +++ b/frontend/src/models/Editor.ts @@ -142,7 +142,39 @@ export class RawEditor implements EditorInterface { this.status_code = 200; this.visible = true; } + async runGenAIQuery() { + this.loading = true; + this.error = null; + this.executed = true; + let local = this; + const startTime = new Date(); + try { + let info = { connection: local.connection, query: local.contents }; + await instance.post('genai_query', info).then(function (response) { + const columnMap = new Map(); + for (const [key, value] of response.data.columns) { + columnMap.set(key, value); + } + local.status_code = 200; + local.results = new Results(response.data.results, columnMap); //response.data; + const endTime = new Date(); + local.duration = endTime.getTime() - startTime.getTime(); + local.executed = true; + }) + // this.last_passed_query_text = current_query; + } catch (error) { + if (error instanceof Error) { + const resultCode = axiosHelpers.getResultCode(error); + local.status_code = resultCode; + local.error = axiosHelpers.getErrorMessage(error); + local.duration = null; + local.executed = false; + } + } finally { + local.loading = false; + } + } async runQuery() { this.loading = true; this.error = null; diff --git a/frontend/src/models/GenAIConnection.ts b/frontend/src/models/GenAIConnection.ts new file mode 100644 index 0000000..4310744 --- /dev/null +++ b/frontend/src/models/GenAIConnection.ts @@ -0,0 +1,30 @@ +export enum GenAIType { + OPENAI = "openai", + GOOGLE = "google" +} + +export class GenAIConnection { + type: GenAIType; + apiKey: string; + name: string; + active: boolean; + extra: object | null; + + constructor( + type: GenAIType, + apiKey: string, + name: string, + active: boolean, + extra: object | null + ) { + this.type = type; + this.apiKey = apiKey; + this.name = name; + this.active = active || false; + this.extra = extra; + } + + public static fromJSON({ type, apiKey, name, active, extra }) { + return new GenAIConnection(type, apiKey, name, active, extra); + } +} diff --git a/frontend/src/store/modules/nlp.ts b/frontend/src/store/modules/nlp.ts index ed07cbf..3a5a3dc 100644 --- a/frontend/src/store/modules/nlp.ts +++ b/frontend/src/store/modules/nlp.ts @@ -1,12 +1,26 @@ +import { GenAIConnection } from "/src/models/GenAIConnection.ts"; -import { ChatType, ChatConnection } from '/src/models/ChatConnection'; +import Store from "electron-store"; +import instance from "/src/api/instance"; +const storageName = "nlp"; -import Store from 'electron-store' -import instance from '/src/api/instance' +const storageAPI = { + setGenAIConnections(value: Array) { + // const buffer = safeStorage.encryptString(value); + store.set(storageName, value); + // store.set(key, buffer.toString(encoding)); + }, - -const storageName = 'nlp'; + getGenAIConnections(): Array { + const data = store.get(storageName, []) as Array; + const parsed = data.map((dict) => { + return GenAIConnection.fromJSON(dict); + }); + console.log(parsed); + return parsed; + }, +}; const store = new Store>({ name: storageName, @@ -14,38 +28,52 @@ const store = new Store>({ }); const state = { - chatConnections: [], + genAIConnections: storageAPI.getGenAIConnections(), }; const getters = { - chatConnections: state => state.chatConnections, - getChatConnectionByName: (state) => (name) => { - return state.chatConnections.find(conn => conn.name === name) - }, + hasGenAIConnection: (state) => state.genAIConnections.length>0, + genAIConnections: (state) => state.genAIConnections, + getGenAIConnectionByName: (state) => (name) => { + return state.genAIConnections.find((conn) => conn.name === name); + }, }; - const actions = { - async addChatConnection({ commit, rootGetters }, data) { - const apiArgs = getConnectionArgument(rootGetters, data) - instance.post('/connection', apiArgs).then(() => { - commit('setConnectionActive', data) - }) - }, - + async addGenAIConnection({ commit, rootGetters }, data) { + console.log(data); + instance.post("/gen_ai_connection", data).then(() => { + const connection = GenAIConnection.fromJSON({ + type: data.type, + name: data.name, + extra: data.extra, + apiKey: data.extra, + active: true, + }); + commit("addGenAIConnection", connection); + }); + }, + async setGenAIConnectionState({ commit, rootGetters }, args) { + commit("setGenAIConnectionState", args); + }, }; - const mutations = { - async addChatConnection(state, connection) { - const index = state.connections.findIndex(c => c.name === connection.name) - state.chatConnections[index].active = true - } + async addGenAIConnection(state, connection) { + state.genAIConnections.push(connection); + storageAPI.setGenAIConnections(state.genAIConnections); + }, + async setGenAIConnectionState(state, args) { + const index = state.genAIConnections.findIndex( + (c) => c.name === args.connection.name + ); + state.genAIConnections[index].active = args.active; + }, }; export default { - state, - getters, - actions, - mutations -}; \ No newline at end of file + state, + getters, + actions, + mutations, +};