|
| 1 | +import { zValidator } from "@hono/zod-validator"; |
1 | 2 | import { Hono } from "hono"; |
2 | 3 | import { proxy } from "hono/proxy"; |
| 4 | +import { z } from "zod"; |
3 | 5 |
|
4 | 6 | import { getEnv } from "./env"; |
5 | 7 | import { supabaseMiddleware } from "./middleware/supabase"; |
| 8 | +import type { Env } from "./types"; |
6 | 9 |
|
7 | | -const app = new Hono(); |
| 10 | +const app = new Hono<Env>(); |
8 | 11 | app.use("/v1", supabaseMiddleware()); |
9 | 12 |
|
10 | 13 | app.get("/health", (c) => c.text("OK")); |
11 | 14 |
|
12 | | -app.post("/v1/write", async (c) => { |
13 | | - return c.json({ message: "OK" }); |
14 | | -}); |
| 15 | +app.post( |
| 16 | + "/v1/write", |
| 17 | + zValidator( |
| 18 | + "json", |
| 19 | + z.discriminatedUnion("operation", [ |
| 20 | + z.object({ |
| 21 | + table: z.string(), |
| 22 | + row_id: z.string(), |
| 23 | + operation: z.literal("delete"), |
| 24 | + }), |
| 25 | + z.object({ |
| 26 | + table: z.string(), |
| 27 | + row_id: z.string(), |
| 28 | + data: z.record(z.string(), z.unknown()), |
| 29 | + operation: z.literal("update"), |
| 30 | + }), |
| 31 | + ]), |
| 32 | + ), |
| 33 | + async (c) => { |
| 34 | + const supabase = c.get("supabase"); |
| 35 | + const user = c.get("user"); |
| 36 | + const body = c.req.valid("json"); |
| 37 | + |
| 38 | + // TODO: use RPC / transaction |
| 39 | + if (body.operation === "delete") { |
| 40 | + await supabase.from(body.table).delete().eq("id", body.row_id); |
| 41 | + } else { |
| 42 | + await supabase.from(body.table).upsert({ |
| 43 | + ...body.data, |
| 44 | + id: body.row_id, |
| 45 | + user_id: user.id, |
| 46 | + }); |
| 47 | + } |
| 48 | + |
| 49 | + return c.json({ message: "OK" }); |
| 50 | + }, |
| 51 | +); |
15 | 52 |
|
16 | 53 | app.post("/v1/chat/completions", async (c) => { |
17 | 54 | const { OPENAI_BASE_URL, OPENAI_DEFAULT_MODEL, OPENAI_API_KEY } = getEnv(c); |
18 | | - |
19 | 55 | const data = await c.req.json(); |
20 | 56 |
|
21 | 57 | const res = await proxy( |
|
0 commit comments