From 2bfaf428d9af56ec4b13d2228748fb465e653429 Mon Sep 17 00:00:00 2001 From: Ishaan Dey Date: Fri, 3 May 2024 00:52:01 -0700 Subject: [PATCH] finish generate logic --- backend/server/dist/index.js | 35 +++++-- backend/server/src/index.ts | 51 ++++++++-- frontend/components/editor/generate.tsx | 125 +++++++++++++++++++----- frontend/components/editor/index.tsx | 21 +++- frontend/lib/actions.ts | 32 +----- 5 files changed, 189 insertions(+), 75 deletions(-) diff --git a/backend/server/dist/index.js b/backend/server/dist/index.js index 47a8e4a..c5fbff5 100644 --- a/backend/server/dist/index.js +++ b/backend/server/dist/index.js @@ -100,7 +100,6 @@ io.on("connection", (socket) => __awaiter(void 0, void 0, void 0, function* () { })); socket.on("createFile", (name) => __awaiter(void 0, void 0, void 0, function* () { const id = `projects/${data.id}/${name}`; - console.log("create file", id, name); fs_1.default.writeFile(path_1.default.join(dirName, id), "", function (err) { if (err) throw err; @@ -143,14 +142,12 @@ io.on("connection", (socket) => __awaiter(void 0, void 0, void 0, function* () { callback(newFiles.files); })); socket.on("createTerminal", ({ id }) => { - console.log("creating terminal, id=" + id); const pty = (0, node_pty_1.spawn)(os_1.default.platform() === "win32" ? "cmd.exe" : "bash", [], { name: "xterm", cols: 100, cwd: path_1.default.join(dirName, "projects", data.id), }); const onData = pty.onData((data) => { - console.log(data); socket.emit("terminalResponse", { // data: Buffer.from(data, "utf-8").toString("base64"), data, @@ -165,11 +162,7 @@ io.on("connection", (socket) => __awaiter(void 0, void 0, void 0, function* () { }; }); socket.on("terminalData", (id, data) => { - // socket.on("terminalData", (data: string) => { - console.log(`Received data for terminal ${id}: ${data}`); - // pty.write(data) if (!terminals[id]) { - console.log("terminal not found"); console.log("terminals", terminals); return; } @@ -180,6 +173,34 @@ io.on("connection", (socket) => __awaiter(void 0, void 0, void 0, function* () { console.log("Error writing to terminal", e); } }); + socket.on("generateCode", (fileName, code, line, instructions, callback) => __awaiter(void 0, void 0, void 0, function* () { + console.log("Generating code..."); + const res = yield fetch(`https://api.cloudflare.com/client/v4/accounts/${process.env.CF_USER_ID}/ai/run/@cf/meta/llama-3-8b-instruct`, { + method: "POST", + headers: { + "Content-Type": "application/json", + Authorization: `Bearer ${process.env.CF_API_TOKEN}`, + }, + body: JSON.stringify({ + messages: [ + { + role: "system", + content: "You are an expert coding assistant. You read code from a file, and you suggest new code to add to the file. You may be given instructions on what to generate, which you should follow. You should generate code that is correct, efficient, and follows best practices. You should also generate code that is clear and easy to read.", + }, + { + role: "user", + content: `The file is called ${fileName}. Here are my instructions on what to generate: ${instructions}. Suggest me code to insert at line ${line} in my file. + +My code file content: ${code} + +Return only the code, and nothing else. Do not include backticks.`, + }, + ], + }), + }); + const json = yield res.json(); + callback(json); + })); socket.on("disconnect", () => { Object.entries(terminals).forEach((t) => { const { terminal, onData, onExit } = t[1]; diff --git a/backend/server/src/index.ts b/backend/server/src/index.ts index af95922..136eac6 100644 --- a/backend/server/src/index.ts +++ b/backend/server/src/index.ts @@ -116,7 +116,6 @@ io.on("connection", async (socket) => { socket.on("createFile", async (name: string) => { const id = `projects/${data.id}/${name}` - console.log("create file", id, name) fs.writeFile(path.join(dirName, id), "", function (err) { if (err) throw err @@ -170,7 +169,6 @@ io.on("connection", async (socket) => { }) socket.on("createTerminal", ({ id }: { id: string }) => { - console.log("creating terminal, id=" + id) const pty = spawn(os.platform() === "win32" ? "cmd.exe" : "bash", [], { name: "xterm", cols: 100, @@ -178,7 +176,6 @@ io.on("connection", async (socket) => { }) const onData = pty.onData((data) => { - console.log(data) socket.emit("terminalResponse", { // data: Buffer.from(data, "utf-8").toString("base64"), data, @@ -197,12 +194,7 @@ io.on("connection", async (socket) => { }) socket.on("terminalData", (id: string, data: string) => { - // socket.on("terminalData", (data: string) => { - console.log(`Received data for terminal ${id}: ${data}`) - // pty.write(data) - if (!terminals[id]) { - console.log("terminal not found") console.log("terminals", terminals) return } @@ -214,6 +206,49 @@ io.on("connection", async (socket) => { } }) + socket.on( + "generateCode", + async ( + fileName: string, + code: string, + line: number, + instructions: string, + callback + ) => { + console.log("Generating code...") + const res = await fetch( + `https://api.cloudflare.com/client/v4/accounts/${process.env.CF_USER_ID}/ai/run/@cf/meta/llama-3-8b-instruct`, + { + method: "POST", + headers: { + "Content-Type": "application/json", + Authorization: `Bearer ${process.env.CF_API_TOKEN}`, + }, + body: JSON.stringify({ + messages: [ + { + role: "system", + content: + "You are an expert coding assistant. You read code from a file, and you suggest new code to add to the file. You may be given instructions on what to generate, which you should follow. You should generate code that is correct, efficient, and follows best practices. You should also generate code that is clear and easy to read.", + }, + { + role: "user", + content: `The file is called ${fileName}. Here are my instructions on what to generate: ${instructions}. Suggest me code to insert at line ${line} in my file. + +My code file content: ${code} + +Return only the code, and nothing else. Do not include backticks.`, + }, + ], + }), + } + ) + + const json = await res.json() + callback(json) + } + ) + socket.on("disconnect", () => { Object.entries(terminals).forEach((t) => { const { terminal, onData, onExit } = t[1] diff --git a/frontend/components/editor/generate.tsx b/frontend/components/editor/generate.tsx index 762560c..c56ed56 100644 --- a/frontend/components/editor/generate.tsx +++ b/frontend/components/editor/generate.tsx @@ -1,33 +1,41 @@ "use client" import { useEffect, useRef, useState } from "react" -import { Input } from "../ui/input" import { Button } from "../ui/button" import { Check, Loader2, RotateCw, Sparkles, X } from "lucide-react" +import { Socket } from "socket.io-client" +import { Editor } from "@monaco-editor/react" +// import monaco from "monaco-editor" export default function GenerateInput({ - cancel, - submit, + socket, width, + data, + editor, onExpand, onAccept, }: { - cancel: () => void - submit: (input: string) => void + socket: Socket width: number + data: { + fileName: string + code: string + line: number + } + editor: { + language: string + } onExpand: () => void onAccept: (code: string) => void }) { const inputRef = useRef(null) - const [code, setCode] = useState(`function add(a: number, b: number): number { - return a + b; - } - - const result = add(2, 3); - console.log(result);`) + const [code, setCode] = useState("") const [expanded, setExpanded] = useState(false) - const [loading, setLoading] = useState(false) + const [loading, setLoading] = useState({ + generate: false, + regenerate: false, + }) const [input, setInput] = useState("") const [currentPrompt, setCurrentPrompt] = useState("") @@ -37,17 +45,45 @@ export default function GenerateInput({ }, 0) }, []) - const handleGenerate = async () => { - setLoading(true) + const handleGenerate = async ({ + regenerate = false, + }: { + regenerate?: boolean + }) => { + setLoading({ generate: !regenerate, regenerate }) setCurrentPrompt(input) - // const res = await generateCode() - await new Promise((resolve) => setTimeout(resolve, 1000)) + socket.emit( + "generateCode", + data.fileName, + data.code, + data.line, + regenerate ? currentPrompt : input, + (res: { + result: { + response: string + } + success: boolean + errors: any[] + messages: any[] + }) => { + if (!res.success) { + console.error(res.errors) + return + } - setExpanded(true) - onExpand() - setLoading(false) + setCode(res.result.response) + } + ) } + useEffect(() => { + if (code) { + setExpanded(true) + onExpand() + setLoading({ generate: false, regenerate: false }) + } + }, [code]) + return (
@@ -64,10 +100,10 @@ export default function GenerateInput({
diff --git a/frontend/components/editor/index.tsx b/frontend/components/editor/index.tsx index 5d36c0f..5e089e3 100644 --- a/frontend/components/editor/index.tsx +++ b/frontend/components/editor/index.tsx @@ -2,7 +2,7 @@ import Editor, { BeforeMount, OnMount } from "@monaco-editor/react" import monaco from "monaco-editor" -import { use, useEffect, useRef, useState } from "react" +import { useEffect, useRef, useState } from "react" // import theme from "./theme.json" import { @@ -372,9 +372,16 @@ export default function CodeEditor({
{generate.show ? ( {}} - submit={(str: string) => {}} + socket={socket} width={generate.width - 90} + data={{ + fileName: tabs.find((t) => t.id === activeId)?.name ?? "", + code: editorRef.current?.getValue() ?? "", + line: generate.line, + }} + editor={{ + language: editorLanguage, + }} onExpand={() => { editorRef.current?.changeViewZones(function (changeAccessor) { changeAccessor.removeZone(generate.id) @@ -391,13 +398,19 @@ export default function CodeEditor({ }) }} onAccept={(code: string) => { + const line = generate.line setGenerate((prev) => { return { ...prev, show: !prev.show, } }) - console.log("accepted:", code) + const file = editorRef.current?.getValue() + + const lines = file?.split("\n") || [] + lines.splice(line - 1, 0, code) + const updatedFile = lines.join("\n") + editorRef.current?.setValue(updatedFile) }} /> ) : null} diff --git a/frontend/lib/actions.ts b/frontend/lib/actions.ts index 6f83aa9..58cfa29 100644 --- a/frontend/lib/actions.ts +++ b/frontend/lib/actions.ts @@ -24,7 +24,7 @@ export async function updateSandbox(body: { name?: string visibility?: "public" | "private" }) { - const res = await fetch("http://localhost:8787/api/sandbox", { + await fetch("http://localhost:8787/api/sandbox", { method: "POST", headers: { "Content-Type": "application/json", @@ -36,7 +36,7 @@ export async function updateSandbox(body: { } export async function deleteSandbox(id: string) { - const res = await fetch(`http://localhost:8787/api/sandbox?id=${id}`, { + await fetch(`http://localhost:8787/api/sandbox?id=${id}`, { method: "DELETE", }) @@ -62,7 +62,7 @@ export async function shareSandbox(sandboxId: string, email: string) { } export async function unshareSandbox(sandboxId: string, userId: string) { - const res = await fetch("http://localhost:8787/api/sandbox/share", { + await fetch("http://localhost:8787/api/sandbox/share", { method: "DELETE", headers: { "Content-Type": "application/json", @@ -72,29 +72,3 @@ export async function unshareSandbox(sandboxId: string, userId: string) { revalidatePath(`/code/${sandboxId}`) } - -export async function generateCode(code: string, line: number) { - const res = await fetch( - "https://api.cloudflare.com/client/v4/accounts/d18f2f848da38e37adc9a34eab3d5ae2/ai/run/@cf/meta/llama-3-8b-instruct", - { - method: "POST", - headers: { - "Content-Type": "application/json", - Authorization: `Bearer ${process.env.CF_API_TOKEN}`, - }, - body: JSON.stringify({ - messages: [ - { - role: "system", - content: - "You are an expert coding assistant who reads from an existing code file, and suggests code to add to the file.", - }, - { - role: "user", - content: "", //todo - }, - ], - }), - } - ) -}