Compare commits
109 Commits
more-js-ap
...
hide-model
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ead981b900 | ||
|
|
e0d0cc0df1 | ||
|
|
b4cb931f6c | ||
|
|
7df1c59bd3 | ||
|
|
c83863f468 | ||
|
|
40638a7848 | ||
|
|
33ca98b267 | ||
|
|
39c943f2ec | ||
|
|
14eae45d18 | ||
|
|
2aa4ac1594 | ||
|
|
13bac46e0b | ||
|
|
42ade01f22 | ||
|
|
59b79049c1 | ||
|
|
0d7433cb7e | ||
|
|
12d01cd3d5 | ||
|
|
ec59252010 | ||
|
|
87e2339df2 | ||
|
|
75ad6619a5 | ||
|
|
4b8941d53a | ||
|
|
0d691d17cc | ||
|
|
815d4faad2 | ||
|
|
9632ccbc71 | ||
|
|
a4131e4a10 | ||
|
|
db1c8f171d | ||
|
|
678392ef17 | ||
|
|
af722128e8 | ||
|
|
50a79b6e3a | ||
|
|
f59150ff5b | ||
|
|
b58e0a8d54 | ||
|
|
dc82a3fa82 | ||
|
|
fedbf5784e | ||
|
|
888c04af50 | ||
|
|
1b36453051 | ||
|
|
2f37b3ed87 | ||
|
|
8fa7b691db | ||
|
|
17866a5249 | ||
|
|
947eba3216 | ||
|
|
ef1f9458f4 | ||
|
|
c6c7e746ee | ||
|
|
3be0a90960 | ||
|
|
9b1f2ac30a | ||
|
|
1b394cc72b | ||
|
|
26b9731bab | ||
|
|
7c8ec8f6a7 | ||
|
|
10dd53e7f6 | ||
|
|
b1802fc04b | ||
|
|
f2135ddc72 | ||
|
|
ca89eafb0b | ||
|
|
b50d47beaf | ||
|
|
733d53625b | ||
|
|
a5e59e4235 | ||
|
|
d0102e3202 | ||
|
|
bd571c4c4e | ||
|
|
296eb23d97 | ||
|
|
4e2ae7a441 | ||
|
|
072dcee376 | ||
|
|
94464c0617 | ||
|
|
980644f13c | ||
|
|
6a56250001 | ||
|
|
b1c7bbbd4a | ||
|
|
3e20fa31ca | ||
|
|
48a8e64be1 | ||
|
|
f3a5f11195 | ||
|
|
da5cbaf4dc | ||
|
|
acf74909c9 | ||
|
|
edac8da4a8 | ||
|
|
687f3dd85f | ||
|
|
0cef3ab5bd | ||
|
|
756b3185de | ||
|
|
3776ffc4c3 | ||
|
|
82549122e1 | ||
|
|
56a96a7db6 | ||
|
|
1596b15727 | ||
|
|
70d4a5bd9a | ||
|
|
c6ec901374 | ||
|
|
ad7665664a | ||
|
|
108e3d1e85 | ||
|
|
76f600722a | ||
|
|
d9a0e4581f | ||
|
|
b9251ad93c | ||
|
|
809ef04dc1 | ||
|
|
0fba2c9ee7 | ||
|
|
ac2ca0f617 | ||
|
|
73b9e40ced | ||
|
|
3447e863cc | ||
|
|
897e77b054 | ||
|
|
b22a4cd93b | ||
|
|
3547c85c86 | ||
|
|
9636fa033e | ||
|
|
890a738568 | ||
|
|
7003595e76 | ||
|
|
00df4453d3 | ||
|
|
4c325fc1cc | ||
|
|
dfee8a0ed7 | ||
|
|
0b4e116783 | ||
|
|
2bcb1d16a3 | ||
|
|
6e7efee21e | ||
|
|
bb9c3a9e61 | ||
|
|
11bfb5d5e4 | ||
|
|
b00ab933b3 | ||
|
|
634739c045 | ||
|
|
9a9cbe8fd4 | ||
|
|
649dc3376b | ||
|
|
05e774d021 | ||
|
|
0e328b13dc | ||
|
|
0a18ca9cd6 | ||
|
|
a5fe35912e | ||
|
|
3d3ddbe7a9 | ||
|
|
d8a5617dee |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -3,3 +3,4 @@
|
|||||||
*.pyc
|
*.pyc
|
||||||
node_modules/
|
node_modules/
|
||||||
*.tsbuildinfo
|
*.tsbuildinfo
|
||||||
|
dist/
|
||||||
2
.prettierignore
Normal file
2
.prettierignore
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
*.schema.json
|
||||||
|
app/pnpm-lock.yaml
|
||||||
104
README.md
104
README.md
@@ -1,16 +1,52 @@
|
|||||||
<!-- <img src="https://github.com/openpipe/openpipe/assets/41524992/ca59596e-eb80-40f9-921f-6d67f6e6d8fa" width="72px" /> -->
|
<p align="center">
|
||||||
|
<a href="https://openpipe.ai">
|
||||||
|
<img height="70" src="https://github.com/openpipe/openpipe/assets/41524992/70af25fb-1f90-42d9-8a20-3606e3b5aaba" alt="logo">
|
||||||
|
</a>
|
||||||
|
</p>
|
||||||
|
<h1 align="center">
|
||||||
|
OpenPipe
|
||||||
|
</h1>
|
||||||
|
|
||||||
# OpenPipe
|
<p align="center">
|
||||||
|
<i>Turn expensive prompts into cheap fine-tuned models.</i>
|
||||||
|
</p>
|
||||||
|
|
||||||
OpenPipe is a flexible playground for comparing and optimizing LLM prompts. It lets you quickly generate, test and compare candidate prompts, and can automatically [translate](#-translate-between-model-apis) those prompts between models.
|
<p align="center">
|
||||||
|
<a href="/LICENSE"><img alt="License Apache-2.0" src="https://img.shields.io/github/license/openpipe/openpipe?style=flat-square"></a>
|
||||||
|
<a href='http://makeapullrequest.com'><img alt='PRs Welcome' src='https://img.shields.io/badge/PRs-welcome-brightgreen.svg?style=flat-square'/></a>
|
||||||
|
<a href="https://github.com/openpipe/openpipe/graphs/commit-activity"><img alt="GitHub commit activity" src="https://img.shields.io/github/commit-activity/m/openpipe/openpipe?style=flat-square"/></a>
|
||||||
|
<a href="https://github.com/openpipe/openpipe/issues"><img alt="GitHub closed issues" src="https://img.shields.io/github/issues-closed/openpipe/openpipe?style=flat-square"/></a>
|
||||||
|
</p>
|
||||||
|
|
||||||
<img src="https://github.com/openpipe/openpipe/assets/41524992/219a844e-3f4e-4f6b-8066-41348b42977b" alt="demo">
|
<p align="center">
|
||||||
|
<a href="https://app.openpipe.ai/">Hosted App</a> - <a href="#running-locally">Running Locally</a> - <a href="#sample-experiments">Experiments</a>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<br>
|
||||||
|
Use powerful but expensive LLMs to fine-tune smaller and cheaper models suited to your exact needs. Evaluate model and prompt combinations in the playground. Query your past requests and export optimized training data. Try it out at https://app.openpipe.ai or <a href="#running-locally">run it locally</a>.
|
||||||
|
<br>
|
||||||
|
|
||||||
|
|
||||||
|
## 🪛 Features
|
||||||
|
|
||||||
|
* <b>Experiment</b>
|
||||||
|
* Bulk-test wide-reaching scenarios using code templating.
|
||||||
|
* Seamlessly translate prompts across different model APIs.
|
||||||
|
* Tap into autogenerated scenarios for fresh test perspectives.
|
||||||
|
|
||||||
|
* <b>Fine-Tune (Beta)</b>
|
||||||
|
* Easy integration with OpenPipe's SDK in both Python and JS.
|
||||||
|
* Swiftly query logs using intuitive built-in filters.
|
||||||
|
* Export data in multiple training formats, including Alpaca and ChatGPT, with deduplication.
|
||||||
|
|
||||||
|
<img src="https://github.com/openpipe/openpipe/assets/41524992/eaa8b92d-4536-4f63-bbef-4b0b1a60f6b5" alt="fine-tune demo">
|
||||||
|
|
||||||
|
<!-- <img height="400px" src="https://github.com/openpipe/openpipe/assets/41524992/66bb1843-cb72-4130-a369-eec2df3b8201" alt="playground demo"> -->
|
||||||
|
|
||||||
You can use our hosted version of OpenPipe at https://openpipe.ai. You can also clone this repository and [run it locally](#running-locally).
|
|
||||||
|
|
||||||
## Sample Experiments
|
## Sample Experiments
|
||||||
|
|
||||||
These are simple experiments users have created that show how OpenPipe works. Feel free to fork them and start experimenting yourself.
|
These are sample experiments users have created that show how OpenPipe works. Feel free to fork them and start experimenting yourself.
|
||||||
|
|
||||||
- [Twitter Sentiment Analysis](https://app.openpipe.ai/experiments/62c20a73-2012-4a64-973c-4b665ad46a57)
|
- [Twitter Sentiment Analysis](https://app.openpipe.ai/experiments/62c20a73-2012-4a64-973c-4b665ad46a57)
|
||||||
- [Reddit User Needs](https://app.openpipe.ai/experiments/22222222-2222-2222-2222-222222222222)
|
- [Reddit User Needs](https://app.openpipe.ai/experiments/22222222-2222-2222-2222-222222222222)
|
||||||
@@ -19,43 +55,25 @@ These are simple experiments users have created that show how OpenPipe works. Fe
|
|||||||
|
|
||||||
## Supported Models
|
## Supported Models
|
||||||
|
|
||||||
- All models available through the OpenAI [chat completion API](https://platform.openai.com/docs/guides/gpt/chat-completions-api)
|
#### OpenAI
|
||||||
- Llama2 [7b chat](https://replicate.com/a16z-infra/llama7b-v2-chat), [13b chat](https://replicate.com/a16z-infra/llama13b-v2-chat), [70b chat](https://replicate.com/replicate/llama70b-v2-chat).
|
- [GPT 3.5 Turbo](https://platform.openai.com/docs/guides/gpt/chat-completions-api)
|
||||||
- Anthropic's [Claude 1 Instant](https://www.anthropic.com/index/introducing-claude) and [Claude 2](https://www.anthropic.com/index/claude-2)
|
- [GPT 3.5 Turbo 16k](https://platform.openai.com/docs/guides/gpt/chat-completions-api)
|
||||||
|
- [GPT 4](https://openai.com/gpt-4)
|
||||||
## Features
|
#### Llama2
|
||||||
|
- [7b chat](https://replicate.com/a16z-infra/llama7b-v2-chat)
|
||||||
### 🔍 Visualize Responses
|
- [13b chat](https://replicate.com/a16z-infra/llama13b-v2-chat)
|
||||||
|
- [70b chat](https://replicate.com/replicate/llama70b-v2-chat)
|
||||||
Inspect prompt completions side-by-side.
|
#### Llama2 Fine-Tunes
|
||||||
|
- [Open-Orca/OpenOrcaxOpenChat-Preview2-13B](https://huggingface.co/Open-Orca/OpenOrcaxOpenChat-Preview2-13B)
|
||||||
### 🧪 Bulk-Test
|
- [Open-Orca/OpenOrca-Platypus2-13B](https://huggingface.co/Open-Orca/OpenOrca-Platypus2-13B)
|
||||||
|
- [NousResearch/Nous-Hermes-Llama2-13b](https://huggingface.co/NousResearch/Nous-Hermes-Llama2-13b)
|
||||||
OpenPipe lets you _template_ a prompt. Use the templating feature to run the prompts you're testing against many potential inputs for broad coverage of your problem space.
|
- [jondurbin/airoboros-l2-13b-gpt4-2.0](https://huggingface.co/jondurbin/airoboros-l2-13b-gpt4-2.0)
|
||||||
|
- [lmsys/vicuna-13b-v1.5](https://huggingface.co/lmsys/vicuna-13b-v1.5)
|
||||||
### 📟 Translate between Model APIs
|
- [Gryphe/MythoMax-L2-13b](https://huggingface.co/Gryphe/MythoMax-L2-13b)
|
||||||
|
- [NousResearch/Nous-Hermes-llama-2-7b](https://huggingface.co/NousResearch/Nous-Hermes-llama-2-7b)
|
||||||
Write your prompt in one format and automatically convert it to work with any other model.
|
#### Anthropic
|
||||||
|
- [Claude 1 Instant](https://www.anthropic.com/index/introducing-claude)
|
||||||
<img width="480" alt="Screenshot 2023-08-01 at 11 55 38 PM" src="https://github.com/OpenPipe/OpenPipe/assets/41524992/1e19ccf2-96b6-4e93-a3a5-1449710d1b5b" alt="translate between models">
|
- [Claude 2](https://www.anthropic.com/index/claude-2)
|
||||||
|
|
||||||
<br><br>
|
|
||||||
|
|
||||||
### 🛠️ Refine Your Prompts Automatically
|
|
||||||
|
|
||||||
Use a growing database of best-practice refinements to improve your prompts automatically.
|
|
||||||
|
|
||||||
<img width="480" alt="Screenshot 2023-08-01 at 11 55 38 PM" src="https://github.com/OpenPipe/OpenPipe/assets/41524992/87a27fe7-daef-445c-a5e2-1c82b23f9f99" alt="add function call">
|
|
||||||
|
|
||||||
<br><br>
|
|
||||||
|
|
||||||
### 🪄 Auto-generate Test Scenarios
|
|
||||||
|
|
||||||
OpenPipe includes a tool to generate new test scenarios based on your existing prompts and scenarios. Just click "Autogenerate Scenario" to try it out!
|
|
||||||
|
|
||||||
<img width="600" src="https://github.com/openpipe/openpipe/assets/41524992/219a844e-3f4e-4f6b-8066-41348b42977b" alt="auto-generate">
|
|
||||||
|
|
||||||
<br><br>
|
|
||||||
|
|
||||||
## Running Locally
|
## Running Locally
|
||||||
|
|
||||||
|
|||||||
@@ -34,3 +34,9 @@ GITHUB_CLIENT_SECRET="your_secret"
|
|||||||
|
|
||||||
OPENPIPE_BASE_URL="http://localhost:3000/api/v1"
|
OPENPIPE_BASE_URL="http://localhost:3000/api/v1"
|
||||||
OPENPIPE_API_KEY="your_key"
|
OPENPIPE_API_KEY="your_key"
|
||||||
|
|
||||||
|
SENDER_EMAIL="placeholder"
|
||||||
|
SMTP_HOST="placeholder"
|
||||||
|
SMTP_PORT="placeholder"
|
||||||
|
SMTP_LOGIN="placeholder"
|
||||||
|
SMTP_PASSWORD="placeholder"
|
||||||
|
|||||||
@@ -1,2 +0,0 @@
|
|||||||
*.schema.json
|
|
||||||
pnpm-lock.yaml
|
|
||||||
7
app/@types/nextjs-routes.d.ts
vendored
7
app/@types/nextjs-routes.d.ts
vendored
@@ -12,17 +12,18 @@ declare module "nextjs-routes" {
|
|||||||
|
|
||||||
export type Route =
|
export type Route =
|
||||||
| StaticRoute<"/account/signin">
|
| StaticRoute<"/account/signin">
|
||||||
|
| StaticRoute<"/admin/jobs">
|
||||||
| DynamicRoute<"/api/auth/[...nextauth]", { "nextauth": string[] }>
|
| DynamicRoute<"/api/auth/[...nextauth]", { "nextauth": string[] }>
|
||||||
| StaticRoute<"/api/experiments/og-image">
|
| StaticRoute<"/api/experiments/og-image">
|
||||||
| DynamicRoute<"/api/trpc/[trpc]", { "trpc": string }>
|
| DynamicRoute<"/api/trpc/[trpc]", { "trpc": string }>
|
||||||
| DynamicRoute<"/api/v1/[...trpc]", { "trpc": string[] }>
|
| DynamicRoute<"/api/v1/[...trpc]", { "trpc": string[] }>
|
||||||
| StaticRoute<"/api/v1/openapi">
|
| StaticRoute<"/api/v1/openapi">
|
||||||
| StaticRoute<"/dashboard">
|
| StaticRoute<"/dashboard">
|
||||||
| DynamicRoute<"/data/[id]", { "id": string }>
|
| DynamicRoute<"/experiments/[experimentSlug]", { "experimentSlug": string }>
|
||||||
| StaticRoute<"/data">
|
|
||||||
| DynamicRoute<"/experiments/[id]", { "id": string }>
|
|
||||||
| StaticRoute<"/experiments">
|
| StaticRoute<"/experiments">
|
||||||
|
| StaticRoute<"/fine-tunes">
|
||||||
| StaticRoute<"/">
|
| StaticRoute<"/">
|
||||||
|
| DynamicRoute<"/invitations/[invitationToken]", { "invitationToken": string }>
|
||||||
| StaticRoute<"/project/settings">
|
| StaticRoute<"/project/settings">
|
||||||
| StaticRoute<"/request-logs">
|
| StaticRoute<"/request-logs">
|
||||||
| StaticRoute<"/sentry-example-page">
|
| StaticRoute<"/sentry-example-page">
|
||||||
|
|||||||
@@ -23,7 +23,6 @@ ARG NEXT_PUBLIC_SOCKET_URL
|
|||||||
ARG NEXT_PUBLIC_HOST
|
ARG NEXT_PUBLIC_HOST
|
||||||
ARG NEXT_PUBLIC_SENTRY_DSN
|
ARG NEXT_PUBLIC_SENTRY_DSN
|
||||||
ARG SENTRY_AUTH_TOKEN
|
ARG SENTRY_AUTH_TOKEN
|
||||||
ARG NEXT_PUBLIC_FF_SHOW_LOGGED_CALLS
|
|
||||||
|
|
||||||
WORKDIR /code
|
WORKDIR /code
|
||||||
COPY --from=deps /code/node_modules ./node_modules
|
COPY --from=deps /code/node_modules ./node_modules
|
||||||
@@ -45,4 +44,4 @@ EXPOSE 3000
|
|||||||
ENV PORT 3000
|
ENV PORT 3000
|
||||||
|
|
||||||
# Run the "run-prod.sh" script
|
# Run the "run-prod.sh" script
|
||||||
CMD /code/app/run-prod.sh
|
CMD /code/app/scripts/run-prod.sh
|
||||||
@@ -10,14 +10,15 @@
|
|||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "next build",
|
"build": "next build",
|
||||||
"dev:next": "next dev",
|
"dev:next": "TZ=UTC next dev",
|
||||||
"dev:wss": "pnpm tsx --watch src/wss-server.ts",
|
"dev:wss": "pnpm tsx --watch src/wss-server.ts",
|
||||||
"dev:worker": "NODE_ENV='development' pnpm tsx --watch src/server/tasks/worker.ts",
|
"worker": "NODE_ENV='development' pnpm tsx --watch src/server/tasks/worker.ts",
|
||||||
"dev": "concurrently --kill-others 'pnpm dev:next' 'pnpm dev:wss' 'pnpm dev:worker'",
|
"dev": "concurrently --kill-others 'pnpm dev:next' 'pnpm dev:wss' 'pnpm worker --watch'",
|
||||||
"postinstall": "prisma generate",
|
"postinstall": "prisma generate",
|
||||||
"lint": "next lint",
|
"lint": "next lint",
|
||||||
"start": "next start",
|
"start": "TZ=UTC next start",
|
||||||
"codegen:clients": "tsx src/server/scripts/client-codegen.ts",
|
"codegen:clients": "tsx src/server/scripts/client-codegen.ts",
|
||||||
|
"codegen:db": "prisma generate && kysely-codegen --dialect postgres --out-file src/server/db.types.ts",
|
||||||
"seed": "tsx prisma/seed.ts",
|
"seed": "tsx prisma/seed.ts",
|
||||||
"check": "concurrently 'pnpm lint' 'pnpm tsc' 'pnpm prettier . --check'",
|
"check": "concurrently 'pnpm lint' 'pnpm tsc' 'pnpm prettier . --check'",
|
||||||
"test": "pnpm vitest"
|
"test": "pnpm vitest"
|
||||||
@@ -37,6 +38,7 @@
|
|||||||
"@monaco-editor/loader": "^1.3.3",
|
"@monaco-editor/loader": "^1.3.3",
|
||||||
"@next-auth/prisma-adapter": "^1.0.5",
|
"@next-auth/prisma-adapter": "^1.0.5",
|
||||||
"@prisma/client": "^4.14.0",
|
"@prisma/client": "^4.14.0",
|
||||||
|
"@sendinblue/client": "^3.3.1",
|
||||||
"@sentry/nextjs": "^7.61.0",
|
"@sentry/nextjs": "^7.61.0",
|
||||||
"@t3-oss/env-nextjs": "^0.3.1",
|
"@t3-oss/env-nextjs": "^0.3.1",
|
||||||
"@tabler/icons-react": "^2.22.0",
|
"@tabler/icons-react": "^2.22.0",
|
||||||
@@ -46,6 +48,7 @@
|
|||||||
"@trpc/react-query": "^10.26.0",
|
"@trpc/react-query": "^10.26.0",
|
||||||
"@trpc/server": "^10.26.0",
|
"@trpc/server": "^10.26.0",
|
||||||
"@vercel/og": "^0.5.9",
|
"@vercel/og": "^0.5.9",
|
||||||
|
"archiver": "^6.0.0",
|
||||||
"ast-types": "^0.14.2",
|
"ast-types": "^0.14.2",
|
||||||
"chroma-js": "^2.4.2",
|
"chroma-js": "^2.4.2",
|
||||||
"concurrently": "^8.2.0",
|
"concurrently": "^8.2.0",
|
||||||
@@ -58,19 +61,23 @@
|
|||||||
"framer-motion": "^10.12.17",
|
"framer-motion": "^10.12.17",
|
||||||
"gpt-tokens": "^1.0.10",
|
"gpt-tokens": "^1.0.10",
|
||||||
"graphile-worker": "^0.13.0",
|
"graphile-worker": "^0.13.0",
|
||||||
|
"human-id": "^4.0.0",
|
||||||
"immer": "^10.0.2",
|
"immer": "^10.0.2",
|
||||||
"isolated-vm": "^4.5.0",
|
"isolated-vm": "^4.5.0",
|
||||||
"json-schema-to-typescript": "^13.0.2",
|
"json-schema-to-typescript": "^13.0.2",
|
||||||
"json-stringify-pretty-compact": "^4.0.0",
|
"json-stringify-pretty-compact": "^4.0.0",
|
||||||
"jsonschema": "^1.4.1",
|
"jsonschema": "^1.4.1",
|
||||||
"kysely": "^0.26.1",
|
"kysely": "^0.26.1",
|
||||||
|
"kysely-codegen": "^0.10.1",
|
||||||
"lodash-es": "^4.17.21",
|
"lodash-es": "^4.17.21",
|
||||||
"lucide-react": "^0.265.0",
|
"lucide-react": "^0.265.0",
|
||||||
|
"marked": "^7.0.3",
|
||||||
"next": "^13.4.2",
|
"next": "^13.4.2",
|
||||||
"next-auth": "^4.22.1",
|
"next-auth": "^4.22.1",
|
||||||
"next-query-params": "^4.2.3",
|
"next-query-params": "^4.2.3",
|
||||||
"nextjs-cors": "^2.1.2",
|
"nextjs-cors": "^2.1.2",
|
||||||
"nextjs-routes": "^2.0.1",
|
"nextjs-routes": "^2.0.1",
|
||||||
|
"nodemailer": "^6.9.4",
|
||||||
"openai": "4.0.0-beta.7",
|
"openai": "4.0.0-beta.7",
|
||||||
"openpipe": "workspace:*",
|
"openpipe": "workspace:*",
|
||||||
"pg": "^8.11.2",
|
"pg": "^8.11.2",
|
||||||
@@ -93,6 +100,7 @@
|
|||||||
"replicate": "^0.12.3",
|
"replicate": "^0.12.3",
|
||||||
"socket.io": "^4.7.1",
|
"socket.io": "^4.7.1",
|
||||||
"socket.io-client": "^4.7.1",
|
"socket.io-client": "^4.7.1",
|
||||||
|
"stream-buffers": "^3.0.2",
|
||||||
"superjson": "1.12.2",
|
"superjson": "1.12.2",
|
||||||
"trpc-openapi": "^1.2.0",
|
"trpc-openapi": "^1.2.0",
|
||||||
"tsx": "^3.12.7",
|
"tsx": "^3.12.7",
|
||||||
@@ -105,6 +113,7 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@openapi-contrib/openapi-schema-to-json-schema": "^4.0.5",
|
"@openapi-contrib/openapi-schema-to-json-schema": "^4.0.5",
|
||||||
|
"@types/archiver": "^5.3.2",
|
||||||
"@types/babel__core": "^7.20.1",
|
"@types/babel__core": "^7.20.1",
|
||||||
"@types/babel__standalone": "^7.1.4",
|
"@types/babel__standalone": "^7.1.4",
|
||||||
"@types/chroma-js": "^2.4.0",
|
"@types/chroma-js": "^2.4.0",
|
||||||
@@ -114,12 +123,14 @@
|
|||||||
"@types/json-schema": "^7.0.12",
|
"@types/json-schema": "^7.0.12",
|
||||||
"@types/lodash-es": "^4.17.8",
|
"@types/lodash-es": "^4.17.8",
|
||||||
"@types/node": "^18.16.0",
|
"@types/node": "^18.16.0",
|
||||||
|
"@types/nodemailer": "^6.4.9",
|
||||||
"@types/pg": "^8.10.2",
|
"@types/pg": "^8.10.2",
|
||||||
"@types/pluralize": "^0.0.30",
|
"@types/pluralize": "^0.0.30",
|
||||||
"@types/prismjs": "^1.26.0",
|
"@types/prismjs": "^1.26.0",
|
||||||
"@types/react": "^18.2.6",
|
"@types/react": "^18.2.6",
|
||||||
"@types/react-dom": "^18.2.4",
|
"@types/react-dom": "^18.2.4",
|
||||||
"@types/react-syntax-highlighter": "^15.5.7",
|
"@types/react-syntax-highlighter": "^15.5.7",
|
||||||
|
"@types/stream-buffers": "^3.0.4",
|
||||||
"@types/uuid": "^9.0.2",
|
"@types/uuid": "^9.0.2",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.59.6",
|
"@typescript-eslint/eslint-plugin": "^5.59.6",
|
||||||
"@typescript-eslint/parser": "^5.59.6",
|
"@typescript-eslint/parser": "^5.59.6",
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
-- DropIndex
|
||||||
|
DROP INDEX "LoggedCallTag_name_idx";
|
||||||
|
DROP INDEX "LoggedCallTag_name_value_idx";
|
||||||
|
|
||||||
|
-- AlterTable: Add projectId column without NOT NULL constraint for now
|
||||||
|
ALTER TABLE "LoggedCallTag" ADD COLUMN "projectId" UUID;
|
||||||
|
|
||||||
|
-- Set the default value
|
||||||
|
UPDATE "LoggedCallTag" lct
|
||||||
|
SET "projectId" = lc."projectId"
|
||||||
|
FROM "LoggedCall" lc
|
||||||
|
WHERE lct."loggedCallId" = lc.id;
|
||||||
|
|
||||||
|
-- Now set the NOT NULL constraint
|
||||||
|
ALTER TABLE "LoggedCallTag" ALTER COLUMN "projectId" SET NOT NULL;
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "LoggedCallTag_projectId_name_idx" ON "LoggedCallTag"("projectId", "name");
|
||||||
|
CREATE INDEX "LoggedCallTag_projectId_name_value_idx" ON "LoggedCallTag"("projectId", "name", "value");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "LoggedCallTag_loggedCallId_name_key" ON "LoggedCallTag"("loggedCallId", "name");
|
||||||
@@ -0,0 +1,25 @@
|
|||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "UserInvitation" (
|
||||||
|
"id" UUID NOT NULL,
|
||||||
|
"projectId" UUID NOT NULL,
|
||||||
|
"email" TEXT NOT NULL,
|
||||||
|
"role" "ProjectUserRole" NOT NULL,
|
||||||
|
"invitationToken" TEXT NOT NULL,
|
||||||
|
"senderId" UUID NOT NULL,
|
||||||
|
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "UserInvitation_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "UserInvitation_invitationToken_key" ON "UserInvitation"("invitationToken");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "UserInvitation_projectId_email_key" ON "UserInvitation"("projectId", "email");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "UserInvitation" ADD CONSTRAINT "UserInvitation_projectId_fkey" FOREIGN KEY ("projectId") REFERENCES "Project"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "UserInvitation" ADD CONSTRAINT "UserInvitation_senderId_fkey" FOREIGN KEY ("senderId") REFERENCES "User"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2023 Viascom Ltd liab. Co
|
||||||
|
*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. The ASF licenses this file
|
||||||
|
* to you under the Apache License, Version 2.0 (the
|
||||||
|
* "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing,
|
||||||
|
* software distributed under the License is distributed on an
|
||||||
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
||||||
|
* KIND, either express or implied. See the License for the
|
||||||
|
* specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
CREATE EXTENSION IF NOT EXISTS pgcrypto;
|
||||||
|
|
||||||
|
CREATE OR REPLACE FUNCTION nanoid(
|
||||||
|
size int DEFAULT 21,
|
||||||
|
alphabet text DEFAULT '_-0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'
|
||||||
|
)
|
||||||
|
RETURNS text
|
||||||
|
LANGUAGE plpgsql
|
||||||
|
volatile
|
||||||
|
AS
|
||||||
|
$$
|
||||||
|
DECLARE
|
||||||
|
idBuilder text := '';
|
||||||
|
counter int := 0;
|
||||||
|
bytes bytea;
|
||||||
|
alphabetIndex int;
|
||||||
|
alphabetArray text[];
|
||||||
|
alphabetLength int;
|
||||||
|
mask int;
|
||||||
|
step int;
|
||||||
|
BEGIN
|
||||||
|
alphabetArray := regexp_split_to_array(alphabet, '');
|
||||||
|
alphabetLength := array_length(alphabetArray, 1);
|
||||||
|
mask := (2 << cast(floor(log(alphabetLength - 1) / log(2)) as int)) - 1;
|
||||||
|
step := cast(ceil(1.6 * mask * size / alphabetLength) AS int);
|
||||||
|
|
||||||
|
while true
|
||||||
|
loop
|
||||||
|
bytes := gen_random_bytes(step);
|
||||||
|
while counter < step
|
||||||
|
loop
|
||||||
|
alphabetIndex := (get_byte(bytes, counter) & mask) + 1;
|
||||||
|
if alphabetIndex <= alphabetLength then
|
||||||
|
idBuilder := idBuilder || alphabetArray[alphabetIndex];
|
||||||
|
if length(idBuilder) = size then
|
||||||
|
return idBuilder;
|
||||||
|
end if;
|
||||||
|
end if;
|
||||||
|
counter := counter + 1;
|
||||||
|
end loop;
|
||||||
|
|
||||||
|
counter := 0;
|
||||||
|
end loop;
|
||||||
|
END
|
||||||
|
$$;
|
||||||
|
|
||||||
|
|
||||||
|
-- Make a short_nanoid function that uses the default alphabet and length of 15
|
||||||
|
CREATE OR REPLACE FUNCTION short_nanoid()
|
||||||
|
RETURNS text
|
||||||
|
LANGUAGE plpgsql
|
||||||
|
volatile
|
||||||
|
AS
|
||||||
|
$$
|
||||||
|
BEGIN
|
||||||
|
RETURN nanoid(15, '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ');
|
||||||
|
END
|
||||||
|
$$;
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "Experiment" ADD COLUMN "slug" TEXT NOT NULL DEFAULT short_nanoid();
|
||||||
|
|
||||||
|
-- For existing experiments, keep the existing id as the slug for backwards compatibility
|
||||||
|
UPDATE "Experiment" SET "slug" = "id";
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "Experiment_slug_key" ON "Experiment"("slug");
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
/*
|
||||||
|
Warnings:
|
||||||
|
|
||||||
|
- You are about to drop the column `input` on the `DatasetEntry` table. All the data in the column will be lost.
|
||||||
|
- You are about to drop the column `output` on the `DatasetEntry` table. All the data in the column will be lost.
|
||||||
|
- Added the required column `loggedCallId` to the `DatasetEntry` table without a default value. This is not possible if the table is not empty.
|
||||||
|
|
||||||
|
*/
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "DatasetEntry" DROP COLUMN "input",
|
||||||
|
DROP COLUMN "output",
|
||||||
|
ADD COLUMN "loggedCallId" UUID NOT NULL;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "DatasetEntry" ADD CONSTRAINT "DatasetEntry_loggedCallId_fkey" FOREIGN KEY ("loggedCallId") REFERENCES "LoggedCall"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "LoggedCallModelResponse" ALTER COLUMN "cost" SET DATA TYPE DOUBLE PRECISION;
|
||||||
|
|
||||||
|
-- CreateEnum
|
||||||
|
CREATE TYPE "FineTuneStatus" AS ENUM ('PENDING', 'TRAINING', 'AWAITING_DEPLOYMENT', 'DEPLOYING', 'DEPLOYED', 'ERROR');
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "FineTune" (
|
||||||
|
"id" UUID NOT NULL,
|
||||||
|
"slug" TEXT NOT NULL,
|
||||||
|
"baseModel" TEXT NOT NULL,
|
||||||
|
"status" "FineTuneStatus" NOT NULL DEFAULT 'PENDING',
|
||||||
|
"trainingStartedAt" TIMESTAMP(3),
|
||||||
|
"trainingFinishedAt" TIMESTAMP(3),
|
||||||
|
"deploymentStartedAt" TIMESTAMP(3),
|
||||||
|
"deploymentFinishedAt" TIMESTAMP(3),
|
||||||
|
"datasetId" UUID NOT NULL,
|
||||||
|
"projectId" UUID NOT NULL,
|
||||||
|
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "FineTune_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE UNIQUE INDEX "FineTune_slug_key" ON "FineTune"("slug");
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "FineTune" ADD CONSTRAINT "FineTune_datasetId_fkey" FOREIGN KEY ("datasetId") REFERENCES "Dataset"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "FineTune" ADD CONSTRAINT "FineTune_projectId_fkey" FOREIGN KEY ("projectId") REFERENCES "Project"("id") ON DELETE CASCADE ON UPDATE CASCADE;
|
||||||
@@ -11,7 +11,9 @@ datasource db {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model Experiment {
|
model Experiment {
|
||||||
id String @id @default(uuid()) @db.Uuid
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
|
slug String @unique @default(dbgenerated("short_nanoid()"))
|
||||||
label String
|
label String
|
||||||
|
|
||||||
sortIndex Int @default(0)
|
sortIndex Int @default(0)
|
||||||
@@ -179,6 +181,7 @@ model Dataset {
|
|||||||
|
|
||||||
name String
|
name String
|
||||||
datasetEntries DatasetEntry[]
|
datasetEntries DatasetEntry[]
|
||||||
|
fineTunes FineTune[]
|
||||||
|
|
||||||
projectId String @db.Uuid
|
projectId String @db.Uuid
|
||||||
project Project @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
project Project @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
||||||
@@ -190,8 +193,8 @@ model Dataset {
|
|||||||
model DatasetEntry {
|
model DatasetEntry {
|
||||||
id String @id @default(uuid()) @db.Uuid
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
input String
|
loggedCallId String @db.Uuid
|
||||||
output String?
|
loggedCall LoggedCall @relation(fields: [loggedCallId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
datasetId String @db.Uuid
|
datasetId String @db.Uuid
|
||||||
dataset Dataset? @relation(fields: [datasetId], references: [id], onDelete: Cascade)
|
dataset Dataset? @relation(fields: [datasetId], references: [id], onDelete: Cascade)
|
||||||
@@ -207,13 +210,15 @@ model Project {
|
|||||||
personalProjectUserId String? @unique @db.Uuid
|
personalProjectUserId String? @unique @db.Uuid
|
||||||
personalProjectUser User? @relation(fields: [personalProjectUserId], references: [id], onDelete: Cascade)
|
personalProjectUser User? @relation(fields: [personalProjectUserId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
projectUsers ProjectUser[]
|
projectUsers ProjectUser[]
|
||||||
experiments Experiment[]
|
projectUserInvitations UserInvitation[]
|
||||||
datasets Dataset[]
|
experiments Experiment[]
|
||||||
loggedCalls LoggedCall[]
|
datasets Dataset[]
|
||||||
apiKeys ApiKey[]
|
loggedCalls LoggedCall[]
|
||||||
|
fineTunes FineTune[]
|
||||||
|
apiKeys ApiKey[]
|
||||||
}
|
}
|
||||||
|
|
||||||
enum ProjectUserRole {
|
enum ProjectUserRole {
|
||||||
@@ -273,8 +278,9 @@ model LoggedCall {
|
|||||||
projectId String @db.Uuid
|
projectId String @db.Uuid
|
||||||
project Project? @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
project Project? @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
model String?
|
model String?
|
||||||
tags LoggedCallTag[]
|
tags LoggedCallTag[]
|
||||||
|
datasetEntries DatasetEntry[]
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
@@ -309,7 +315,7 @@ model LoggedCallModelResponse {
|
|||||||
outputTokens Int?
|
outputTokens Int?
|
||||||
finishReason String?
|
finishReason String?
|
||||||
completionId String?
|
completionId String?
|
||||||
cost Decimal? @db.Decimal(18, 12)
|
cost Float?
|
||||||
|
|
||||||
// The LoggedCall that created this LoggedCallModelResponse
|
// The LoggedCall that created this LoggedCallModelResponse
|
||||||
originalLoggedCallId String @unique @db.Uuid
|
originalLoggedCallId String @unique @db.Uuid
|
||||||
@@ -323,15 +329,17 @@ model LoggedCallModelResponse {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model LoggedCallTag {
|
model LoggedCallTag {
|
||||||
id String @id @default(uuid()) @db.Uuid
|
id String @id @default(uuid()) @db.Uuid
|
||||||
name String
|
name String
|
||||||
value String?
|
value String?
|
||||||
|
projectId String @db.Uuid
|
||||||
|
|
||||||
loggedCallId String @db.Uuid
|
loggedCallId String @db.Uuid
|
||||||
loggedCall LoggedCall @relation(fields: [loggedCallId], references: [id], onDelete: Cascade)
|
loggedCall LoggedCall @relation(fields: [loggedCallId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
@@index([name])
|
@@unique([loggedCallId, name])
|
||||||
@@index([name, value])
|
@@index([projectId, name])
|
||||||
|
@@index([projectId, name, value])
|
||||||
}
|
}
|
||||||
|
|
||||||
model ApiKey {
|
model ApiKey {
|
||||||
@@ -388,16 +396,33 @@ model User {
|
|||||||
|
|
||||||
role UserRole @default(USER)
|
role UserRole @default(USER)
|
||||||
|
|
||||||
accounts Account[]
|
accounts Account[]
|
||||||
sessions Session[]
|
sessions Session[]
|
||||||
projectUsers ProjectUser[]
|
projectUsers ProjectUser[]
|
||||||
projects Project[]
|
projects Project[]
|
||||||
worldChampEntrant WorldChampEntrant?
|
worldChampEntrant WorldChampEntrant?
|
||||||
|
sentUserInvitations UserInvitation[]
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @default(now()) @updatedAt
|
updatedAt DateTime @default(now()) @updatedAt
|
||||||
}
|
}
|
||||||
|
|
||||||
|
model UserInvitation {
|
||||||
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
|
projectId String @db.Uuid
|
||||||
|
project Project @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
||||||
|
email String
|
||||||
|
role ProjectUserRole
|
||||||
|
invitationToken String @unique
|
||||||
|
senderId String @db.Uuid
|
||||||
|
sender User @relation(fields: [senderId], references: [id], onDelete: Cascade)
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
@@unique([projectId, email])
|
||||||
|
}
|
||||||
|
|
||||||
model VerificationToken {
|
model VerificationToken {
|
||||||
identifier String
|
identifier String
|
||||||
token String @unique
|
token String @unique
|
||||||
@@ -405,3 +430,33 @@ model VerificationToken {
|
|||||||
|
|
||||||
@@unique([identifier, token])
|
@@unique([identifier, token])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum FineTuneStatus {
|
||||||
|
PENDING
|
||||||
|
TRAINING
|
||||||
|
AWAITING_DEPLOYMENT
|
||||||
|
DEPLOYING
|
||||||
|
DEPLOYED
|
||||||
|
ERROR
|
||||||
|
}
|
||||||
|
|
||||||
|
model FineTune {
|
||||||
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
|
slug String @unique
|
||||||
|
baseModel String
|
||||||
|
status FineTuneStatus @default(PENDING)
|
||||||
|
trainingStartedAt DateTime?
|
||||||
|
trainingFinishedAt DateTime?
|
||||||
|
deploymentStartedAt DateTime?
|
||||||
|
deploymentFinishedAt DateTime?
|
||||||
|
|
||||||
|
datasetId String @db.Uuid
|
||||||
|
dataset Dataset @relation(fields: [datasetId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
projectId String @db.Uuid
|
||||||
|
project Project @relation(fields: [projectId], references: [id], onDelete: Cascade)
|
||||||
|
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
}
|
||||||
|
|||||||
@@ -10,6 +10,14 @@ await prisma.project.deleteMany({
|
|||||||
where: { id: defaultId },
|
where: { id: defaultId },
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Mark all users as admins
|
||||||
|
await prisma.user.updateMany({
|
||||||
|
where: {},
|
||||||
|
data: {
|
||||||
|
role: "ADMIN",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
// If there's an existing project, just seed into it
|
// If there's an existing project, just seed into it
|
||||||
const project =
|
const project =
|
||||||
(await prisma.project.findFirst({})) ??
|
(await prisma.project.findFirst({})) ??
|
||||||
@@ -18,12 +26,16 @@ const project =
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
if (env.OPENPIPE_API_KEY) {
|
if (env.OPENPIPE_API_KEY) {
|
||||||
await prisma.apiKey.create({
|
await prisma.apiKey.upsert({
|
||||||
data: {
|
where: {
|
||||||
|
apiKey: env.OPENPIPE_API_KEY,
|
||||||
|
},
|
||||||
|
create: {
|
||||||
projectId: project.id,
|
projectId: project.id,
|
||||||
name: "Default API Key",
|
name: "Default API Key",
|
||||||
apiKey: env.OPENPIPE_API_KEY,
|
apiKey: env.OPENPIPE_API_KEY,
|
||||||
},
|
},
|
||||||
|
update: {},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: number;
|
inputTokens: number;
|
||||||
outputTokens: number;
|
outputTokens: number;
|
||||||
finishReason: string;
|
finishReason: string;
|
||||||
|
tags: { name: string; value: string }[];
|
||||||
}[] = [
|
}[] = [
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -107,6 +108,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: 236,
|
inputTokens: 236,
|
||||||
outputTokens: 5,
|
outputTokens: 5,
|
||||||
finishReason: "stop",
|
finishReason: "stop",
|
||||||
|
tags: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -193,6 +195,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: 222,
|
inputTokens: 222,
|
||||||
outputTokens: 5,
|
outputTokens: 5,
|
||||||
finishReason: "stop",
|
finishReason: "stop",
|
||||||
|
tags: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -231,6 +234,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: 14,
|
inputTokens: 14,
|
||||||
outputTokens: 7,
|
outputTokens: 7,
|
||||||
finishReason: "stop",
|
finishReason: "stop",
|
||||||
|
tags: [{ name: "prompt_id", value: "id2" }],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -306,6 +310,10 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: 2802,
|
inputTokens: 2802,
|
||||||
outputTokens: 108,
|
outputTokens: 108,
|
||||||
finishReason: "stop",
|
finishReason: "stop",
|
||||||
|
tags: [
|
||||||
|
{ name: "prompt_id", value: "chatcmpl-7lQS3MktOT8BTgNEytl9dkyssCQqL" },
|
||||||
|
{ name: "some_other_tag", value: "some_other_value" },
|
||||||
|
],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -349,6 +357,7 @@ for (let i = 0; i < 1437; i++) {
|
|||||||
cacheHit: false,
|
cacheHit: false,
|
||||||
requestedAt,
|
requestedAt,
|
||||||
projectId: project.id,
|
projectId: project.id,
|
||||||
|
model: template.reqPayload.model,
|
||||||
createdAt: requestedAt,
|
createdAt: requestedAt,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -388,11 +397,14 @@ for (let i = 0; i < 1437; i++) {
|
|||||||
modelResponseId: loggedCallModelResponseId,
|
modelResponseId: loggedCallModelResponseId,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
loggedCallTagsToCreate.push({
|
for (const tag of template.tags) {
|
||||||
loggedCallId,
|
loggedCallTagsToCreate.push({
|
||||||
name: "$model",
|
projectId: project.id,
|
||||||
value: template.reqPayload.model,
|
loggedCallId,
|
||||||
});
|
name: tag.name,
|
||||||
|
value: tag.value,
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await prisma.$transaction([
|
await prisma.$transaction([
|
||||||
|
|||||||
6
app/scripts/debug-prod.sh
Normal file
6
app/scripts/debug-prod.sh
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
#! /bin/bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
cd "$(dirname "$0")/.."
|
||||||
|
apt-get update
|
||||||
|
apt-get install -y htop psql
|
||||||
@@ -10,6 +10,4 @@ pnpm tsx src/promptConstructor/migrate.ts
|
|||||||
|
|
||||||
echo "Starting the server"
|
echo "Starting the server"
|
||||||
|
|
||||||
pnpm concurrently --kill-others \
|
pnpm start
|
||||||
"pnpm start" \
|
|
||||||
"pnpm tsx src/server/tasks/worker.ts"
|
|
||||||
10
app/scripts/run-workers-prod.sh
Executable file
10
app/scripts/run-workers-prod.sh
Executable file
@@ -0,0 +1,10 @@
|
|||||||
|
#! /bin/bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
echo "Migrating the database"
|
||||||
|
pnpm prisma migrate deploy
|
||||||
|
|
||||||
|
echo "Starting 4 workers"
|
||||||
|
|
||||||
|
pnpm concurrently "pnpm worker" "pnpm worker" "pnpm worker" "pnpm worker"
|
||||||
13
app/scripts/test-docker.sh
Executable file
13
app/scripts/test-docker.sh
Executable file
@@ -0,0 +1,13 @@
|
|||||||
|
#! /bin/bash
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
cd "$(dirname "$0")/../.."
|
||||||
|
|
||||||
|
echo "Env is"
|
||||||
|
echo $ENVIRONMENT
|
||||||
|
|
||||||
|
docker build . --file app/Dockerfile --tag "openpipe-prod"
|
||||||
|
|
||||||
|
# Run the image
|
||||||
|
docker run --env-file app/.env -it --entrypoint "/bin/bash" "openpipe-prod"
|
||||||
@@ -3,6 +3,7 @@
|
|||||||
// https://docs.sentry.io/platforms/javascript/guides/nextjs/
|
// https://docs.sentry.io/platforms/javascript/guides/nextjs/
|
||||||
|
|
||||||
import * as Sentry from "@sentry/nextjs";
|
import * as Sentry from "@sentry/nextjs";
|
||||||
|
import { isError } from "lodash-es";
|
||||||
import { env } from "~/env.mjs";
|
import { env } from "~/env.mjs";
|
||||||
|
|
||||||
if (env.NEXT_PUBLIC_SENTRY_DSN) {
|
if (env.NEXT_PUBLIC_SENTRY_DSN) {
|
||||||
@@ -15,4 +16,10 @@ if (env.NEXT_PUBLIC_SENTRY_DSN) {
|
|||||||
// Setting this option to true will print useful information to the console while you're setting up Sentry.
|
// Setting this option to true will print useful information to the console while you're setting up Sentry.
|
||||||
debug: false,
|
debug: false,
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
// Install local debug exception handler for rejected promises
|
||||||
|
process.on("unhandledRejection", (reason) => {
|
||||||
|
const reasonDetails = isError(reason) ? reason?.stack : reason;
|
||||||
|
console.log("Unhandled Rejection at:", reasonDetails);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
import { Textarea, type TextareaProps } from "@chakra-ui/react";
|
import { Textarea, type TextareaProps } from "@chakra-ui/react";
|
||||||
import ResizeTextarea from "react-textarea-autosize";
|
import ResizeTextarea from "react-textarea-autosize";
|
||||||
import React, { useLayoutEffect, useState } from "react";
|
import React, { useEffect, useState } from "react";
|
||||||
|
|
||||||
export const AutoResizeTextarea: React.ForwardRefRenderFunction<
|
export const AutoResizeTextarea: React.ForwardRefRenderFunction<
|
||||||
HTMLTextAreaElement,
|
HTMLTextAreaElement,
|
||||||
TextareaProps & { minRows?: number }
|
TextareaProps & { minRows?: number }
|
||||||
> = ({ minRows = 1, overflowY = "hidden", ...props }, ref) => {
|
> = ({ minRows = 1, overflowY = "hidden", ...props }, ref) => {
|
||||||
const [isRerendered, setIsRerendered] = useState(false);
|
const [isRerendered, setIsRerendered] = useState(false);
|
||||||
useLayoutEffect(() => setIsRerendered(true), []);
|
useEffect(() => setIsRerendered(true), []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Textarea
|
<Textarea
|
||||||
|
|||||||
@@ -87,7 +87,7 @@ export const ModelStatsCard = ({
|
|||||||
label="Price"
|
label="Price"
|
||||||
info={
|
info={
|
||||||
<Text>
|
<Text>
|
||||||
${model.pricePerSecond.toFixed(3)}
|
${model.pricePerSecond.toFixed(4)}
|
||||||
<Text color="gray.500"> / second</Text>
|
<Text color="gray.500"> / second</Text>
|
||||||
</Text>
|
</Text>
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,15 +1,16 @@
|
|||||||
import { HStack, Icon, IconButton, Tooltip, Text } from "@chakra-ui/react";
|
import { HStack, Icon, IconButton, Tooltip, Text, type StackProps } from "@chakra-ui/react";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { MdContentCopy } from "react-icons/md";
|
import { MdContentCopy } from "react-icons/md";
|
||||||
import { useHandledAsyncCallback } from "~/utils/hooks";
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
|
||||||
const CopiableCode = ({ code }: { code: string }) => {
|
const CopiableCode = ({ code, ...rest }: { code: string } & StackProps) => {
|
||||||
const [copied, setCopied] = useState(false);
|
const [copied, setCopied] = useState(false);
|
||||||
|
|
||||||
const [copyToClipboard] = useHandledAsyncCallback(async () => {
|
const [copyToClipboard] = useHandledAsyncCallback(async () => {
|
||||||
await navigator.clipboard.writeText(code);
|
await navigator.clipboard.writeText(code);
|
||||||
setCopied(true);
|
setCopied(true);
|
||||||
}, [code]);
|
}, [code]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HStack
|
<HStack
|
||||||
backgroundColor="blackAlpha.800"
|
backgroundColor="blackAlpha.800"
|
||||||
@@ -18,9 +19,19 @@ const CopiableCode = ({ code }: { code: string }) => {
|
|||||||
padding={3}
|
padding={3}
|
||||||
w="full"
|
w="full"
|
||||||
justifyContent="space-between"
|
justifyContent="space-between"
|
||||||
|
alignItems="flex-start"
|
||||||
|
{...rest}
|
||||||
>
|
>
|
||||||
<Text fontFamily="inconsolata" fontWeight="bold" letterSpacing={0.5} overflowX="auto">
|
<Text
|
||||||
|
fontFamily="inconsolata"
|
||||||
|
fontWeight="bold"
|
||||||
|
letterSpacing={0.5}
|
||||||
|
overflowX="auto"
|
||||||
|
whiteSpace="pre-wrap"
|
||||||
|
>
|
||||||
{code}
|
{code}
|
||||||
|
{/* Necessary for trailing newline to actually be displayed */}
|
||||||
|
{code.endsWith("\n") ? "\n" : ""}
|
||||||
</Text>
|
</Text>
|
||||||
<Tooltip closeOnClick={false} label={copied ? "Copied!" : "Copy to clipboard"}>
|
<Tooltip closeOnClick={false} label={copied ? "Copied!" : "Copy to clipboard"}>
|
||||||
<IconButton
|
<IconButton
|
||||||
|
|||||||
14
app/src/components/InfoCircle.tsx
Normal file
14
app/src/components/InfoCircle.tsx
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { Tooltip, Icon, VStack } from "@chakra-ui/react";
|
||||||
|
import { RiInformationFill } from "react-icons/ri";
|
||||||
|
|
||||||
|
const InfoCircle = ({ tooltipText }: { tooltipText: string }) => {
|
||||||
|
return (
|
||||||
|
<Tooltip label={tooltipText} fontSize="sm" shouldWrapChildren maxW={80}>
|
||||||
|
<VStack>
|
||||||
|
<Icon as={RiInformationFill} boxSize={5} color="gray.500" />
|
||||||
|
</VStack>
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default InfoCircle;
|
||||||
91
app/src/components/InputDropdown.tsx
Normal file
91
app/src/components/InputDropdown.tsx
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
import {
|
||||||
|
Input,
|
||||||
|
InputGroup,
|
||||||
|
InputRightElement,
|
||||||
|
Icon,
|
||||||
|
Popover,
|
||||||
|
PopoverTrigger,
|
||||||
|
PopoverContent,
|
||||||
|
VStack,
|
||||||
|
HStack,
|
||||||
|
Button,
|
||||||
|
Text,
|
||||||
|
useDisclosure,
|
||||||
|
type InputGroupProps,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
|
||||||
|
import { FiChevronDown } from "react-icons/fi";
|
||||||
|
import { BiCheck } from "react-icons/bi";
|
||||||
|
|
||||||
|
type InputDropdownProps<T> = {
|
||||||
|
options: ReadonlyArray<T>;
|
||||||
|
selectedOption: T;
|
||||||
|
onSelect: (option: T) => void;
|
||||||
|
inputGroupProps?: InputGroupProps;
|
||||||
|
};
|
||||||
|
|
||||||
|
const InputDropdown = <T,>({
|
||||||
|
options,
|
||||||
|
selectedOption,
|
||||||
|
onSelect,
|
||||||
|
inputGroupProps,
|
||||||
|
}: InputDropdownProps<T>) => {
|
||||||
|
const popover = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Popover placement="bottom-start" {...popover}>
|
||||||
|
<PopoverTrigger>
|
||||||
|
<InputGroup
|
||||||
|
cursor="pointer"
|
||||||
|
w={(selectedOption as string).length * 14 + 180}
|
||||||
|
{...inputGroupProps}
|
||||||
|
>
|
||||||
|
<Input
|
||||||
|
value={selectedOption as string}
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-empty-function -- controlled input requires onChange
|
||||||
|
onChange={() => {}}
|
||||||
|
cursor="pointer"
|
||||||
|
borderColor={popover.isOpen ? "blue.500" : undefined}
|
||||||
|
_hover={popover.isOpen ? { borderColor: "blue.500" } : undefined}
|
||||||
|
contentEditable={false}
|
||||||
|
// disable focus
|
||||||
|
onFocus={(e) => {
|
||||||
|
e.target.blur();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<InputRightElement>
|
||||||
|
<Icon as={FiChevronDown} />
|
||||||
|
</InputRightElement>
|
||||||
|
</InputGroup>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent boxShadow="0 0 40px 4px rgba(0, 0, 0, 0.1);" minW={0} w="auto">
|
||||||
|
<VStack spacing={0}>
|
||||||
|
{options?.map((option, index) => (
|
||||||
|
<HStack
|
||||||
|
key={index}
|
||||||
|
as={Button}
|
||||||
|
onClick={() => {
|
||||||
|
onSelect(option);
|
||||||
|
popover.onClose();
|
||||||
|
}}
|
||||||
|
w="full"
|
||||||
|
variant="ghost"
|
||||||
|
justifyContent="space-between"
|
||||||
|
fontWeight="semibold"
|
||||||
|
borderRadius={0}
|
||||||
|
colorScheme="blue"
|
||||||
|
color="black"
|
||||||
|
fontSize="sm"
|
||||||
|
borderBottomWidth={1}
|
||||||
|
>
|
||||||
|
<Text mr={16}>{option as string}</Text>
|
||||||
|
{option === selectedOption && <Icon as={BiCheck} color="blue.500" boxSize={5} />}
|
||||||
|
</HStack>
|
||||||
|
))}
|
||||||
|
</VStack>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default InputDropdown;
|
||||||
@@ -8,7 +8,7 @@ import {
|
|||||||
useHandledAsyncCallback,
|
useHandledAsyncCallback,
|
||||||
useVisibleScenarioIds,
|
useVisibleScenarioIds,
|
||||||
} from "~/utils/hooks";
|
} from "~/utils/hooks";
|
||||||
import { cellPadding } from "../constants";
|
import { cellPadding } from "./constants";
|
||||||
import { ActionButton } from "./ScenariosHeader";
|
import { ActionButton } from "./ScenariosHeader";
|
||||||
|
|
||||||
export default function AddVariantButton() {
|
export default function AddVariantButton() {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { HStack, Icon, IconButton, Spinner, Tooltip, useDisclosure } from "@chakra-ui/react";
|
import { HStack, Icon, IconButton, Spinner, Tooltip, useDisclosure } from "@chakra-ui/react";
|
||||||
import { BsArrowClockwise, BsInfoCircle } from "react-icons/bs";
|
import { BsArrowClockwise, BsInfoCircle } from "react-icons/bs";
|
||||||
import { useExperimentAccess } from "~/utils/hooks";
|
import { useExperimentAccess } from "~/utils/hooks";
|
||||||
import ExpandedModal from "./PromptModal";
|
import PromptModal from "./PromptModal";
|
||||||
import { type RouterOutputs } from "~/utils/api";
|
import { type RouterOutputs } from "~/utils/api";
|
||||||
|
|
||||||
export const CellOptions = ({
|
export const CellOptions = ({
|
||||||
@@ -32,7 +32,7 @@ export const CellOptions = ({
|
|||||||
variant="ghost"
|
variant="ghost"
|
||||||
/>
|
/>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
<ExpandedModal cell={cell} disclosure={modalDisclosure} />
|
<PromptModal cell={cell} disclosure={modalDisclosure} />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{canModify && (
|
{canModify && (
|
||||||
29
app/src/components/OutputsTable/OutputCell/CellWrapper.tsx
Normal file
29
app/src/components/OutputsTable/OutputCell/CellWrapper.tsx
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
import { type StackProps, VStack } from "@chakra-ui/react";
|
||||||
|
import { type RouterOutputs } from "~/utils/api";
|
||||||
|
import { type Scenario } from "../types";
|
||||||
|
import { CellOptions } from "./CellOptions";
|
||||||
|
import { OutputStats } from "./OutputStats";
|
||||||
|
|
||||||
|
const CellWrapper: React.FC<
|
||||||
|
StackProps & {
|
||||||
|
cell: RouterOutputs["scenarioVariantCells"]["get"] | undefined;
|
||||||
|
hardRefetching: boolean;
|
||||||
|
hardRefetch: () => void;
|
||||||
|
mostRecentResponse:
|
||||||
|
| NonNullable<RouterOutputs["scenarioVariantCells"]["get"]>["modelResponses"][0]
|
||||||
|
| undefined;
|
||||||
|
scenario: Scenario;
|
||||||
|
}
|
||||||
|
> = ({ children, cell, hardRefetching, hardRefetch, mostRecentResponse, scenario, ...props }) => (
|
||||||
|
<VStack w="full" alignItems="flex-start" {...props} px={2} py={2} h="100%">
|
||||||
|
{cell && (
|
||||||
|
<CellOptions refetchingOutput={hardRefetching} refetchOutput={hardRefetch} cell={cell} />
|
||||||
|
)}
|
||||||
|
<VStack w="full" alignItems="flex-start" maxH={500} overflowY="auto" flex={1}>
|
||||||
|
{children}
|
||||||
|
</VStack>
|
||||||
|
{mostRecentResponse && <OutputStats modelResponse={mostRecentResponse} scenario={scenario} />}
|
||||||
|
</VStack>
|
||||||
|
);
|
||||||
|
|
||||||
|
export default CellWrapper;
|
||||||
@@ -1,17 +1,16 @@
|
|||||||
import { api } from "~/utils/api";
|
import { Text } from "@chakra-ui/react";
|
||||||
import { type PromptVariant, type Scenario } from "../types";
|
import stringify from "json-stringify-pretty-compact";
|
||||||
import { type StackProps, Text, VStack } from "@chakra-ui/react";
|
import { Fragment, useEffect, useState, type ReactElement } from "react";
|
||||||
import { useScenarioVars, useHandledAsyncCallback } from "~/utils/hooks";
|
|
||||||
import SyntaxHighlighter from "react-syntax-highlighter";
|
import SyntaxHighlighter from "react-syntax-highlighter";
|
||||||
import { docco } from "react-syntax-highlighter/dist/cjs/styles/hljs";
|
import { docco } from "react-syntax-highlighter/dist/cjs/styles/hljs";
|
||||||
import stringify from "json-stringify-pretty-compact";
|
|
||||||
import { type ReactElement, useState, useEffect, Fragment, useCallback } from "react";
|
|
||||||
import useSocket from "~/utils/useSocket";
|
|
||||||
import { OutputStats } from "./OutputStats";
|
|
||||||
import { RetryCountdown } from "./RetryCountdown";
|
|
||||||
import frontendModelProviders from "~/modelProviders/frontendModelProviders";
|
import frontendModelProviders from "~/modelProviders/frontendModelProviders";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useHandledAsyncCallback, useScenarioVars } from "~/utils/hooks";
|
||||||
|
import useSocket from "~/utils/useSocket";
|
||||||
|
import { type PromptVariant, type Scenario } from "../types";
|
||||||
|
import CellWrapper from "./CellWrapper";
|
||||||
import { ResponseLog } from "./ResponseLog";
|
import { ResponseLog } from "./ResponseLog";
|
||||||
import { CellOptions } from "./TopActions";
|
import { RetryCountdown } from "./RetryCountdown";
|
||||||
|
|
||||||
const WAITING_MESSAGE_INTERVAL = 20000;
|
const WAITING_MESSAGE_INTERVAL = 20000;
|
||||||
|
|
||||||
@@ -44,7 +43,7 @@ export default function OutputCell({
|
|||||||
|
|
||||||
type OutputSchema = Parameters<typeof provider.normalizeOutput>[0];
|
type OutputSchema = Parameters<typeof provider.normalizeOutput>[0];
|
||||||
|
|
||||||
const { mutateAsync: hardRefetchMutate } = api.scenarioVariantCells.forceRefetch.useMutation();
|
const { mutateAsync: hardRefetchMutate } = api.scenarioVariantCells.hardRefetch.useMutation();
|
||||||
const [hardRefetch, hardRefetching] = useHandledAsyncCallback(async () => {
|
const [hardRefetch, hardRefetching] = useHandledAsyncCallback(async () => {
|
||||||
await hardRefetchMutate({ scenarioId: scenario.id, variantId: variant.id });
|
await hardRefetchMutate({ scenarioId: scenario.id, variantId: variant.id });
|
||||||
await utils.scenarioVariantCells.get.invalidate({
|
await utils.scenarioVariantCells.get.invalidate({
|
||||||
@@ -72,35 +71,26 @@ export default function OutputCell({
|
|||||||
|
|
||||||
const mostRecentResponse = cell?.modelResponses[cell.modelResponses.length - 1];
|
const mostRecentResponse = cell?.modelResponses[cell.modelResponses.length - 1];
|
||||||
|
|
||||||
const CellWrapper = useCallback(
|
const wrapperProps: Parameters<typeof CellWrapper>[0] = {
|
||||||
({ children, ...props }: StackProps) => (
|
cell,
|
||||||
<VStack w="full" alignItems="flex-start" {...props} px={2} py={2} h="100%">
|
hardRefetching,
|
||||||
{cell && (
|
hardRefetch,
|
||||||
<CellOptions refetchingOutput={hardRefetching} refetchOutput={hardRefetch} cell={cell} />
|
mostRecentResponse,
|
||||||
)}
|
scenario,
|
||||||
<VStack w="full" alignItems="flex-start" maxH={500} overflowY="auto" flex={1}>
|
};
|
||||||
{children}
|
|
||||||
</VStack>
|
|
||||||
{mostRecentResponse && (
|
|
||||||
<OutputStats modelResponse={mostRecentResponse} scenario={scenario} />
|
|
||||||
)}
|
|
||||||
</VStack>
|
|
||||||
),
|
|
||||||
[hardRefetching, hardRefetch, mostRecentResponse, scenario, cell],
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!vars) return null;
|
if (!vars) return null;
|
||||||
|
|
||||||
if (!cell && !fetchingOutput)
|
if (!cell && !fetchingOutput)
|
||||||
return (
|
return (
|
||||||
<CellWrapper>
|
<CellWrapper {...wrapperProps}>
|
||||||
<Text color="gray.500">Error retrieving output</Text>
|
<Text color="gray.500">Error retrieving output</Text>
|
||||||
</CellWrapper>
|
</CellWrapper>
|
||||||
);
|
);
|
||||||
|
|
||||||
if (cell && cell.errorMessage) {
|
if (cell && cell.errorMessage) {
|
||||||
return (
|
return (
|
||||||
<CellWrapper>
|
<CellWrapper {...wrapperProps}>
|
||||||
<Text color="red.500">{cell.errorMessage}</Text>
|
<Text color="red.500">{cell.errorMessage}</Text>
|
||||||
</CellWrapper>
|
</CellWrapper>
|
||||||
);
|
);
|
||||||
@@ -112,7 +102,12 @@ export default function OutputCell({
|
|||||||
|
|
||||||
if (showLogs)
|
if (showLogs)
|
||||||
return (
|
return (
|
||||||
<CellWrapper alignItems="flex-start" fontFamily="inconsolata, monospace" spacing={0}>
|
<CellWrapper
|
||||||
|
{...wrapperProps}
|
||||||
|
alignItems="flex-start"
|
||||||
|
fontFamily="inconsolata, monospace"
|
||||||
|
spacing={0}
|
||||||
|
>
|
||||||
{cell?.jobQueuedAt && <ResponseLog time={cell.jobQueuedAt} title="Job queued" />}
|
{cell?.jobQueuedAt && <ResponseLog time={cell.jobQueuedAt} title="Job queued" />}
|
||||||
{cell?.jobStartedAt && <ResponseLog time={cell.jobStartedAt} title="Job started" />}
|
{cell?.jobStartedAt && <ResponseLog time={cell.jobStartedAt} title="Job started" />}
|
||||||
{cell?.modelResponses?.map((response) => {
|
{cell?.modelResponses?.map((response) => {
|
||||||
@@ -152,9 +147,10 @@ export default function OutputCell({
|
|||||||
<ResponseLog
|
<ResponseLog
|
||||||
time={response.receivedAt}
|
time={response.receivedAt}
|
||||||
title="Response received from API"
|
title="Response received from API"
|
||||||
message={`statusCode: ${response.statusCode ?? ""}\n ${
|
message={[
|
||||||
response.errorMessage ?? ""
|
response.statusCode ? `Status: ${response.statusCode}\n` : "",
|
||||||
}`}
|
response.errorMessage ?? "",
|
||||||
|
].join("")}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</Fragment>
|
</Fragment>
|
||||||
@@ -174,7 +170,7 @@ export default function OutputCell({
|
|||||||
|
|
||||||
if (mostRecentResponse?.respPayload && normalizedOutput?.type === "json") {
|
if (mostRecentResponse?.respPayload && normalizedOutput?.type === "json") {
|
||||||
return (
|
return (
|
||||||
<CellWrapper>
|
<CellWrapper {...wrapperProps}>
|
||||||
<SyntaxHighlighter
|
<SyntaxHighlighter
|
||||||
customStyle={{ overflowX: "unset", width: "100%", flex: 1 }}
|
customStyle={{ overflowX: "unset", width: "100%", flex: 1 }}
|
||||||
language="json"
|
language="json"
|
||||||
@@ -193,7 +189,7 @@ export default function OutputCell({
|
|||||||
const contentToDisplay = (normalizedOutput?.type === "text" && normalizedOutput.value) || "";
|
const contentToDisplay = (normalizedOutput?.type === "text" && normalizedOutput.value) || "";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<CellWrapper>
|
<CellWrapper {...wrapperProps}>
|
||||||
<Text whiteSpace="pre-wrap">{contentToDisplay}</Text>
|
<Text whiteSpace="pre-wrap">{contentToDisplay}</Text>
|
||||||
</CellWrapper>
|
</CellWrapper>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -5,30 +5,103 @@ import {
|
|||||||
ModalContent,
|
ModalContent,
|
||||||
ModalHeader,
|
ModalHeader,
|
||||||
ModalOverlay,
|
ModalOverlay,
|
||||||
|
VStack,
|
||||||
|
Text,
|
||||||
|
Box,
|
||||||
type UseDisclosureReturn,
|
type UseDisclosureReturn,
|
||||||
|
Link,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { type RouterOutputs } from "~/utils/api";
|
import { api, type RouterOutputs } from "~/utils/api";
|
||||||
import { JSONTree } from "react-json-tree";
|
import { JSONTree } from "react-json-tree";
|
||||||
|
import CopiableCode from "~/components/CopiableCode";
|
||||||
|
|
||||||
export default function ExpandedModal(props: {
|
const theme = {
|
||||||
|
scheme: "chalk",
|
||||||
|
author: "chris kempson (http://chriskempson.com)",
|
||||||
|
base00: "transparent",
|
||||||
|
base01: "#202020",
|
||||||
|
base02: "#303030",
|
||||||
|
base03: "#505050",
|
||||||
|
base04: "#b0b0b0",
|
||||||
|
base05: "#d0d0d0",
|
||||||
|
base06: "#e0e0e0",
|
||||||
|
base07: "#f5f5f5",
|
||||||
|
base08: "#fb9fb1",
|
||||||
|
base09: "#eda987",
|
||||||
|
base0A: "#ddb26f",
|
||||||
|
base0B: "#acc267",
|
||||||
|
base0C: "#12cfc0",
|
||||||
|
base0D: "#6fc2ef",
|
||||||
|
base0E: "#e1a3ee",
|
||||||
|
base0F: "#deaf8f",
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function PromptModal(props: {
|
||||||
cell: NonNullable<RouterOutputs["scenarioVariantCells"]["get"]>;
|
cell: NonNullable<RouterOutputs["scenarioVariantCells"]["get"]>;
|
||||||
disclosure: UseDisclosureReturn;
|
disclosure: UseDisclosureReturn;
|
||||||
}) {
|
}) {
|
||||||
|
const { data } = api.scenarioVariantCells.getTemplatedPromptMessage.useQuery(
|
||||||
|
{
|
||||||
|
cellId: props.cell.id,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
enabled: props.disclosure.isOpen,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal isOpen={props.disclosure.isOpen} onClose={props.disclosure.onClose} size="2xl">
|
<Modal isOpen={props.disclosure.isOpen} onClose={props.disclosure.onClose} size="xl">
|
||||||
<ModalOverlay />
|
<ModalOverlay />
|
||||||
<ModalContent>
|
<ModalContent>
|
||||||
<ModalHeader>Prompt</ModalHeader>
|
<ModalHeader>Prompt Details</ModalHeader>
|
||||||
<ModalCloseButton />
|
<ModalCloseButton />
|
||||||
<ModalBody>
|
<ModalBody>
|
||||||
<JSONTree
|
<VStack py={4} w="">
|
||||||
data={props.cell.prompt}
|
<VStack w="full" alignItems="flex-start">
|
||||||
invertTheme={true}
|
<Text fontWeight="bold">Full Prompt</Text>
|
||||||
theme="chalk"
|
<Box
|
||||||
shouldExpandNodeInitially={() => true}
|
w="full"
|
||||||
getItemString={() => ""}
|
p={4}
|
||||||
hideRoot
|
alignItems="flex-start"
|
||||||
/>
|
backgroundColor="blackAlpha.800"
|
||||||
|
borderRadius={4}
|
||||||
|
>
|
||||||
|
<JSONTree
|
||||||
|
data={props.cell.prompt}
|
||||||
|
theme={theme}
|
||||||
|
shouldExpandNodeInitially={() => true}
|
||||||
|
getItemString={() => ""}
|
||||||
|
hideRoot
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
</VStack>
|
||||||
|
{data?.templatedPrompt && (
|
||||||
|
<VStack w="full" mt={4} alignItems="flex-start">
|
||||||
|
<Text fontWeight="bold">Templated prompt message:</Text>
|
||||||
|
<CopiableCode
|
||||||
|
w="full"
|
||||||
|
// bgColor="gray.100"
|
||||||
|
p={4}
|
||||||
|
borderWidth={1}
|
||||||
|
whiteSpace="pre-wrap"
|
||||||
|
code={data.templatedPrompt}
|
||||||
|
/>
|
||||||
|
</VStack>
|
||||||
|
)}
|
||||||
|
{data?.learnMoreUrl && (
|
||||||
|
<Link
|
||||||
|
href={data.learnMoreUrl}
|
||||||
|
isExternal
|
||||||
|
color="blue.500"
|
||||||
|
fontWeight="bold"
|
||||||
|
fontSize="sm"
|
||||||
|
mt={4}
|
||||||
|
alignSelf="flex-end"
|
||||||
|
>
|
||||||
|
Learn More
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</VStack>
|
||||||
</ModalBody>
|
</ModalBody>
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
</Modal>
|
</Modal>
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ import {
|
|||||||
VStack,
|
VStack,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { BsArrowsAngleExpand, BsX } from "react-icons/bs";
|
import { BsArrowsAngleExpand, BsX } from "react-icons/bs";
|
||||||
import { cellPadding } from "../constants";
|
import { cellPadding } from "./constants";
|
||||||
import { FloatingLabelInput } from "./FloatingLabelInput";
|
import { FloatingLabelInput } from "./FloatingLabelInput";
|
||||||
import { ScenarioEditorModal } from "./ScenarioEditorModal";
|
import { ScenarioEditorModal } from "./ScenarioEditorModal";
|
||||||
|
|
||||||
@@ -111,25 +111,23 @@ export default function ScenarioEditor({
|
|||||||
onDrop={onReorder}
|
onDrop={onReorder}
|
||||||
backgroundColor={isDragTarget ? "gray.100" : "transparent"}
|
backgroundColor={isDragTarget ? "gray.100" : "transparent"}
|
||||||
>
|
>
|
||||||
{variableLabels.length === 0 ? (
|
{
|
||||||
<Box color="gray.500">
|
|
||||||
{vars.data ? "No scenario variables configured" : "Loading..."}
|
|
||||||
</Box>
|
|
||||||
) : (
|
|
||||||
<VStack spacing={4} flex={1} py={2}>
|
<VStack spacing={4} flex={1} py={2}>
|
||||||
<HStack justifyContent="space-between" w="100%" align="center" spacing={0}>
|
<HStack justifyContent="space-between" w="100%" align="center" spacing={0}>
|
||||||
<Text flex={1}>Scenario</Text>
|
<Text flex={1}>Scenario</Text>
|
||||||
<Tooltip label="Expand" hasArrow>
|
{variableLabels.length && (
|
||||||
<IconButton
|
<Tooltip label="Expand" hasArrow>
|
||||||
aria-label="Expand"
|
<IconButton
|
||||||
icon={<Icon as={BsArrowsAngleExpand} boxSize={3} />}
|
aria-label="Expand"
|
||||||
onClick={() => setScenarioEditorModalOpen(true)}
|
icon={<Icon as={BsArrowsAngleExpand} boxSize={3} />}
|
||||||
size="xs"
|
onClick={() => setScenarioEditorModalOpen(true)}
|
||||||
colorScheme="gray"
|
size="xs"
|
||||||
color="gray.500"
|
colorScheme="gray"
|
||||||
variant="ghost"
|
color="gray.500"
|
||||||
/>
|
variant="ghost"
|
||||||
</Tooltip>
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
)}
|
||||||
{canModify && props.canHide && (
|
{canModify && props.canHide && (
|
||||||
<Tooltip label="Delete" hasArrow>
|
<Tooltip label="Delete" hasArrow>
|
||||||
<IconButton
|
<IconButton
|
||||||
@@ -150,31 +148,38 @@ export default function ScenarioEditor({
|
|||||||
</Tooltip>
|
</Tooltip>
|
||||||
)}
|
)}
|
||||||
</HStack>
|
</HStack>
|
||||||
{variableLabels.map((key) => {
|
|
||||||
const value = values[key] ?? "";
|
{variableLabels.length === 0 ? (
|
||||||
return (
|
<Box color="gray.500">
|
||||||
<FloatingLabelInput
|
{vars.data ? "No scenario variables configured" : "Loading..."}
|
||||||
key={key}
|
</Box>
|
||||||
label={key}
|
) : (
|
||||||
isDisabled={!canModify}
|
variableLabels.map((key) => {
|
||||||
style={{ width: "100%" }}
|
const value = values[key] ?? "";
|
||||||
maxHeight={32}
|
return (
|
||||||
value={value}
|
<FloatingLabelInput
|
||||||
onChange={(e) => {
|
key={key}
|
||||||
setValues((prev) => ({ ...prev, [key]: e.target.value }));
|
label={key}
|
||||||
}}
|
isDisabled={!canModify}
|
||||||
onKeyDown={(e) => {
|
style={{ width: "100%" }}
|
||||||
if (e.key === "Enter" && (e.metaKey || e.ctrlKey)) {
|
maxHeight={32}
|
||||||
e.preventDefault();
|
value={value}
|
||||||
e.currentTarget.blur();
|
onChange={(e) => {
|
||||||
onSave();
|
setValues((prev) => ({ ...prev, [key]: e.target.value }));
|
||||||
}
|
}}
|
||||||
}}
|
onKeyDown={(e) => {
|
||||||
onMouseEnter={() => setVariableInputHovered(true)}
|
if (e.key === "Enter" && (e.metaKey || e.ctrlKey)) {
|
||||||
onMouseLeave={() => setVariableInputHovered(false)}
|
e.preventDefault();
|
||||||
/>
|
e.currentTarget.blur();
|
||||||
);
|
onSave();
|
||||||
})}
|
}
|
||||||
|
}}
|
||||||
|
onMouseEnter={() => setVariableInputHovered(true)}
|
||||||
|
onMouseLeave={() => setVariableInputHovered(false)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
)}
|
||||||
{hasChanged && (
|
{hasChanged && (
|
||||||
<HStack justify="right">
|
<HStack justify="right">
|
||||||
<Button
|
<Button
|
||||||
@@ -192,7 +197,7 @@ export default function ScenarioEditor({
|
|||||||
</HStack>
|
</HStack>
|
||||||
)}
|
)}
|
||||||
</VStack>
|
</VStack>
|
||||||
)}
|
}
|
||||||
</HStack>
|
</HStack>
|
||||||
{scenarioEditorModalOpen && (
|
{scenarioEditorModalOpen && (
|
||||||
<ScenarioEditorModal
|
<ScenarioEditorModal
|
||||||
|
|||||||
@@ -65,11 +65,11 @@ export const ScenarioEditorModal = ({
|
|||||||
<Modal
|
<Modal
|
||||||
isOpen
|
isOpen
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
size={{ base: "xl", sm: "2xl", md: "3xl", lg: "5xl", xl: "7xl" }}
|
size={{ base: "xl", sm: "2xl", md: "3xl", lg: "4xl", xl: "5xl" }}
|
||||||
>
|
>
|
||||||
<ModalOverlay />
|
<ModalOverlay />
|
||||||
<ModalContent w={1200}>
|
<ModalContent w={1200}>
|
||||||
<ModalHeader />
|
<ModalHeader>Edit Scenario</ModalHeader>
|
||||||
<ModalCloseButton />
|
<ModalCloseButton />
|
||||||
<ModalBody maxW="unset">
|
<ModalBody maxW="unset">
|
||||||
<VStack spacing={8}>
|
<VStack spacing={8}>
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ import {
|
|||||||
IconButton,
|
IconButton,
|
||||||
Spinner,
|
Spinner,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { cellPadding } from "../constants";
|
import { cellPadding } from "./constants";
|
||||||
import {
|
import {
|
||||||
useExperiment,
|
useExperiment,
|
||||||
useExperimentAccess,
|
useExperimentAccess,
|
||||||
|
|||||||
@@ -110,7 +110,7 @@ export default function VariantEditor(props: { variant: PromptVariant }) {
|
|||||||
setIsChanged(false);
|
setIsChanged(false);
|
||||||
|
|
||||||
await utils.promptVariants.list.invalidate();
|
await utils.promptVariants.list.invalidate();
|
||||||
}, [checkForChanges]);
|
}, [checkForChanges, replaceVariant.mutateAsync]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (monaco) {
|
if (monaco) {
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import { useState, type DragEvent } from "react";
|
import { useState, type DragEvent } from "react";
|
||||||
import { type PromptVariant } from "../OutputsTable/types";
|
import { type PromptVariant } from "../types";
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import { RiDraggable } from "react-icons/ri";
|
import { RiDraggable } from "react-icons/ri";
|
||||||
import { useExperimentAccess, useHandledAsyncCallback } from "~/utils/hooks";
|
import { useExperimentAccess, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
import { HStack, Icon, Text, GridItem, type GridItemProps } from "@chakra-ui/react"; // Changed here
|
import { HStack, Icon, Text, GridItem, type GridItemProps } from "@chakra-ui/react"; // Changed here
|
||||||
import { cellPadding, headerMinHeight } from "../constants";
|
import { cellPadding, headerMinHeight } from "../constants";
|
||||||
import AutoResizeTextArea from "../AutoResizeTextArea";
|
import AutoResizeTextArea from "../../AutoResizeTextArea";
|
||||||
import VariantHeaderMenuButton from "./VariantHeaderMenuButton";
|
import VariantHeaderMenuButton from "./VariantHeaderMenuButton";
|
||||||
|
|
||||||
export default function VariantHeader(
|
export default function VariantHeader(
|
||||||
@@ -75,7 +75,7 @@ export default function VariantHeader(
|
|||||||
padding={0}
|
padding={0}
|
||||||
sx={{
|
sx={{
|
||||||
position: "sticky",
|
position: "sticky",
|
||||||
top: "-2",
|
top: "0",
|
||||||
// Ensure that the menu always appears above the sticky header of other variants
|
// Ensure that the menu always appears above the sticky header of other variants
|
||||||
zIndex: menuOpen ? "dropdown" : 10,
|
zIndex: menuOpen ? "dropdown" : 10,
|
||||||
}}
|
}}
|
||||||
@@ -1,6 +1,4 @@
|
|||||||
import { type PromptVariant } from "../OutputsTable/types";
|
import { useState } from "react";
|
||||||
import { api } from "~/utils/api";
|
|
||||||
import { useHandledAsyncCallback, useVisibleScenarioIds } from "~/utils/hooks";
|
|
||||||
import {
|
import {
|
||||||
Icon,
|
Icon,
|
||||||
Menu,
|
Menu,
|
||||||
@@ -14,10 +12,13 @@ import {
|
|||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { BsFillTrashFill, BsGear, BsStars } from "react-icons/bs";
|
import { BsFillTrashFill, BsGear, BsStars } from "react-icons/bs";
|
||||||
import { FaRegClone } from "react-icons/fa";
|
import { FaRegClone } from "react-icons/fa";
|
||||||
import { useState } from "react";
|
|
||||||
import { RefinePromptModal } from "../RefinePromptModal/RefinePromptModal";
|
|
||||||
import { RiExchangeFundsFill } from "react-icons/ri";
|
import { RiExchangeFundsFill } from "react-icons/ri";
|
||||||
import { ChangeModelModal } from "../ChangeModelModal/ChangeModelModal";
|
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useHandledAsyncCallback, useVisibleScenarioIds } from "~/utils/hooks";
|
||||||
|
import { type PromptVariant } from "../types";
|
||||||
|
import { RefinePromptModal } from "../../RefinePromptModal/RefinePromptModal";
|
||||||
|
import { ChangeModelModal } from "../../ChangeModelModal/ChangeModelModal";
|
||||||
|
|
||||||
export default function VariantHeaderMenuButton({
|
export default function VariantHeaderMenuButton({
|
||||||
variant,
|
variant,
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { HStack, Icon, Text, useToken } from "@chakra-ui/react";
|
import { HStack, Icon, Text, useToken } from "@chakra-ui/react";
|
||||||
import { type PromptVariant } from "./types";
|
import { type PromptVariant } from "./types";
|
||||||
import { cellPadding } from "../constants";
|
import { cellPadding } from "./constants";
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import chroma from "chroma-js";
|
import chroma from "chroma-js";
|
||||||
import { BsCurrencyDollar } from "react-icons/bs";
|
import { BsCurrencyDollar } from "react-icons/bs";
|
||||||
|
|||||||
@@ -3,13 +3,14 @@ import { api } from "~/utils/api";
|
|||||||
import AddVariantButton from "./AddVariantButton";
|
import AddVariantButton from "./AddVariantButton";
|
||||||
import ScenarioRow from "./ScenarioRow";
|
import ScenarioRow from "./ScenarioRow";
|
||||||
import VariantEditor from "./VariantEditor";
|
import VariantEditor from "./VariantEditor";
|
||||||
import VariantHeader from "../VariantHeader/VariantHeader";
|
import VariantHeader from "./VariantHeader/VariantHeader";
|
||||||
import VariantStats from "./VariantStats";
|
import VariantStats from "./VariantStats";
|
||||||
import { ScenariosHeader } from "./ScenariosHeader";
|
import { ScenariosHeader } from "./ScenariosHeader";
|
||||||
import { borders } from "./styles";
|
import { borders } from "./styles";
|
||||||
import { useScenarios } from "~/utils/hooks";
|
import { useScenarios } from "~/utils/hooks";
|
||||||
import ScenarioPaginator from "./ScenarioPaginator";
|
import ScenarioPaginator from "./ScenarioPaginator";
|
||||||
import { Fragment } from "react";
|
import { Fragment } from "react";
|
||||||
|
import useScrolledPast from "./useHasScrolledPast";
|
||||||
|
|
||||||
export default function OutputsTable({ experimentId }: { experimentId: string | undefined }) {
|
export default function OutputsTable({ experimentId }: { experimentId: string | undefined }) {
|
||||||
const variants = api.promptVariants.list.useQuery(
|
const variants = api.promptVariants.list.useQuery(
|
||||||
@@ -18,6 +19,7 @@ export default function OutputsTable({ experimentId }: { experimentId: string |
|
|||||||
);
|
);
|
||||||
|
|
||||||
const scenarios = useScenarios();
|
const scenarios = useScenarios();
|
||||||
|
const shouldFlattenHeader = useScrolledPast(50);
|
||||||
|
|
||||||
if (!variants.data || !scenarios.data) return null;
|
if (!variants.data || !scenarios.data) return null;
|
||||||
|
|
||||||
@@ -63,8 +65,8 @@ export default function OutputsTable({ experimentId }: { experimentId: string |
|
|||||||
variant={variant}
|
variant={variant}
|
||||||
canHide={variants.data.length > 1}
|
canHide={variants.data.length > 1}
|
||||||
rowStart={1}
|
rowStart={1}
|
||||||
borderTopLeftRadius={isFirst ? 8 : 0}
|
borderTopLeftRadius={isFirst && !shouldFlattenHeader ? 8 : 0}
|
||||||
borderTopRightRadius={isLast ? 8 : 0}
|
borderTopRightRadius={isLast && !shouldFlattenHeader ? 8 : 0}
|
||||||
{...sharedProps}
|
{...sharedProps}
|
||||||
/>
|
/>
|
||||||
<GridItem rowStart={2} {...sharedProps}>
|
<GridItem rowStart={2} {...sharedProps}>
|
||||||
@@ -75,6 +77,7 @@ export default function OutputsTable({ experimentId }: { experimentId: string |
|
|||||||
{...sharedProps}
|
{...sharedProps}
|
||||||
borderBottomLeftRadius={isFirst ? 8 : 0}
|
borderBottomLeftRadius={isFirst ? 8 : 0}
|
||||||
borderBottomRightRadius={isLast ? 8 : 0}
|
borderBottomRightRadius={isLast ? 8 : 0}
|
||||||
|
boxShadow="5px 5px 15px 1px rgba(0, 0, 0, 0.1);"
|
||||||
>
|
>
|
||||||
<VariantStats variant={variant} />
|
<VariantStats variant={variant} />
|
||||||
</GridItem>
|
</GridItem>
|
||||||
|
|||||||
34
app/src/components/OutputsTable/useHasScrolledPast.tsx
Normal file
34
app/src/components/OutputsTable/useHasScrolledPast.tsx
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
|
|
||||||
|
const useScrolledPast = (scrollThreshold: number) => {
|
||||||
|
const [hasScrolledPast, setHasScrolledPast] = useState(true);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const container = document.getElementById("output-container");
|
||||||
|
|
||||||
|
if (!container) {
|
||||||
|
console.warn('Element with id "outputs-container" not found.');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const checkScroll = () => {
|
||||||
|
const { scrollTop } = container;
|
||||||
|
|
||||||
|
// Check if scrollTop is greater than or equal to scrollThreshold
|
||||||
|
setHasScrolledPast(scrollTop > scrollThreshold);
|
||||||
|
};
|
||||||
|
|
||||||
|
checkScroll();
|
||||||
|
|
||||||
|
container.addEventListener("scroll", checkScroll);
|
||||||
|
|
||||||
|
// Cleanup
|
||||||
|
return () => {
|
||||||
|
container.removeEventListener("scroll", checkScroll);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return hasScrolledPast;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useScrolledPast;
|
||||||
@@ -1,15 +1,19 @@
|
|||||||
import { HStack, IconButton, Text, Select, type StackProps, Icon } from "@chakra-ui/react";
|
import {
|
||||||
|
HStack,
|
||||||
|
IconButton,
|
||||||
|
Text,
|
||||||
|
Select,
|
||||||
|
type StackProps,
|
||||||
|
Icon,
|
||||||
|
useBreakpointValue,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
import React, { useCallback } from "react";
|
import React, { useCallback } from "react";
|
||||||
import { FiChevronsLeft, FiChevronsRight, FiChevronLeft, FiChevronRight } from "react-icons/fi";
|
import { FiChevronsLeft, FiChevronsRight, FiChevronLeft, FiChevronRight } from "react-icons/fi";
|
||||||
import { usePageParams } from "~/utils/hooks";
|
import { usePageParams } from "~/utils/hooks";
|
||||||
|
|
||||||
const pageSizeOptions = [10, 25, 50, 100];
|
const pageSizeOptions = [10, 25, 50, 100];
|
||||||
|
|
||||||
const Paginator = ({
|
const Paginator = ({ count, ...props }: { count: number; condense?: boolean } & StackProps) => {
|
||||||
count,
|
|
||||||
condense,
|
|
||||||
...props
|
|
||||||
}: { count: number; condense?: boolean } & StackProps) => {
|
|
||||||
const { page, pageSize, setPageParams } = usePageParams();
|
const { page, pageSize, setPageParams } = usePageParams();
|
||||||
|
|
||||||
const lastPage = Math.ceil(count / pageSize);
|
const lastPage = Math.ceil(count / pageSize);
|
||||||
@@ -37,6 +41,11 @@ const Paginator = ({
|
|||||||
const goToLastPage = () => setPageParams({ page: lastPage }, "replace");
|
const goToLastPage = () => setPageParams({ page: lastPage }, "replace");
|
||||||
const goToFirstPage = () => setPageParams({ page: 1 }, "replace");
|
const goToFirstPage = () => setPageParams({ page: 1 }, "replace");
|
||||||
|
|
||||||
|
const isMobile = useBreakpointValue({ base: true, md: false });
|
||||||
|
const condense = isMobile || props.condense;
|
||||||
|
|
||||||
|
if (count === 0) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HStack
|
<HStack
|
||||||
pt={4}
|
pt={4}
|
||||||
|
|||||||
@@ -1,112 +0,0 @@
|
|||||||
import {
|
|
||||||
HStack,
|
|
||||||
Icon,
|
|
||||||
VStack,
|
|
||||||
Text,
|
|
||||||
Divider,
|
|
||||||
Spinner,
|
|
||||||
AspectRatio,
|
|
||||||
SkeletonText,
|
|
||||||
} from "@chakra-ui/react";
|
|
||||||
import { RiDatabase2Line } from "react-icons/ri";
|
|
||||||
import { formatTimePast } from "~/utils/dayjs";
|
|
||||||
import Link from "next/link";
|
|
||||||
import { useRouter } from "next/router";
|
|
||||||
import { BsPlusSquare } from "react-icons/bs";
|
|
||||||
import { api } from "~/utils/api";
|
|
||||||
import { useHandledAsyncCallback } from "~/utils/hooks";
|
|
||||||
import { useAppStore } from "~/state/store";
|
|
||||||
|
|
||||||
type DatasetData = {
|
|
||||||
name: string;
|
|
||||||
numEntries: number;
|
|
||||||
id: string;
|
|
||||||
createdAt: Date;
|
|
||||||
updatedAt: Date;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const DatasetCard = ({ dataset }: { dataset: DatasetData }) => {
|
|
||||||
return (
|
|
||||||
<AspectRatio ratio={1.2} w="full">
|
|
||||||
<VStack
|
|
||||||
as={Link}
|
|
||||||
href={{ pathname: "/data/[id]", query: { id: dataset.id } }}
|
|
||||||
bg="gray.50"
|
|
||||||
_hover={{ bg: "gray.100" }}
|
|
||||||
transition="background 0.2s"
|
|
||||||
cursor="pointer"
|
|
||||||
borderColor="gray.200"
|
|
||||||
borderWidth={1}
|
|
||||||
p={4}
|
|
||||||
justify="space-between"
|
|
||||||
>
|
|
||||||
<HStack w="full" color="gray.700" justify="center">
|
|
||||||
<Icon as={RiDatabase2Line} boxSize={4} />
|
|
||||||
<Text fontWeight="bold">{dataset.name}</Text>
|
|
||||||
</HStack>
|
|
||||||
<HStack h="full" spacing={4} flex={1} align="center">
|
|
||||||
<CountLabel label="Rows" count={dataset.numEntries} />
|
|
||||||
</HStack>
|
|
||||||
<HStack w="full" color="gray.500" fontSize="xs" textAlign="center">
|
|
||||||
<Text flex={1}>Created {formatTimePast(dataset.createdAt)}</Text>
|
|
||||||
<Divider h={4} orientation="vertical" />
|
|
||||||
<Text flex={1}>Updated {formatTimePast(dataset.updatedAt)}</Text>
|
|
||||||
</HStack>
|
|
||||||
</VStack>
|
|
||||||
</AspectRatio>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const CountLabel = ({ label, count }: { label: string; count: number }) => {
|
|
||||||
return (
|
|
||||||
<VStack alignItems="center" flex={1}>
|
|
||||||
<Text color="gray.500" fontWeight="bold">
|
|
||||||
{label}
|
|
||||||
</Text>
|
|
||||||
<Text fontSize="sm" color="gray.500">
|
|
||||||
{count}
|
|
||||||
</Text>
|
|
||||||
</VStack>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const NewDatasetCard = () => {
|
|
||||||
const router = useRouter();
|
|
||||||
const selectedProjectId = useAppStore((s) => s.selectedProjectId);
|
|
||||||
const createMutation = api.datasets.create.useMutation();
|
|
||||||
const [createDataset, isLoading] = useHandledAsyncCallback(async () => {
|
|
||||||
const newDataset = await createMutation.mutateAsync({ projectId: selectedProjectId ?? "" });
|
|
||||||
await router.push({ pathname: "/data/[id]", query: { id: newDataset.id } });
|
|
||||||
}, [createMutation, router, selectedProjectId]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<AspectRatio ratio={1.2} w="full">
|
|
||||||
<VStack
|
|
||||||
align="center"
|
|
||||||
justify="center"
|
|
||||||
_hover={{ cursor: "pointer", bg: "gray.50" }}
|
|
||||||
transition="background 0.2s"
|
|
||||||
cursor="pointer"
|
|
||||||
borderColor="gray.200"
|
|
||||||
borderWidth={1}
|
|
||||||
p={4}
|
|
||||||
onClick={createDataset}
|
|
||||||
>
|
|
||||||
<Icon as={isLoading ? Spinner : BsPlusSquare} boxSize={8} />
|
|
||||||
<Text display={{ base: "none", md: "block" }} ml={2}>
|
|
||||||
New Dataset
|
|
||||||
</Text>
|
|
||||||
</VStack>
|
|
||||||
</AspectRatio>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const DatasetCardSkeleton = () => (
|
|
||||||
<AspectRatio ratio={1.2} w="full">
|
|
||||||
<VStack align="center" borderColor="gray.200" borderWidth={1} p={4} bg="gray.50">
|
|
||||||
<SkeletonText noOfLines={1} w="80%" />
|
|
||||||
<SkeletonText noOfLines={2} w="60%" />
|
|
||||||
<SkeletonText noOfLines={1} w="80%" />
|
|
||||||
</VStack>
|
|
||||||
</AspectRatio>
|
|
||||||
);
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
import { type StackProps } from "@chakra-ui/react";
|
|
||||||
|
|
||||||
import { useDatasetEntries } from "~/utils/hooks";
|
|
||||||
import Paginator from "../Paginator";
|
|
||||||
|
|
||||||
const DatasetEntriesPaginator = (props: StackProps) => {
|
|
||||||
const { data } = useDatasetEntries();
|
|
||||||
|
|
||||||
if (!data) return null;
|
|
||||||
|
|
||||||
const { count } = data;
|
|
||||||
|
|
||||||
return <Paginator count={count} {...props} />;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default DatasetEntriesPaginator;
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
import { type StackProps, VStack, Table, Th, Tr, Thead, Tbody, Text } from "@chakra-ui/react";
|
|
||||||
import { useDatasetEntries } from "~/utils/hooks";
|
|
||||||
import TableRow from "./TableRow";
|
|
||||||
import DatasetEntriesPaginator from "./DatasetEntriesPaginator";
|
|
||||||
|
|
||||||
const DatasetEntriesTable = (props: StackProps) => {
|
|
||||||
const { data } = useDatasetEntries();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<VStack justifyContent="space-between" {...props}>
|
|
||||||
<Table variant="simple" sx={{ "table-layout": "fixed", width: "full" }}>
|
|
||||||
<Thead>
|
|
||||||
<Tr>
|
|
||||||
<Th>Input</Th>
|
|
||||||
<Th>Output</Th>
|
|
||||||
</Tr>
|
|
||||||
</Thead>
|
|
||||||
<Tbody>{data?.entries.map((entry) => <TableRow key={entry.id} entry={entry} />)}</Tbody>
|
|
||||||
</Table>
|
|
||||||
{(!data || data.entries.length) === 0 ? (
|
|
||||||
<Text alignSelf="flex-start" pl={6} color="gray.500">
|
|
||||||
No entries found
|
|
||||||
</Text>
|
|
||||||
) : (
|
|
||||||
<DatasetEntriesPaginator />
|
|
||||||
)}
|
|
||||||
</VStack>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default DatasetEntriesTable;
|
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
import { Button, HStack, useDisclosure } from "@chakra-ui/react";
|
|
||||||
import { BiImport } from "react-icons/bi";
|
|
||||||
import { BsStars } from "react-icons/bs";
|
|
||||||
|
|
||||||
import { GenerateDataModal } from "./GenerateDataModal";
|
|
||||||
|
|
||||||
export const DatasetHeaderButtons = () => {
|
|
||||||
const generateModalDisclosure = useDisclosure();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<HStack>
|
|
||||||
<Button leftIcon={<BiImport />} colorScheme="blue" variant="ghost">
|
|
||||||
Import Data
|
|
||||||
</Button>
|
|
||||||
<Button leftIcon={<BsStars />} colorScheme="blue" onClick={generateModalDisclosure.onOpen}>
|
|
||||||
Generate Data
|
|
||||||
</Button>
|
|
||||||
</HStack>
|
|
||||||
<GenerateDataModal
|
|
||||||
isOpen={generateModalDisclosure.isOpen}
|
|
||||||
onClose={generateModalDisclosure.onClose}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1,128 +0,0 @@
|
|||||||
import {
|
|
||||||
Modal,
|
|
||||||
ModalBody,
|
|
||||||
ModalCloseButton,
|
|
||||||
ModalContent,
|
|
||||||
ModalHeader,
|
|
||||||
ModalOverlay,
|
|
||||||
ModalFooter,
|
|
||||||
Text,
|
|
||||||
HStack,
|
|
||||||
VStack,
|
|
||||||
Icon,
|
|
||||||
NumberInput,
|
|
||||||
NumberInputField,
|
|
||||||
NumberInputStepper,
|
|
||||||
NumberIncrementStepper,
|
|
||||||
NumberDecrementStepper,
|
|
||||||
Button,
|
|
||||||
} from "@chakra-ui/react";
|
|
||||||
import { BsStars } from "react-icons/bs";
|
|
||||||
import { useState } from "react";
|
|
||||||
import { useDataset, useHandledAsyncCallback } from "~/utils/hooks";
|
|
||||||
import { api } from "~/utils/api";
|
|
||||||
import AutoResizeTextArea from "~/components/AutoResizeTextArea";
|
|
||||||
|
|
||||||
export const GenerateDataModal = ({
|
|
||||||
isOpen,
|
|
||||||
onClose,
|
|
||||||
}: {
|
|
||||||
isOpen: boolean;
|
|
||||||
onClose: () => void;
|
|
||||||
}) => {
|
|
||||||
const utils = api.useContext();
|
|
||||||
|
|
||||||
const datasetId = useDataset().data?.id;
|
|
||||||
|
|
||||||
const [numToGenerate, setNumToGenerate] = useState<number>(20);
|
|
||||||
const [inputDescription, setInputDescription] = useState<string>(
|
|
||||||
"Each input should contain an email body. Half of the emails should contain event details, and the other half should not.",
|
|
||||||
);
|
|
||||||
const [outputDescription, setOutputDescription] = useState<string>(
|
|
||||||
`Each output should contain "true" or "false", where "true" indicates that the email contains event details.`,
|
|
||||||
);
|
|
||||||
|
|
||||||
const generateEntriesMutation = api.datasetEntries.autogenerateEntries.useMutation();
|
|
||||||
|
|
||||||
const [generateEntries, generateEntriesInProgress] = useHandledAsyncCallback(async () => {
|
|
||||||
if (!inputDescription || !outputDescription || !numToGenerate || !datasetId) return;
|
|
||||||
await generateEntriesMutation.mutateAsync({
|
|
||||||
datasetId,
|
|
||||||
inputDescription,
|
|
||||||
outputDescription,
|
|
||||||
numToGenerate,
|
|
||||||
});
|
|
||||||
await utils.datasetEntries.list.invalidate();
|
|
||||||
onClose();
|
|
||||||
}, [
|
|
||||||
generateEntriesMutation,
|
|
||||||
onClose,
|
|
||||||
inputDescription,
|
|
||||||
outputDescription,
|
|
||||||
numToGenerate,
|
|
||||||
datasetId,
|
|
||||||
]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Modal isOpen={isOpen} onClose={onClose} size={{ base: "xl", sm: "2xl", md: "3xl" }}>
|
|
||||||
<ModalOverlay />
|
|
||||||
<ModalContent w={1200}>
|
|
||||||
<ModalHeader>
|
|
||||||
<HStack>
|
|
||||||
<Icon as={BsStars} />
|
|
||||||
<Text>Generate Data</Text>
|
|
||||||
</HStack>
|
|
||||||
</ModalHeader>
|
|
||||||
<ModalCloseButton />
|
|
||||||
<ModalBody maxW="unset">
|
|
||||||
<VStack w="full" spacing={8} padding={8} alignItems="flex-start">
|
|
||||||
<VStack alignItems="flex-start" spacing={2}>
|
|
||||||
<Text fontWeight="bold">Number of Rows:</Text>
|
|
||||||
<NumberInput
|
|
||||||
step={5}
|
|
||||||
defaultValue={15}
|
|
||||||
min={0}
|
|
||||||
max={100}
|
|
||||||
onChange={(valueString) => setNumToGenerate(parseInt(valueString) || 0)}
|
|
||||||
value={numToGenerate}
|
|
||||||
w="24"
|
|
||||||
>
|
|
||||||
<NumberInputField />
|
|
||||||
<NumberInputStepper>
|
|
||||||
<NumberIncrementStepper />
|
|
||||||
<NumberDecrementStepper />
|
|
||||||
</NumberInputStepper>
|
|
||||||
</NumberInput>
|
|
||||||
</VStack>
|
|
||||||
<VStack alignItems="flex-start" w="full" spacing={2}>
|
|
||||||
<Text fontWeight="bold">Input Description:</Text>
|
|
||||||
<AutoResizeTextArea
|
|
||||||
value={inputDescription}
|
|
||||||
onChange={(e) => setInputDescription(e.target.value)}
|
|
||||||
placeholder="Each input should contain..."
|
|
||||||
/>
|
|
||||||
</VStack>
|
|
||||||
<VStack alignItems="flex-start" w="full" spacing={2}>
|
|
||||||
<Text fontWeight="bold">Output Description (optional):</Text>
|
|
||||||
<AutoResizeTextArea
|
|
||||||
value={outputDescription}
|
|
||||||
onChange={(e) => setOutputDescription(e.target.value)}
|
|
||||||
placeholder="The output should contain..."
|
|
||||||
/>
|
|
||||||
</VStack>
|
|
||||||
</VStack>
|
|
||||||
</ModalBody>
|
|
||||||
<ModalFooter>
|
|
||||||
<Button
|
|
||||||
colorScheme="blue"
|
|
||||||
isLoading={generateEntriesInProgress}
|
|
||||||
isDisabled={!numToGenerate || !inputDescription || !outputDescription}
|
|
||||||
onClick={generateEntries}
|
|
||||||
>
|
|
||||||
Generate
|
|
||||||
</Button>
|
|
||||||
</ModalFooter>
|
|
||||||
</ModalContent>
|
|
||||||
</Modal>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1,13 +0,0 @@
|
|||||||
import { Td, Tr } from "@chakra-ui/react";
|
|
||||||
import { type DatasetEntry } from "@prisma/client";
|
|
||||||
|
|
||||||
const TableRow = ({ entry }: { entry: DatasetEntry }) => {
|
|
||||||
return (
|
|
||||||
<Tr key={entry.id}>
|
|
||||||
<Td>{entry.input}</Td>
|
|
||||||
<Td>{entry.output}</Td>
|
|
||||||
</Tr>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default TableRow;
|
|
||||||
@@ -14,21 +14,11 @@ import { formatTimePast } from "~/utils/dayjs";
|
|||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { BsPlusSquare } from "react-icons/bs";
|
import { BsPlusSquare } from "react-icons/bs";
|
||||||
import { api } from "~/utils/api";
|
import { RouterOutputs, api } from "~/utils/api";
|
||||||
import { useHandledAsyncCallback } from "~/utils/hooks";
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
import { useAppStore } from "~/state/store";
|
import { useAppStore } from "~/state/store";
|
||||||
|
|
||||||
type ExperimentData = {
|
export const ExperimentCard = ({ exp }: { exp: RouterOutputs["experiments"]["list"][0] }) => {
|
||||||
testScenarioCount: number;
|
|
||||||
promptVariantCount: number;
|
|
||||||
id: string;
|
|
||||||
label: string;
|
|
||||||
sortIndex: number;
|
|
||||||
createdAt: Date;
|
|
||||||
updatedAt: Date;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const ExperimentCard = ({ exp }: { exp: ExperimentData }) => {
|
|
||||||
return (
|
return (
|
||||||
<Card
|
<Card
|
||||||
w="full"
|
w="full"
|
||||||
@@ -45,7 +35,7 @@ export const ExperimentCard = ({ exp }: { exp: ExperimentData }) => {
|
|||||||
as={Link}
|
as={Link}
|
||||||
w="full"
|
w="full"
|
||||||
h="full"
|
h="full"
|
||||||
href={{ pathname: "/experiments/[id]", query: { id: exp.id } }}
|
href={{ pathname: "/experiments/[experimentSlug]", query: { experimentSlug: exp.slug } }}
|
||||||
justify="space-between"
|
justify="space-between"
|
||||||
>
|
>
|
||||||
<HStack w="full" color="gray.700" justify="center">
|
<HStack w="full" color="gray.700" justify="center">
|
||||||
@@ -89,8 +79,8 @@ export const NewExperimentCard = () => {
|
|||||||
projectId: selectedProjectId ?? "",
|
projectId: selectedProjectId ?? "",
|
||||||
});
|
});
|
||||||
await router.push({
|
await router.push({
|
||||||
pathname: "/experiments/[id]",
|
pathname: "/experiments/[experimentSlug]",
|
||||||
query: { id: newExperiment.id },
|
query: { experimentSlug: newExperiment.slug },
|
||||||
});
|
});
|
||||||
}, [createMutation, router, selectedProjectId]);
|
}, [createMutation, router, selectedProjectId]);
|
||||||
|
|
||||||
|
|||||||
@@ -16,11 +16,14 @@ export const useOnForkButtonPressed = () => {
|
|||||||
|
|
||||||
const [onFork, isForking] = useHandledAsyncCallback(async () => {
|
const [onFork, isForking] = useHandledAsyncCallback(async () => {
|
||||||
if (!experiment.data?.id || !selectedProjectId) return;
|
if (!experiment.data?.id || !selectedProjectId) return;
|
||||||
const forkedExperimentId = await forkMutation.mutateAsync({
|
const newExperiment = await forkMutation.mutateAsync({
|
||||||
id: experiment.data.id,
|
id: experiment.data.id,
|
||||||
projectId: selectedProjectId,
|
projectId: selectedProjectId,
|
||||||
});
|
});
|
||||||
await router.push({ pathname: "/experiments/[id]", query: { id: forkedExperimentId } });
|
await router.push({
|
||||||
|
pathname: "/experiments/[experimentSlug]",
|
||||||
|
query: { experimentSlug: newExperiment.slug },
|
||||||
|
});
|
||||||
}, [forkMutation, experiment.data?.id, router]);
|
}, [forkMutation, experiment.data?.id, router]);
|
||||||
|
|
||||||
const onForkButtonPressed = useCallback(() => {
|
const onForkButtonPressed = useCallback(() => {
|
||||||
|
|||||||
65
app/src/components/fineTunes/FineTunesTable.tsx
Normal file
65
app/src/components/fineTunes/FineTunesTable.tsx
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
import { Card, Table, Thead, Tr, Th, Tbody, Td, VStack, Icon, Text } from "@chakra-ui/react";
|
||||||
|
import { FaTable } from "react-icons/fa";
|
||||||
|
import { type FineTuneStatus } from "@prisma/client";
|
||||||
|
|
||||||
|
import dayjs from "~/utils/dayjs";
|
||||||
|
import { useFineTunes } from "~/utils/hooks";
|
||||||
|
|
||||||
|
const FineTunesTable = ({}) => {
|
||||||
|
const { data } = useFineTunes();
|
||||||
|
|
||||||
|
const fineTunes = data?.fineTunes || [];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card width="100%" overflowX="auto">
|
||||||
|
{fineTunes.length ? (
|
||||||
|
<Table>
|
||||||
|
<Thead>
|
||||||
|
<Tr>
|
||||||
|
<Th>ID</Th>
|
||||||
|
<Th>Created At</Th>
|
||||||
|
<Th>Base Model</Th>
|
||||||
|
<Th>Dataset Size</Th>
|
||||||
|
<Th>Status</Th>
|
||||||
|
</Tr>
|
||||||
|
</Thead>
|
||||||
|
<Tbody>
|
||||||
|
{fineTunes.map((fineTune) => {
|
||||||
|
return (
|
||||||
|
<Tr key={fineTune.id}>
|
||||||
|
<Td>{fineTune.slug}</Td>
|
||||||
|
<Td>{dayjs(fineTune.createdAt).format("MMMM D h:mm A")}</Td>
|
||||||
|
<Td>{fineTune.baseModel}</Td>
|
||||||
|
<Td>{fineTune.dataset._count.datasetEntries}</Td>
|
||||||
|
<Td fontSize="sm" fontWeight="bold">
|
||||||
|
<Text color={getStatusColor(fineTune.status)}>{fineTune.status}</Text>
|
||||||
|
</Td>
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</Tbody>
|
||||||
|
</Table>
|
||||||
|
) : (
|
||||||
|
<VStack py={8}>
|
||||||
|
<Icon as={FaTable} boxSize={16} color="gray.300" />
|
||||||
|
<Text color="gray.400" fontSize="lg" fontWeight="bold">
|
||||||
|
No Fine Tunes Found
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FineTunesTable;
|
||||||
|
|
||||||
|
const getStatusColor = (status: FineTuneStatus) => {
|
||||||
|
switch (status) {
|
||||||
|
case "DEPLOYED":
|
||||||
|
return "green.500";
|
||||||
|
case "ERROR":
|
||||||
|
return "red.500";
|
||||||
|
default:
|
||||||
|
return "yellow.500";
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -15,12 +15,14 @@ import Head from "next/head";
|
|||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { BsGearFill, BsGithub, BsPersonCircle } from "react-icons/bs";
|
import { BsGearFill, BsGithub, BsPersonCircle } from "react-icons/bs";
|
||||||
import { IoStatsChartOutline } from "react-icons/io5";
|
import { IoStatsChartOutline } from "react-icons/io5";
|
||||||
import { RiHome3Line, RiDatabase2Line, RiFlaskLine } from "react-icons/ri";
|
import { RiHome3Line, RiFlaskLine } from "react-icons/ri";
|
||||||
|
import { FaRobot } from "react-icons/fa";
|
||||||
import { signIn, useSession } from "next-auth/react";
|
import { signIn, useSession } from "next-auth/react";
|
||||||
import { env } from "~/env.mjs";
|
|
||||||
import ProjectMenu from "./ProjectMenu";
|
import ProjectMenu from "./ProjectMenu";
|
||||||
import NavSidebarOption from "./NavSidebarOption";
|
import NavSidebarOption from "./NavSidebarOption";
|
||||||
import IconLink from "./IconLink";
|
import IconLink from "./IconLink";
|
||||||
|
import { BetaModal } from "./BetaModal";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
|
||||||
const Divider = () => <Box h="1px" bgColor="gray.300" w="full" />;
|
const Divider = () => <Box h="1px" bgColor="gray.300" w="full" />;
|
||||||
|
|
||||||
@@ -71,21 +73,10 @@ const NavSidebar = () => {
|
|||||||
<ProjectMenu />
|
<ProjectMenu />
|
||||||
<Divider />
|
<Divider />
|
||||||
|
|
||||||
{env.NEXT_PUBLIC_FF_SHOW_LOGGED_CALLS && (
|
<IconLink icon={RiHome3Line} label="Dashboard" href="/dashboard" beta />
|
||||||
<>
|
<IconLink icon={IoStatsChartOutline} label="Request Logs" href="/request-logs" beta />
|
||||||
<IconLink icon={RiHome3Line} label="Dashboard" href="/dashboard" beta />
|
<IconLink icon={FaRobot} label="Fine Tunes" href="/fine-tunes" beta />
|
||||||
<IconLink
|
|
||||||
icon={IoStatsChartOutline}
|
|
||||||
label="Request Logs"
|
|
||||||
href="/request-logs"
|
|
||||||
beta
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
<IconLink icon={RiFlaskLine} label="Experiments" href="/experiments" />
|
<IconLink icon={RiFlaskLine} label="Experiments" href="/experiments" />
|
||||||
{env.NEXT_PUBLIC_SHOW_DATA && (
|
|
||||||
<IconLink icon={RiDatabase2Line} label="Data" href="/data" />
|
|
||||||
)}
|
|
||||||
<VStack w="full" alignItems="flex-start" spacing={0} pt={8}>
|
<VStack w="full" alignItems="flex-start" spacing={0} pt={8}>
|
||||||
<Text
|
<Text
|
||||||
pl={2}
|
pl={2}
|
||||||
@@ -105,7 +96,7 @@ const NavSidebar = () => {
|
|||||||
<NavSidebarOption>
|
<NavSidebarOption>
|
||||||
<HStack
|
<HStack
|
||||||
w="full"
|
w="full"
|
||||||
p={4}
|
p={{ base: 2, md: 4 }}
|
||||||
as={ChakraLink}
|
as={ChakraLink}
|
||||||
justifyContent="start"
|
justifyContent="start"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
@@ -141,10 +132,12 @@ export default function AppShell({
|
|||||||
children,
|
children,
|
||||||
title,
|
title,
|
||||||
requireAuth,
|
requireAuth,
|
||||||
|
requireBeta,
|
||||||
}: {
|
}: {
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
title?: string;
|
title?: string;
|
||||||
requireAuth?: boolean;
|
requireAuth?: boolean;
|
||||||
|
requireBeta?: boolean;
|
||||||
}) {
|
}) {
|
||||||
const [vh, setVh] = useState("100vh"); // Default height to prevent flicker on initial render
|
const [vh, setVh] = useState("100vh"); // Default height to prevent flicker on initial render
|
||||||
|
|
||||||
@@ -174,15 +167,21 @@ export default function AppShell({
|
|||||||
}
|
}
|
||||||
}, [requireAuth, user, authLoading]);
|
}, [requireAuth, user, authLoading]);
|
||||||
|
|
||||||
|
const flags = useAppStore((s) => s.featureFlags.featureFlags);
|
||||||
|
const flagsLoaded = useAppStore((s) => s.featureFlags.flagsLoaded);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Flex h={vh} w="100vw">
|
<>
|
||||||
<Head>
|
<Flex h={vh} w="100vw">
|
||||||
<title>{title ? `${title} | OpenPipe` : "OpenPipe"}</title>
|
<Head>
|
||||||
</Head>
|
<title>{title ? `${title} | OpenPipe` : "OpenPipe"}</title>
|
||||||
<NavSidebar />
|
</Head>
|
||||||
<Box h="100%" flex={1} overflowY="auto" bgColor="gray.50">
|
<NavSidebar />
|
||||||
{children}
|
<Box h="100%" flex={1} overflowY="auto" bgColor="gray.50">
|
||||||
</Box>
|
{children}
|
||||||
</Flex>
|
</Box>
|
||||||
|
</Flex>
|
||||||
|
{requireBeta && flagsLoaded && !flags.betaAccess && <BetaModal />}
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
67
app/src/components/nav/BetaModal.tsx
Normal file
67
app/src/components/nav/BetaModal.tsx
Normal file
@@ -0,0 +1,67 @@
|
|||||||
|
import {
|
||||||
|
Button,
|
||||||
|
Modal,
|
||||||
|
ModalBody,
|
||||||
|
ModalContent,
|
||||||
|
ModalFooter,
|
||||||
|
ModalHeader,
|
||||||
|
ModalOverlay,
|
||||||
|
VStack,
|
||||||
|
Text,
|
||||||
|
HStack,
|
||||||
|
Icon,
|
||||||
|
Link,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { BsStars } from "react-icons/bs";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { useSession } from "next-auth/react";
|
||||||
|
|
||||||
|
export const BetaModal = () => {
|
||||||
|
const router = useRouter();
|
||||||
|
const session = useSession();
|
||||||
|
|
||||||
|
const email = session.data?.user.email ?? "";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
isOpen
|
||||||
|
onClose={router.back}
|
||||||
|
closeOnOverlayClick={false}
|
||||||
|
size={{ base: "xl", md: "2xl" }}
|
||||||
|
>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={BsStars} />
|
||||||
|
<Text>Beta-Only Feature</Text>
|
||||||
|
</HStack>
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalBody maxW="unset">
|
||||||
|
<VStack spacing={8} py={4} alignItems="flex-start">
|
||||||
|
<Text fontSize="md">
|
||||||
|
This feature is currently in beta. To receive early access to beta-only features, join
|
||||||
|
the waitlist. You'll receive an email at <b>{email}</b> when you're approved.
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter>
|
||||||
|
<HStack spacing={4}>
|
||||||
|
<Button
|
||||||
|
as={Link}
|
||||||
|
textDecoration="none !important"
|
||||||
|
colorScheme="orange"
|
||||||
|
target="_blank"
|
||||||
|
href={`https://ax3nafkw0jp.typeform.com/to/ZNpYqvAc#email=${email}`}
|
||||||
|
>
|
||||||
|
Join Waitlist
|
||||||
|
</Button>
|
||||||
|
<Button colorScheme="blue" onClick={router.back}>
|
||||||
|
Done
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -14,8 +14,9 @@ import {
|
|||||||
Link as ChakraLink,
|
Link as ChakraLink,
|
||||||
Image,
|
Image,
|
||||||
Box,
|
Box,
|
||||||
|
Portal,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import React, { useEffect, useState } from "react";
|
import { useEffect } from "react";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { BsPlus, BsPersonCircle } from "react-icons/bs";
|
import { BsPlus, BsPersonCircle } from "react-icons/bs";
|
||||||
import { type Project } from "@prisma/client";
|
import { type Project } from "@prisma/client";
|
||||||
@@ -67,7 +68,13 @@ export default function ProjectMenu() {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<VStack w="full" alignItems="flex-start" spacing={0} py={1}>
|
<VStack
|
||||||
|
w="full"
|
||||||
|
alignItems="flex-start"
|
||||||
|
spacing={0}
|
||||||
|
py={1}
|
||||||
|
zIndex={popover.isOpen ? "dropdown" : undefined}
|
||||||
|
>
|
||||||
<Popover
|
<Popover
|
||||||
placement="bottom"
|
placement="bottom"
|
||||||
isOpen={popover.isOpen}
|
isOpen={popover.isOpen}
|
||||||
@@ -103,64 +110,66 @@ export default function ProjectMenu() {
|
|||||||
</HStack>
|
</HStack>
|
||||||
</NavSidebarOption>
|
</NavSidebarOption>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent
|
<Portal>
|
||||||
_focusVisible={{ outline: "unset" }}
|
<PopoverContent
|
||||||
ml={-1}
|
_focusVisible={{ outline: "unset" }}
|
||||||
w={224}
|
w={220}
|
||||||
boxShadow="0 0 40px 4px rgba(0, 0, 0, 0.1);"
|
ml={{ base: 2, md: 0 }}
|
||||||
fontSize="sm"
|
boxShadow="0 0 40px 4px rgba(0, 0, 0, 0.1);"
|
||||||
>
|
fontSize="sm"
|
||||||
<VStack alignItems="flex-start" spacing={1} py={1}>
|
>
|
||||||
<Text px={3} py={2}>
|
<VStack alignItems="flex-start" spacing={1} py={1}>
|
||||||
{user?.user.email}
|
<Text px={3} py={2}>
|
||||||
</Text>
|
{user?.user.email}
|
||||||
<Divider />
|
</Text>
|
||||||
<Text alignSelf="flex-start" fontWeight="bold" px={3} pt={2}>
|
<Divider />
|
||||||
Your Projects
|
<Text alignSelf="flex-start" fontWeight="bold" px={3} pt={2}>
|
||||||
</Text>
|
Your Projects
|
||||||
<VStack spacing={0} w="full" px={1}>
|
</Text>
|
||||||
{projects?.map((proj) => (
|
<VStack spacing={0} w="full" px={1}>
|
||||||
<ProjectOption
|
{projects?.map((proj) => (
|
||||||
key={proj.id}
|
<ProjectOption
|
||||||
proj={proj}
|
key={proj.id}
|
||||||
isActive={proj.id === selectedProjectId}
|
proj={proj}
|
||||||
onClose={popover.onClose}
|
isActive={proj.id === selectedProjectId}
|
||||||
/>
|
onClose={popover.onClose}
|
||||||
))}
|
/>
|
||||||
<HStack
|
))}
|
||||||
as={Button}
|
<HStack
|
||||||
variant="ghost"
|
as={Button}
|
||||||
colorScheme="blue"
|
variant="ghost"
|
||||||
color="blue.400"
|
colorScheme="blue"
|
||||||
fontSize="sm"
|
color="blue.400"
|
||||||
justifyContent="flex-start"
|
fontSize="sm"
|
||||||
onClick={createProject}
|
justifyContent="flex-start"
|
||||||
w="full"
|
onClick={createProject}
|
||||||
borderRadius={4}
|
w="full"
|
||||||
spacing={0}
|
borderRadius={4}
|
||||||
>
|
spacing={0}
|
||||||
<Text>Add project</Text>
|
>
|
||||||
<Icon as={isLoading ? Spinner : BsPlus} boxSize={4} strokeWidth={0.5} />
|
<Text>Add project</Text>
|
||||||
</HStack>
|
<Icon as={isLoading ? Spinner : BsPlus} boxSize={4} strokeWidth={0.5} />
|
||||||
</VStack>
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
<Divider />
|
<Divider />
|
||||||
<VStack w="full" px={1}>
|
<VStack w="full" px={1}>
|
||||||
<ChakraLink
|
<ChakraLink
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
signOut().catch(console.error);
|
signOut().catch(console.error);
|
||||||
}}
|
}}
|
||||||
_hover={{ bgColor: "gray.200", textDecoration: "none" }}
|
_hover={{ bgColor: "gray.200", textDecoration: "none" }}
|
||||||
w="full"
|
w="full"
|
||||||
py={2}
|
py={2}
|
||||||
px={2}
|
px={2}
|
||||||
borderRadius={4}
|
borderRadius={4}
|
||||||
>
|
>
|
||||||
<Text>Sign out</Text>
|
<Text>Sign out</Text>
|
||||||
</ChakraLink>
|
</ChakraLink>
|
||||||
|
</VStack>
|
||||||
</VStack>
|
</VStack>
|
||||||
</VStack>
|
</PopoverContent>
|
||||||
</PopoverContent>
|
</Portal>
|
||||||
</Popover>
|
</Popover>
|
||||||
</VStack>
|
</VStack>
|
||||||
);
|
);
|
||||||
@@ -176,7 +185,6 @@ const ProjectOption = ({
|
|||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
const setSelectedProjectId = useAppStore((s) => s.setSelectedProjectId);
|
const setSelectedProjectId = useAppStore((s) => s.setSelectedProjectId);
|
||||||
const [gearHovered, setGearHovered] = useState(false);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HStack
|
<HStack
|
||||||
@@ -188,8 +196,8 @@ const ProjectOption = ({
|
|||||||
}}
|
}}
|
||||||
w="full"
|
w="full"
|
||||||
justifyContent="space-between"
|
justifyContent="space-between"
|
||||||
_hover={gearHovered ? undefined : { bgColor: "gray.200", textDecoration: "none" }}
|
_hover={{ bgColor: "gray.200", textDecoration: "none" }}
|
||||||
color={isActive ? "blue.400" : undefined}
|
bgColor={isActive ? "gray.100" : undefined}
|
||||||
py={2}
|
py={2}
|
||||||
px={4}
|
px={4}
|
||||||
borderRadius={4}
|
borderRadius={4}
|
||||||
|
|||||||
@@ -23,50 +23,48 @@ export default function UserMenu({ user, ...rest }: { user: Session } & StackPro
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<Popover placement="right">
|
||||||
<Popover placement="right">
|
<PopoverTrigger>
|
||||||
<PopoverTrigger>
|
<NavSidebarOption>
|
||||||
<NavSidebarOption>
|
<HStack
|
||||||
<HStack
|
// Weird values to make mobile look right; can clean up when we make the sidebar disappear on mobile
|
||||||
// Weird values to make mobile look right; can clean up when we make the sidebar disappear on mobile
|
py={2}
|
||||||
py={2}
|
px={1}
|
||||||
px={1}
|
spacing={3}
|
||||||
spacing={3}
|
{...rest}
|
||||||
{...rest}
|
>
|
||||||
>
|
{profileImage}
|
||||||
{profileImage}
|
<VStack spacing={0} align="start" flex={1} flexShrink={1}>
|
||||||
<VStack spacing={0} align="start" flex={1} flexShrink={1}>
|
<Text fontWeight="bold" fontSize="sm">
|
||||||
<Text fontWeight="bold" fontSize="sm">
|
{user.user.name}
|
||||||
{user.user.name}
|
</Text>
|
||||||
</Text>
|
<Text color="gray.500" fontSize="xs">
|
||||||
<Text color="gray.500" fontSize="xs">
|
{/* {user.user.email} */}
|
||||||
{/* {user.user.email} */}
|
</Text>
|
||||||
</Text>
|
</VStack>
|
||||||
</VStack>
|
<Icon as={BsChevronRight} boxSize={4} color="gray.500" />
|
||||||
<Icon as={BsChevronRight} boxSize={4} color="gray.500" />
|
</HStack>
|
||||||
</HStack>
|
</NavSidebarOption>
|
||||||
</NavSidebarOption>
|
</PopoverTrigger>
|
||||||
</PopoverTrigger>
|
<PopoverContent _focusVisible={{ outline: "unset" }} ml={-1} minW={48} w="full">
|
||||||
<PopoverContent _focusVisible={{ outline: "unset" }} ml={-1} minW={48} w="full">
|
<VStack align="stretch" spacing={0}>
|
||||||
<VStack align="stretch" spacing={0}>
|
{/* sign out */}
|
||||||
{/* sign out */}
|
<HStack
|
||||||
<HStack
|
as={Link}
|
||||||
as={Link}
|
onClick={() => {
|
||||||
onClick={() => {
|
signOut().catch(console.error);
|
||||||
signOut().catch(console.error);
|
}}
|
||||||
}}
|
px={4}
|
||||||
px={4}
|
py={2}
|
||||||
py={2}
|
spacing={4}
|
||||||
spacing={4}
|
color="gray.500"
|
||||||
color="gray.500"
|
fontSize="sm"
|
||||||
fontSize="sm"
|
>
|
||||||
>
|
<Icon as={BsBoxArrowRight} boxSize={6} />
|
||||||
<Icon as={BsBoxArrowRight} boxSize={6} />
|
<Text>Sign out</Text>
|
||||||
<Text>Sign out</Text>
|
</HStack>
|
||||||
</HStack>
|
</VStack>
|
||||||
</VStack>
|
</PopoverContent>
|
||||||
</PopoverContent>
|
</Popover>
|
||||||
</Popover>
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
128
app/src/components/projectSettings/InviteMemberModal.tsx
Normal file
128
app/src/components/projectSettings/InviteMemberModal.tsx
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
import {
|
||||||
|
Button,
|
||||||
|
FormControl,
|
||||||
|
FormLabel,
|
||||||
|
Input,
|
||||||
|
FormHelperText,
|
||||||
|
HStack,
|
||||||
|
Modal,
|
||||||
|
ModalBody,
|
||||||
|
ModalCloseButton,
|
||||||
|
ModalContent,
|
||||||
|
ModalFooter,
|
||||||
|
ModalHeader,
|
||||||
|
ModalOverlay,
|
||||||
|
Spinner,
|
||||||
|
Text,
|
||||||
|
VStack,
|
||||||
|
RadioGroup,
|
||||||
|
Radio,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { useState, useEffect } from "react";
|
||||||
|
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useHandledAsyncCallback, useSelectedProject } from "~/utils/hooks";
|
||||||
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
|
import { type ProjectUserRole } from "@prisma/client";
|
||||||
|
|
||||||
|
export const InviteMemberModal = ({
|
||||||
|
isOpen,
|
||||||
|
onClose,
|
||||||
|
}: {
|
||||||
|
isOpen: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
}) => {
|
||||||
|
const selectedProject = useSelectedProject().data;
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const [email, setEmail] = useState("");
|
||||||
|
const [role, setRole] = useState<ProjectUserRole>("MEMBER");
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setEmail("");
|
||||||
|
setRole("MEMBER");
|
||||||
|
}, [isOpen]);
|
||||||
|
|
||||||
|
const emailIsValid = !email || !email.match(/.+@.+\..+/);
|
||||||
|
|
||||||
|
const inviteMemberMutation = api.users.inviteToProject.useMutation();
|
||||||
|
|
||||||
|
const [inviteMember, isInviting] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!selectedProject?.id || !role) return;
|
||||||
|
const resp = await inviteMemberMutation.mutateAsync({
|
||||||
|
projectId: selectedProject.id,
|
||||||
|
email,
|
||||||
|
role,
|
||||||
|
});
|
||||||
|
if (maybeReportError(resp)) return;
|
||||||
|
await utils.projects.get.invalidate();
|
||||||
|
onClose();
|
||||||
|
}, [inviteMemberMutation, email, role, selectedProject?.id, onClose]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal isOpen={isOpen} onClose={onClose}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Text>Invite Member</Text>
|
||||||
|
</HStack>
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalCloseButton />
|
||||||
|
<ModalBody>
|
||||||
|
<VStack spacing={8} alignItems="flex-start">
|
||||||
|
<Text>
|
||||||
|
Invite a new member to <b>{selectedProject?.name}</b>.
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<RadioGroup
|
||||||
|
value={role}
|
||||||
|
onChange={(e) => setRole(e as ProjectUserRole)}
|
||||||
|
colorScheme="orange"
|
||||||
|
>
|
||||||
|
<VStack w="full" alignItems="flex-start">
|
||||||
|
<Radio value="MEMBER">
|
||||||
|
<Text fontSize="sm">MEMBER</Text>
|
||||||
|
</Radio>
|
||||||
|
<Radio value="ADMIN">
|
||||||
|
<Text fontSize="sm">ADMIN</Text>
|
||||||
|
</Radio>
|
||||||
|
</VStack>
|
||||||
|
</RadioGroup>
|
||||||
|
<FormControl>
|
||||||
|
<FormLabel>Email</FormLabel>
|
||||||
|
<Input
|
||||||
|
type="email"
|
||||||
|
value={email}
|
||||||
|
onChange={(e) => setEmail(e.target.value)}
|
||||||
|
onKeyDown={(e) => {
|
||||||
|
if (e.key === "Enter" && (e.metaKey || e.ctrlKey || e.shiftKey)) {
|
||||||
|
e.preventDefault();
|
||||||
|
e.currentTarget.blur();
|
||||||
|
inviteMember();
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<FormHelperText>Enter the email of the person you want to invite.</FormHelperText>
|
||||||
|
</FormControl>
|
||||||
|
</VStack>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter mt={4}>
|
||||||
|
<HStack>
|
||||||
|
<Button colorScheme="gray" onClick={onClose} minW={24}>
|
||||||
|
<Text>Cancel</Text>
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
colorScheme="orange"
|
||||||
|
onClick={inviteMember}
|
||||||
|
minW={24}
|
||||||
|
isDisabled={emailIsValid || isInviting}
|
||||||
|
>
|
||||||
|
{isInviting ? <Spinner boxSize={4} /> : <Text>Send Invitation</Text>}
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
145
app/src/components/projectSettings/MemberTable.tsx
Normal file
145
app/src/components/projectSettings/MemberTable.tsx
Normal file
@@ -0,0 +1,145 @@
|
|||||||
|
import { useMemo, useState } from "react";
|
||||||
|
import {
|
||||||
|
Table,
|
||||||
|
Thead,
|
||||||
|
Tr,
|
||||||
|
Th,
|
||||||
|
Tbody,
|
||||||
|
Td,
|
||||||
|
IconButton,
|
||||||
|
useDisclosure,
|
||||||
|
Text,
|
||||||
|
Button,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { useSession } from "next-auth/react";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
import { type User } from "@prisma/client";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback, useSelectedProject } from "~/utils/hooks";
|
||||||
|
import { InviteMemberModal } from "./InviteMemberModal";
|
||||||
|
import { RemoveMemberDialog } from "./RemoveMemberDialog";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
|
|
||||||
|
const MemberTable = () => {
|
||||||
|
const selectedProject = useSelectedProject().data;
|
||||||
|
const session = useSession().data;
|
||||||
|
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const [memberToRemove, setMemberToRemove] = useState<User | null>(null);
|
||||||
|
const inviteMemberModal = useDisclosure();
|
||||||
|
|
||||||
|
const cancelInvitationMutation = api.users.cancelProjectInvitation.useMutation();
|
||||||
|
|
||||||
|
const [cancelInvitation, isCancelling] = useHandledAsyncCallback(
|
||||||
|
async (invitationToken: string) => {
|
||||||
|
if (!selectedProject?.id) return;
|
||||||
|
const resp = await cancelInvitationMutation.mutateAsync({
|
||||||
|
invitationToken,
|
||||||
|
});
|
||||||
|
if (maybeReportError(resp)) return;
|
||||||
|
await utils.projects.get.invalidate();
|
||||||
|
},
|
||||||
|
[selectedProject?.id, cancelInvitationMutation],
|
||||||
|
);
|
||||||
|
|
||||||
|
const sortedMembers = useMemo(() => {
|
||||||
|
if (!selectedProject?.projectUsers) return [];
|
||||||
|
return selectedProject.projectUsers.sort((a, b) => {
|
||||||
|
if (a.role === b.role) return a.createdAt < b.createdAt ? -1 : 1;
|
||||||
|
// Take advantage of fact that ADMIN is alphabetically before MEMBER
|
||||||
|
return a.role < b.role ? -1 : 1;
|
||||||
|
});
|
||||||
|
}, [selectedProject?.projectUsers]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Table fontSize={{ base: "sm", md: "md" }}>
|
||||||
|
<Thead
|
||||||
|
sx={{
|
||||||
|
th: {
|
||||||
|
base: { px: 0 },
|
||||||
|
md: { px: 6 },
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Tr>
|
||||||
|
<Th>Name</Th>
|
||||||
|
<Th display={{ base: "none", md: "table-cell" }}>Email</Th>
|
||||||
|
<Th>Role</Th>
|
||||||
|
{selectedProject?.role === "ADMIN" && <Th />}
|
||||||
|
</Tr>
|
||||||
|
</Thead>
|
||||||
|
<Tbody
|
||||||
|
sx={{
|
||||||
|
td: {
|
||||||
|
base: { px: 0 },
|
||||||
|
md: { px: 6 },
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{selectedProject &&
|
||||||
|
sortedMembers.map((member) => {
|
||||||
|
return (
|
||||||
|
<Tr key={member.id}>
|
||||||
|
<Td>
|
||||||
|
<Text fontWeight="bold">{member.user.name}</Text>
|
||||||
|
</Td>
|
||||||
|
<Td display={{ base: "none", md: "table-cell" }} h="full">
|
||||||
|
{member.user.email}
|
||||||
|
</Td>
|
||||||
|
<Td fontSize={{ base: "xs", md: "sm" }}>{member.role}</Td>
|
||||||
|
{selectedProject.role === "ADMIN" && (
|
||||||
|
<Td textAlign="end">
|
||||||
|
{member.user.id !== session?.user?.id &&
|
||||||
|
member.user.id !== selectedProject.personalProjectUserId && (
|
||||||
|
<IconButton
|
||||||
|
aria-label="Remove member"
|
||||||
|
colorScheme="red"
|
||||||
|
icon={<BsTrash />}
|
||||||
|
onClick={() => setMemberToRemove(member.user)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Td>
|
||||||
|
)}
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
{selectedProject?.projectUserInvitations?.map((invitation) => {
|
||||||
|
return (
|
||||||
|
<Tr key={invitation.id}>
|
||||||
|
<Td>
|
||||||
|
<Text as="i">Invitation pending</Text>
|
||||||
|
</Td>
|
||||||
|
<Td>{invitation.email}</Td>
|
||||||
|
<Td fontSize="sm">{invitation.role}</Td>
|
||||||
|
{selectedProject.role === "ADMIN" && (
|
||||||
|
<Td textAlign="end">
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
colorScheme="red"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => cancelInvitation(invitation.invitationToken)}
|
||||||
|
isLoading={isCancelling}
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
</Td>
|
||||||
|
)}
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</Tbody>
|
||||||
|
</Table>
|
||||||
|
<InviteMemberModal isOpen={inviteMemberModal.isOpen} onClose={inviteMemberModal.onClose} />
|
||||||
|
<RemoveMemberDialog
|
||||||
|
member={memberToRemove}
|
||||||
|
isOpen={!!memberToRemove}
|
||||||
|
onClose={() => setMemberToRemove(null)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default MemberTable;
|
||||||
71
app/src/components/projectSettings/RemoveMemberDialog.tsx
Normal file
71
app/src/components/projectSettings/RemoveMemberDialog.tsx
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
import {
|
||||||
|
Button,
|
||||||
|
AlertDialog,
|
||||||
|
AlertDialogBody,
|
||||||
|
AlertDialogFooter,
|
||||||
|
AlertDialogHeader,
|
||||||
|
AlertDialogContent,
|
||||||
|
AlertDialogOverlay,
|
||||||
|
Text,
|
||||||
|
VStack,
|
||||||
|
Spinner,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { type User } from "@prisma/client";
|
||||||
|
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { useRef } from "react";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useHandledAsyncCallback, useSelectedProject } from "~/utils/hooks";
|
||||||
|
|
||||||
|
export const RemoveMemberDialog = ({
|
||||||
|
isOpen,
|
||||||
|
onClose,
|
||||||
|
member,
|
||||||
|
}: {
|
||||||
|
isOpen: boolean;
|
||||||
|
onClose: () => void;
|
||||||
|
member: User | null;
|
||||||
|
}) => {
|
||||||
|
const selectedProject = useSelectedProject();
|
||||||
|
const removeUserMutation = api.users.removeUserFromProject.useMutation();
|
||||||
|
const utils = api.useContext();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const cancelRef = useRef<HTMLButtonElement>(null);
|
||||||
|
|
||||||
|
const [onRemoveConfirm, isRemoving] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!selectedProject.data?.id || !member?.id) return;
|
||||||
|
await removeUserMutation.mutateAsync({ projectId: selectedProject.data.id, userId: member.id });
|
||||||
|
await utils.projects.get.invalidate();
|
||||||
|
onClose();
|
||||||
|
}, [removeUserMutation, selectedProject, router]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<AlertDialog isOpen={isOpen} leastDestructiveRef={cancelRef} onClose={onClose}>
|
||||||
|
<AlertDialogOverlay>
|
||||||
|
<AlertDialogContent>
|
||||||
|
<AlertDialogHeader fontSize="lg" fontWeight="bold">
|
||||||
|
Remove Member
|
||||||
|
</AlertDialogHeader>
|
||||||
|
|
||||||
|
<AlertDialogBody>
|
||||||
|
<VStack spacing={4} alignItems="flex-start">
|
||||||
|
<Text>
|
||||||
|
Are you sure you want to remove <b>{member?.name}</b> from the project?
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
</AlertDialogBody>
|
||||||
|
|
||||||
|
<AlertDialogFooter>
|
||||||
|
<Button ref={cancelRef} onClick={onClose}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button colorScheme="red" onClick={onRemoveConfirm} ml={3} w={20}>
|
||||||
|
{isRemoving ? <Spinner /> : "Remove"}
|
||||||
|
</Button>
|
||||||
|
</AlertDialogFooter>
|
||||||
|
</AlertDialogContent>
|
||||||
|
</AlertDialogOverlay>
|
||||||
|
</AlertDialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -21,7 +21,7 @@ const ActionButton = ({
|
|||||||
>
|
>
|
||||||
<HStack spacing={1}>
|
<HStack spacing={1}>
|
||||||
{icon && <Icon as={icon} />}
|
{icon && <Icon as={icon} />}
|
||||||
<Text>{label}</Text>
|
<Text display={{ base: "none", md: "flex" }}>{label}</Text>
|
||||||
</HStack>
|
</HStack>
|
||||||
</Button>
|
</Button>
|
||||||
);
|
);
|
||||||
|
|||||||
117
app/src/components/requestLogs/ColumnVisiblityDropdown.tsx
Normal file
117
app/src/components/requestLogs/ColumnVisiblityDropdown.tsx
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
import {
|
||||||
|
Icon,
|
||||||
|
Popover,
|
||||||
|
PopoverTrigger,
|
||||||
|
PopoverContent,
|
||||||
|
VStack,
|
||||||
|
HStack,
|
||||||
|
Button,
|
||||||
|
Text,
|
||||||
|
useDisclosure,
|
||||||
|
Box,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { BiCheck } from "react-icons/bi";
|
||||||
|
import { BsToggles } from "react-icons/bs";
|
||||||
|
import { useMemo } from "react";
|
||||||
|
|
||||||
|
import { useIsClientRehydrated, useTagNames } from "~/utils/hooks";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { StaticColumnKeys } from "~/state/columnVisiblitySlice";
|
||||||
|
import ActionButton from "./ActionButton";
|
||||||
|
|
||||||
|
const ColumnVisiblityDropdown = () => {
|
||||||
|
const tagNames = useTagNames().data;
|
||||||
|
|
||||||
|
const visibleColumns = useAppStore((s) => s.columnVisibility.visibleColumns);
|
||||||
|
const toggleColumnVisibility = useAppStore((s) => s.columnVisibility.toggleColumnVisibility);
|
||||||
|
const totalColumns = Object.keys(StaticColumnKeys).length + (tagNames?.length ?? 0);
|
||||||
|
|
||||||
|
const popover = useDisclosure();
|
||||||
|
|
||||||
|
const columnVisiblityOptions = useMemo(() => {
|
||||||
|
const options: { label: string; key: string }[] = [
|
||||||
|
{
|
||||||
|
label: "Sent At",
|
||||||
|
key: StaticColumnKeys.SENT_AT,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Model",
|
||||||
|
key: StaticColumnKeys.MODEL,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Duration",
|
||||||
|
key: StaticColumnKeys.DURATION,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Input Tokens",
|
||||||
|
key: StaticColumnKeys.INPUT_TOKENS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Output Tokens",
|
||||||
|
key: StaticColumnKeys.OUTPUT_TOKENS,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Status Code",
|
||||||
|
key: StaticColumnKeys.STATUS_CODE,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
for (const tagName of tagNames ?? []) {
|
||||||
|
options.push({
|
||||||
|
label: tagName,
|
||||||
|
key: tagName,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return options;
|
||||||
|
}, [tagNames]);
|
||||||
|
|
||||||
|
const isClientRehydrated = useIsClientRehydrated();
|
||||||
|
if (!isClientRehydrated) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Popover
|
||||||
|
placement="bottom-start"
|
||||||
|
isOpen={popover.isOpen}
|
||||||
|
onOpen={popover.onOpen}
|
||||||
|
onClose={popover.onClose}
|
||||||
|
>
|
||||||
|
<PopoverTrigger>
|
||||||
|
<Box>
|
||||||
|
<ActionButton
|
||||||
|
label={`Columns (${visibleColumns.size}/${totalColumns})`}
|
||||||
|
icon={BsToggles}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent boxShadow="0 0 40px 4px rgba(0, 0, 0, 0.1);" minW={0} w="auto">
|
||||||
|
<VStack spacing={0} maxH={400} overflowY="auto">
|
||||||
|
{columnVisiblityOptions?.map((option, index) => (
|
||||||
|
<HStack
|
||||||
|
key={index}
|
||||||
|
as={Button}
|
||||||
|
onClick={() => toggleColumnVisibility(option.key)}
|
||||||
|
w="full"
|
||||||
|
minH={10}
|
||||||
|
variant="ghost"
|
||||||
|
justifyContent="space-between"
|
||||||
|
fontWeight="semibold"
|
||||||
|
borderRadius={0}
|
||||||
|
colorScheme="blue"
|
||||||
|
color="black"
|
||||||
|
fontSize="sm"
|
||||||
|
borderBottomWidth={1}
|
||||||
|
>
|
||||||
|
<Text mr={16}>{option.label}</Text>
|
||||||
|
<Box w={5}>
|
||||||
|
{visibleColumns.has(option.key) && (
|
||||||
|
<Icon as={BiCheck} color="blue.500" boxSize={5} />
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
</HStack>
|
||||||
|
))}
|
||||||
|
</VStack>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ColumnVisiblityDropdown;
|
||||||
210
app/src/components/requestLogs/ExportButton.tsx
Normal file
210
app/src/components/requestLogs/ExportButton.tsx
Normal file
@@ -0,0 +1,210 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
|
import {
|
||||||
|
Modal,
|
||||||
|
ModalOverlay,
|
||||||
|
ModalContent,
|
||||||
|
ModalHeader,
|
||||||
|
ModalCloseButton,
|
||||||
|
ModalBody,
|
||||||
|
ModalFooter,
|
||||||
|
HStack,
|
||||||
|
VStack,
|
||||||
|
Icon,
|
||||||
|
Text,
|
||||||
|
Button,
|
||||||
|
Checkbox,
|
||||||
|
NumberInput,
|
||||||
|
NumberInputField,
|
||||||
|
NumberInputStepper,
|
||||||
|
NumberIncrementStepper,
|
||||||
|
NumberDecrementStepper,
|
||||||
|
Collapse,
|
||||||
|
Flex,
|
||||||
|
useDisclosure,
|
||||||
|
type UseDisclosureReturn,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { BiExport } from "react-icons/bi";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import ActionButton from "./ActionButton";
|
||||||
|
import InputDropdown from "../InputDropdown";
|
||||||
|
import { FiChevronUp, FiChevronDown } from "react-icons/fi";
|
||||||
|
import InfoCircle from "../InfoCircle";
|
||||||
|
|
||||||
|
const SUPPORTED_EXPORT_FORMATS = ["alpaca-finetune", "openai-fine-tune", "unformatted"];
|
||||||
|
|
||||||
|
const ExportButton = () => {
|
||||||
|
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ActionButton
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
label="Export"
|
||||||
|
icon={BiExport}
|
||||||
|
isDisabled={selectedLogIds.size === 0}
|
||||||
|
/>
|
||||||
|
<ExportLogsModal disclosure={disclosure} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ExportButton;
|
||||||
|
|
||||||
|
const ExportLogsModal = ({ disclosure }: { disclosure: UseDisclosureReturn }) => {
|
||||||
|
const selectedProjectId = useAppStore((s) => s.selectedProjectId);
|
||||||
|
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
||||||
|
const clearSelectedLogIds = useAppStore((s) => s.selectedLogs.clearSelectedLogIds);
|
||||||
|
|
||||||
|
const [selectedExportFormat, setSelectedExportFormat] = useState(SUPPORTED_EXPORT_FORMATS[0]);
|
||||||
|
const [testingSplit, setTestingSplit] = useState(10);
|
||||||
|
const [removeDuplicates, setRemoveDuplicates] = useState(true);
|
||||||
|
const [showAdvancedOptions, setShowAdvancedOptions] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (disclosure.isOpen) {
|
||||||
|
setSelectedExportFormat(SUPPORTED_EXPORT_FORMATS[0]);
|
||||||
|
setTestingSplit(10);
|
||||||
|
setRemoveDuplicates(true);
|
||||||
|
}
|
||||||
|
}, [disclosure.isOpen]);
|
||||||
|
|
||||||
|
const exportLogsMutation = api.loggedCalls.export.useMutation();
|
||||||
|
|
||||||
|
const [exportLogs, exportInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!selectedProjectId || !selectedLogIds.size || !testingSplit || !selectedExportFormat)
|
||||||
|
return;
|
||||||
|
const response = await exportLogsMutation.mutateAsync({
|
||||||
|
projectId: selectedProjectId,
|
||||||
|
selectedLogIds: Array.from(selectedLogIds),
|
||||||
|
testingSplit,
|
||||||
|
selectedExportFormat,
|
||||||
|
removeDuplicates,
|
||||||
|
});
|
||||||
|
|
||||||
|
const dataUrl = `data:application/pdf;base64,${response}`;
|
||||||
|
const blob = await fetch(dataUrl).then((res) => res.blob());
|
||||||
|
const url = URL.createObjectURL(blob);
|
||||||
|
const a = document.createElement("a");
|
||||||
|
|
||||||
|
a.href = url;
|
||||||
|
a.download = `data.zip`;
|
||||||
|
document.body.appendChild(a);
|
||||||
|
a.click();
|
||||||
|
document.body.removeChild(a);
|
||||||
|
|
||||||
|
disclosure.onClose();
|
||||||
|
clearSelectedLogIds();
|
||||||
|
}, [
|
||||||
|
exportLogsMutation,
|
||||||
|
selectedProjectId,
|
||||||
|
selectedLogIds,
|
||||||
|
testingSplit,
|
||||||
|
selectedExportFormat,
|
||||||
|
removeDuplicates,
|
||||||
|
]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal size={{ base: "xl", md: "2xl" }} {...disclosure}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={BiExport} />
|
||||||
|
<Text>Export Logs</Text>
|
||||||
|
</HStack>
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalCloseButton />
|
||||||
|
<ModalBody maxW="unset">
|
||||||
|
<VStack w="full" spacing={8} pt={4} alignItems="flex-start">
|
||||||
|
<Text>
|
||||||
|
We'll export the <b>{selectedLogIds.size}</b> logs you have selected in the format of
|
||||||
|
your choice.
|
||||||
|
</Text>
|
||||||
|
<VStack alignItems="flex-start" spacing={4}>
|
||||||
|
<Flex
|
||||||
|
flexDir={{ base: "column", md: "row" }}
|
||||||
|
alignItems={{ base: "flex-start", md: "center" }}
|
||||||
|
>
|
||||||
|
<HStack w={48} alignItems="center" spacing={1}>
|
||||||
|
<Text fontWeight="bold">Format:</Text>
|
||||||
|
<InfoCircle tooltipText="Format logs for for fine tuning or export them without formatting." />
|
||||||
|
</HStack>
|
||||||
|
<InputDropdown
|
||||||
|
options={SUPPORTED_EXPORT_FORMATS}
|
||||||
|
selectedOption={selectedExportFormat}
|
||||||
|
onSelect={(option) => setSelectedExportFormat(option)}
|
||||||
|
inputGroupProps={{ w: 48 }}
|
||||||
|
/>
|
||||||
|
</Flex>
|
||||||
|
<Flex
|
||||||
|
flexDir={{ base: "column", md: "row" }}
|
||||||
|
alignItems={{ base: "flex-start", md: "center" }}
|
||||||
|
>
|
||||||
|
<HStack w={48} alignItems="center" spacing={1}>
|
||||||
|
<Text fontWeight="bold">Testing Split:</Text>
|
||||||
|
<InfoCircle tooltipText="The percent of your logs that will be reserved for testing and saved in another file. Logs are split randomly." />
|
||||||
|
</HStack>
|
||||||
|
<HStack>
|
||||||
|
<NumberInput
|
||||||
|
defaultValue={10}
|
||||||
|
onChange={(_, num) => setTestingSplit(num)}
|
||||||
|
min={1}
|
||||||
|
max={100}
|
||||||
|
w={48}
|
||||||
|
>
|
||||||
|
<NumberInputField />
|
||||||
|
<NumberInputStepper>
|
||||||
|
<NumberIncrementStepper />
|
||||||
|
<NumberDecrementStepper />
|
||||||
|
</NumberInputStepper>
|
||||||
|
</NumberInput>
|
||||||
|
</HStack>
|
||||||
|
</Flex>
|
||||||
|
</VStack>
|
||||||
|
<VStack alignItems="flex-start" spacing={0}>
|
||||||
|
<Button
|
||||||
|
variant="unstyled"
|
||||||
|
color="blue.600"
|
||||||
|
onClick={() => setShowAdvancedOptions(!showAdvancedOptions)}
|
||||||
|
>
|
||||||
|
<HStack>
|
||||||
|
<Text>Advanced Options</Text>
|
||||||
|
<Icon as={showAdvancedOptions ? FiChevronUp : FiChevronDown} />
|
||||||
|
</HStack>
|
||||||
|
</Button>
|
||||||
|
<Collapse in={showAdvancedOptions} unmountOnExit={true}>
|
||||||
|
<VStack align="stretch" pt={4}>
|
||||||
|
<HStack>
|
||||||
|
<Checkbox
|
||||||
|
colorScheme="blue"
|
||||||
|
isChecked={removeDuplicates}
|
||||||
|
onChange={(e) => setRemoveDuplicates(e.target.checked)}
|
||||||
|
>
|
||||||
|
<Text>Remove duplicates</Text>
|
||||||
|
</Checkbox>
|
||||||
|
<InfoCircle tooltipText="To avoid overfitting and speed up training, automatically deduplicate logs with matching input and output." />
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
</Collapse>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter>
|
||||||
|
<HStack>
|
||||||
|
<Button colorScheme="gray" onClick={disclosure.onClose} minW={24}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button colorScheme="blue" onClick={exportLogs} isLoading={exportInProgress} minW={24}>
|
||||||
|
Export
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
161
app/src/components/requestLogs/FineTuneButton.tsx
Normal file
161
app/src/components/requestLogs/FineTuneButton.tsx
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
|
import {
|
||||||
|
Modal,
|
||||||
|
ModalOverlay,
|
||||||
|
ModalContent,
|
||||||
|
ModalHeader,
|
||||||
|
ModalCloseButton,
|
||||||
|
ModalBody,
|
||||||
|
ModalFooter,
|
||||||
|
HStack,
|
||||||
|
VStack,
|
||||||
|
Icon,
|
||||||
|
Text,
|
||||||
|
Button,
|
||||||
|
useDisclosure,
|
||||||
|
type UseDisclosureReturn,
|
||||||
|
Input,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { FaRobot } from "react-icons/fa";
|
||||||
|
import humanId from "human-id";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import ActionButton from "./ActionButton";
|
||||||
|
import InputDropdown from "../InputDropdown";
|
||||||
|
import { FiChevronDown } from "react-icons/fi";
|
||||||
|
|
||||||
|
const SUPPORTED_BASE_MODELS = ["llama2-7b", "llama2-13b", "llama2-70b", "gpt-3.5-turbo"];
|
||||||
|
|
||||||
|
const FineTuneButton = () => {
|
||||||
|
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ActionButton
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
label="Fine Tune"
|
||||||
|
icon={FaRobot}
|
||||||
|
isDisabled={selectedLogIds.size === 0}
|
||||||
|
/>
|
||||||
|
<FineTuneModal disclosure={disclosure} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FineTuneButton;
|
||||||
|
|
||||||
|
const FineTuneModal = ({ disclosure }: { disclosure: UseDisclosureReturn }) => {
|
||||||
|
const selectedProjectId = useAppStore((s) => s.selectedProjectId);
|
||||||
|
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
||||||
|
const clearSelectedLogIds = useAppStore((s) => s.selectedLogs.clearSelectedLogIds);
|
||||||
|
|
||||||
|
const [selectedBaseModel, setSelectedBaseModel] = useState(SUPPORTED_BASE_MODELS[0]);
|
||||||
|
const [modelSlug, setModelSlug] = useState(humanId({ separator: "-", capitalize: false }));
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (disclosure.isOpen) {
|
||||||
|
setSelectedBaseModel(SUPPORTED_BASE_MODELS[0]);
|
||||||
|
setModelSlug(humanId({ separator: "-", capitalize: false }));
|
||||||
|
}
|
||||||
|
}, [disclosure.isOpen]);
|
||||||
|
|
||||||
|
const utils = api.useContext();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const createFineTuneMutation = api.fineTunes.create.useMutation();
|
||||||
|
|
||||||
|
const [createFineTune, creationInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!selectedProjectId || !modelSlug || !selectedBaseModel || !selectedLogIds.size) return;
|
||||||
|
await createFineTuneMutation.mutateAsync({
|
||||||
|
projectId: selectedProjectId,
|
||||||
|
slug: modelSlug,
|
||||||
|
baseModel: selectedBaseModel,
|
||||||
|
selectedLogIds: Array.from(selectedLogIds),
|
||||||
|
});
|
||||||
|
|
||||||
|
await utils.fineTunes.list.invalidate();
|
||||||
|
await router.push({ pathname: "/fine-tunes" });
|
||||||
|
clearSelectedLogIds();
|
||||||
|
disclosure.onClose();
|
||||||
|
}, [createFineTuneMutation, selectedProjectId, selectedLogIds, modelSlug, selectedBaseModel]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal size={{ base: "xl", md: "2xl" }} {...disclosure}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={FaRobot} />
|
||||||
|
<Text>Fine Tune</Text>
|
||||||
|
</HStack>
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalCloseButton />
|
||||||
|
<ModalBody maxW="unset">
|
||||||
|
<VStack w="full" spacing={8} pt={4} alignItems="flex-start">
|
||||||
|
<Text>
|
||||||
|
We'll train on the <b>{selectedLogIds.size}</b> logs you've selected.
|
||||||
|
</Text>
|
||||||
|
<VStack>
|
||||||
|
<HStack spacing={2} w="full">
|
||||||
|
<Text fontWeight="bold" w={36}>
|
||||||
|
Model ID:
|
||||||
|
</Text>
|
||||||
|
<Input
|
||||||
|
value={modelSlug}
|
||||||
|
onChange={(e) => setModelSlug(e.target.value)}
|
||||||
|
w={48}
|
||||||
|
placeholder="unique-id"
|
||||||
|
onKeyDown={(e) => {
|
||||||
|
// If the user types anything other than a-z, A-Z, or 0-9, replace it with -
|
||||||
|
if (!/[a-zA-Z0-9]/.test(e.key)) {
|
||||||
|
e.preventDefault();
|
||||||
|
setModelSlug((s) => s && `${s}-`);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
|
<HStack spacing={2}>
|
||||||
|
<Text fontWeight="bold" w={36}>
|
||||||
|
Base model:
|
||||||
|
</Text>
|
||||||
|
<InputDropdown
|
||||||
|
options={SUPPORTED_BASE_MODELS}
|
||||||
|
selectedOption={selectedBaseModel}
|
||||||
|
onSelect={(option) => setSelectedBaseModel(option)}
|
||||||
|
inputGroupProps={{ w: 48 }}
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
<Button variant="unstyled" color="blue.600">
|
||||||
|
<HStack>
|
||||||
|
<Text>Advanced Options</Text>
|
||||||
|
<Icon as={FiChevronDown} />
|
||||||
|
</HStack>
|
||||||
|
</Button>
|
||||||
|
</VStack>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter>
|
||||||
|
<HStack>
|
||||||
|
<Button colorScheme="gray" onClick={disclosure.onClose} minW={24}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
colorScheme="blue"
|
||||||
|
onClick={createFineTune}
|
||||||
|
isLoading={creationInProgress}
|
||||||
|
minW={24}
|
||||||
|
isDisabled={!modelSlug}
|
||||||
|
>
|
||||||
|
Start Training
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
import { Button, HStack, Icon, Text } from "@chakra-ui/react";
|
||||||
|
import { BsPlus } from "react-icons/bs";
|
||||||
|
import { comparators, defaultFilterableFields } from "~/state/logFiltersSlice";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
|
||||||
|
const AddFilterButton = () => {
|
||||||
|
const addFilter = useAppStore((s) => s.logFilters.addFilter);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<HStack
|
||||||
|
as={Button}
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() =>
|
||||||
|
addFilter({
|
||||||
|
id: Date.now().toString(),
|
||||||
|
field: defaultFilterableFields[0],
|
||||||
|
comparator: comparators[0],
|
||||||
|
value: "",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
spacing={0}
|
||||||
|
fontSize="sm"
|
||||||
|
>
|
||||||
|
<Icon as={BsPlus} boxSize={5} />
|
||||||
|
<Text>Add Filter</Text>
|
||||||
|
</HStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default AddFilterButton;
|
||||||
44
app/src/components/requestLogs/LogFilters/LogFilter.tsx
Normal file
44
app/src/components/requestLogs/LogFilters/LogFilter.tsx
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import { useCallback, useState } from "react";
|
||||||
|
import { HStack, IconButton, Input } from "@chakra-ui/react";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
|
||||||
|
import { type LogFilter } from "~/state/logFiltersSlice";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { debounce } from "lodash-es";
|
||||||
|
import SelectFieldDropdown from "./SelectFieldDropdown";
|
||||||
|
import SelectComparatorDropdown from "./SelectComparatorDropdown";
|
||||||
|
|
||||||
|
const LogFilter = ({ filter }: { filter: LogFilter }) => {
|
||||||
|
const updateFilter = useAppStore((s) => s.logFilters.updateFilter);
|
||||||
|
const deleteFilter = useAppStore((s) => s.logFilters.deleteFilter);
|
||||||
|
|
||||||
|
const [editedValue, setEditedValue] = useState(filter.value);
|
||||||
|
|
||||||
|
const debouncedUpdateFilter = useCallback(
|
||||||
|
debounce((filter: LogFilter) => updateFilter(filter), 500, {
|
||||||
|
leading: true,
|
||||||
|
}),
|
||||||
|
[updateFilter],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<HStack>
|
||||||
|
<SelectFieldDropdown filter={filter} />
|
||||||
|
<SelectComparatorDropdown filter={filter} />
|
||||||
|
<Input
|
||||||
|
value={editedValue}
|
||||||
|
onChange={(e) => {
|
||||||
|
setEditedValue(e.target.value);
|
||||||
|
debouncedUpdateFilter({ ...filter, value: e.target.value });
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<IconButton
|
||||||
|
aria-label="Delete Filter"
|
||||||
|
icon={<BsTrash />}
|
||||||
|
onClick={() => deleteFilter(filter.id)}
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default LogFilter;
|
||||||
30
app/src/components/requestLogs/LogFilters/LogFilters.tsx
Normal file
30
app/src/components/requestLogs/LogFilters/LogFilters.tsx
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import { VStack, Text } from "@chakra-ui/react";
|
||||||
|
|
||||||
|
import AddFilterButton from "./AddFilterButton";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import LogFilter from "./LogFilter";
|
||||||
|
|
||||||
|
const LogFilters = () => {
|
||||||
|
const filters = useAppStore((s) => s.logFilters.filters);
|
||||||
|
return (
|
||||||
|
<VStack
|
||||||
|
bgColor="white"
|
||||||
|
borderRadius={8}
|
||||||
|
borderWidth={1}
|
||||||
|
w="full"
|
||||||
|
alignItems="flex-start"
|
||||||
|
p={4}
|
||||||
|
spacing={4}
|
||||||
|
>
|
||||||
|
<Text fontWeight="bold" color="gray.500">
|
||||||
|
Filters
|
||||||
|
</Text>
|
||||||
|
{filters.map((filter) => (
|
||||||
|
<LogFilter key={filter.id} filter={filter} />
|
||||||
|
))}
|
||||||
|
<AddFilterButton />
|
||||||
|
</VStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default LogFilters;
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
import { comparators, type LogFilter } from "~/state/logFiltersSlice";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import InputDropdown from "~/components/InputDropdown";
|
||||||
|
|
||||||
|
const SelectComparatorDropdown = ({ filter }: { filter: LogFilter }) => {
|
||||||
|
const updateFilter = useAppStore((s) => s.logFilters.updateFilter);
|
||||||
|
|
||||||
|
const { comparator } = filter;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<InputDropdown
|
||||||
|
options={comparators}
|
||||||
|
selectedOption={comparator}
|
||||||
|
onSelect={(option) => updateFilter({ ...filter, comparator: option })}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SelectComparatorDropdown;
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
import { defaultFilterableFields, type LogFilter } from "~/state/logFiltersSlice";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { useTagNames } from "~/utils/hooks";
|
||||||
|
import InputDropdown from "~/components/InputDropdown";
|
||||||
|
|
||||||
|
const SelectFieldDropdown = ({ filter }: { filter: LogFilter }) => {
|
||||||
|
const tagNames = useTagNames().data;
|
||||||
|
|
||||||
|
const updateFilter = useAppStore((s) => s.logFilters.updateFilter);
|
||||||
|
|
||||||
|
const { field } = filter;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<InputDropdown
|
||||||
|
options={[...defaultFilterableFields, ...(tagNames || [])]}
|
||||||
|
selectedOption={field}
|
||||||
|
onSelect={(option) => updateFilter({ ...filter, field: option })}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default SelectFieldDropdown;
|
||||||
@@ -5,14 +5,14 @@ import { TableHeader, TableRow } from "./TableRow";
|
|||||||
|
|
||||||
export default function LoggedCallsTable() {
|
export default function LoggedCallsTable() {
|
||||||
const [expandedRow, setExpandedRow] = useState<string | null>(null);
|
const [expandedRow, setExpandedRow] = useState<string | null>(null);
|
||||||
const { data: loggedCalls } = useLoggedCalls();
|
const loggedCalls = useLoggedCalls().data;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card width="100%" overflow="hidden">
|
<Card width="100%" overflowX="auto">
|
||||||
<Table>
|
<Table>
|
||||||
<TableHeader showCheckbox />
|
<TableHeader showOptions />
|
||||||
<Tbody>
|
<Tbody>
|
||||||
{loggedCalls?.calls.map((loggedCall) => {
|
{loggedCalls?.calls?.map((loggedCall) => {
|
||||||
return (
|
return (
|
||||||
<TableRow
|
<TableRow
|
||||||
key={loggedCall.id}
|
key={loggedCall.id}
|
||||||
@@ -25,7 +25,7 @@ export default function LoggedCallsTable() {
|
|||||||
setExpandedRow(loggedCall.id);
|
setExpandedRow(loggedCall.id);
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
showCheckbox
|
showOptions
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
|
|||||||
@@ -14,51 +14,64 @@ import {
|
|||||||
Text,
|
Text,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import dayjs from "dayjs";
|
|
||||||
import relativeTime from "dayjs/plugin/relativeTime";
|
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
|
|
||||||
|
import dayjs from "~/utils/dayjs";
|
||||||
import { type RouterOutputs } from "~/utils/api";
|
import { type RouterOutputs } from "~/utils/api";
|
||||||
import { FormattedJson } from "./FormattedJson";
|
import { FormattedJson } from "./FormattedJson";
|
||||||
import { useAppStore } from "~/state/store";
|
import { useAppStore } from "~/state/store";
|
||||||
import { useLoggedCalls } from "~/utils/hooks";
|
import { useIsClientRehydrated, useLoggedCalls, useTagNames } from "~/utils/hooks";
|
||||||
import { useMemo } from "react";
|
import { useMemo } from "react";
|
||||||
|
import { StaticColumnKeys } from "~/state/columnVisiblitySlice";
|
||||||
dayjs.extend(relativeTime);
|
|
||||||
|
|
||||||
type LoggedCall = RouterOutputs["loggedCalls"]["list"]["calls"][0];
|
type LoggedCall = RouterOutputs["loggedCalls"]["list"]["calls"][0];
|
||||||
|
|
||||||
export const TableHeader = ({ showCheckbox }: { showCheckbox?: boolean }) => {
|
export const TableHeader = ({ showOptions }: { showOptions?: boolean }) => {
|
||||||
const matchingLogIds = useLoggedCalls().data?.matchingLogIds;
|
const matchingLogIds = useLoggedCalls().data?.matchingLogIds;
|
||||||
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
||||||
const addAll = useAppStore((s) => s.selectedLogs.addSelectedLogIds);
|
const addAll = useAppStore((s) => s.selectedLogs.addSelectedLogIds);
|
||||||
const clearAll = useAppStore((s) => s.selectedLogs.clearSelectedLogIds);
|
const clearAll = useAppStore((s) => s.selectedLogs.clearSelectedLogIds);
|
||||||
const allSelected = useMemo(() => {
|
const allSelected = useMemo(() => {
|
||||||
if (!matchingLogIds) return false;
|
if (!matchingLogIds || !matchingLogIds.length) return false;
|
||||||
return matchingLogIds.every((id) => selectedLogIds.has(id));
|
return matchingLogIds.every((id) => selectedLogIds.has(id));
|
||||||
}, [selectedLogIds, matchingLogIds]);
|
}, [selectedLogIds, matchingLogIds]);
|
||||||
|
const tagNames = useTagNames().data;
|
||||||
|
const visibleColumns = useAppStore((s) => s.columnVisibility.visibleColumns);
|
||||||
|
const isClientRehydrated = useIsClientRehydrated();
|
||||||
|
if (!isClientRehydrated) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Thead>
|
<Thead>
|
||||||
<Tr>
|
<Tr>
|
||||||
{showCheckbox && (
|
{showOptions && (
|
||||||
<Th>
|
<Th pr={0}>
|
||||||
<HStack w={8}>
|
<HStack minW={16}>
|
||||||
<Checkbox
|
<Checkbox
|
||||||
isChecked={allSelected}
|
isChecked={allSelected}
|
||||||
onChange={() => {
|
onChange={() => {
|
||||||
allSelected ? clearAll() : addAll(matchingLogIds || []);
|
allSelected ? clearAll() : addAll(matchingLogIds || []);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<Text>({selectedLogIds.size})</Text>
|
<Text>
|
||||||
|
({selectedLogIds.size ? `${selectedLogIds.size}/` : ""}
|
||||||
|
{matchingLogIds?.length || 0})
|
||||||
|
</Text>
|
||||||
</HStack>
|
</HStack>
|
||||||
</Th>
|
</Th>
|
||||||
)}
|
)}
|
||||||
<Th>Time</Th>
|
{visibleColumns.has(StaticColumnKeys.SENT_AT) && <Th>Sent At</Th>}
|
||||||
<Th>Model</Th>
|
{visibleColumns.has(StaticColumnKeys.MODEL) && <Th>Model</Th>}
|
||||||
<Th isNumeric>Duration</Th>
|
{tagNames
|
||||||
<Th isNumeric>Input tokens</Th>
|
?.filter((tagName) => visibleColumns.has(tagName))
|
||||||
<Th isNumeric>Output tokens</Th>
|
.map((tagName) => (
|
||||||
<Th isNumeric>Status</Th>
|
<Th key={tagName} textTransform={"none"}>
|
||||||
|
{tagName}
|
||||||
|
</Th>
|
||||||
|
))}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.DURATION) && <Th isNumeric>Duration</Th>}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.INPUT_TOKENS) && <Th isNumeric>Input tokens</Th>}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.OUTPUT_TOKENS) && <Th isNumeric>Output tokens</Th>}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.STATUS_CODE) && <Th isNumeric>Status</Th>}
|
||||||
</Tr>
|
</Tr>
|
||||||
</Thead>
|
</Thead>
|
||||||
);
|
);
|
||||||
@@ -68,30 +81,30 @@ export const TableRow = ({
|
|||||||
loggedCall,
|
loggedCall,
|
||||||
isExpanded,
|
isExpanded,
|
||||||
onToggle,
|
onToggle,
|
||||||
showCheckbox,
|
showOptions,
|
||||||
}: {
|
}: {
|
||||||
loggedCall: LoggedCall;
|
loggedCall: LoggedCall;
|
||||||
isExpanded: boolean;
|
isExpanded: boolean;
|
||||||
onToggle: () => void;
|
onToggle: () => void;
|
||||||
showCheckbox?: boolean;
|
showOptions?: boolean;
|
||||||
}) => {
|
}) => {
|
||||||
const isError = loggedCall.modelResponse?.statusCode !== 200;
|
const isError = loggedCall.modelResponse?.statusCode !== 200;
|
||||||
const timeAgo = dayjs(loggedCall.requestedAt).fromNow();
|
const requestedAt = dayjs(loggedCall.requestedAt).format("MMMM D h:mm A");
|
||||||
const fullTime = dayjs(loggedCall.requestedAt).toString();
|
const fullTime = dayjs(loggedCall.requestedAt).toString();
|
||||||
|
|
||||||
const durationCell = (
|
|
||||||
<Td isNumeric>
|
|
||||||
{loggedCall.cacheHit ? (
|
|
||||||
<Text color="gray.500">Cached</Text>
|
|
||||||
) : (
|
|
||||||
((loggedCall.modelResponse?.durationMs ?? 0) / 1000).toFixed(2) + "s"
|
|
||||||
)}
|
|
||||||
</Td>
|
|
||||||
);
|
|
||||||
|
|
||||||
const isChecked = useAppStore((s) => s.selectedLogs.selectedLogIds.has(loggedCall.id));
|
const isChecked = useAppStore((s) => s.selectedLogs.selectedLogIds.has(loggedCall.id));
|
||||||
const toggleChecked = useAppStore((s) => s.selectedLogs.toggleSelectedLogId);
|
const toggleChecked = useAppStore((s) => s.selectedLogs.toggleSelectedLogId);
|
||||||
|
|
||||||
|
const tagNames = useTagNames().data;
|
||||||
|
const visibleColumns = useAppStore((s) => s.columnVisibility.visibleColumns);
|
||||||
|
|
||||||
|
const visibleTagNames = useMemo(() => {
|
||||||
|
return tagNames?.filter((tagName) => visibleColumns.has(tagName)) ?? [];
|
||||||
|
}, [tagNames, visibleColumns]);
|
||||||
|
|
||||||
|
const isClientRehydrated = useIsClientRehydrated();
|
||||||
|
if (!isClientRehydrated) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Tr
|
<Tr
|
||||||
@@ -101,43 +114,66 @@ export const TableRow = ({
|
|||||||
sx={{
|
sx={{
|
||||||
"> td": { borderBottom: "none" },
|
"> td": { borderBottom: "none" },
|
||||||
}}
|
}}
|
||||||
|
fontSize="sm"
|
||||||
>
|
>
|
||||||
{showCheckbox && (
|
{showOptions && (
|
||||||
<Td>
|
<Td>
|
||||||
<Checkbox isChecked={isChecked} onChange={() => toggleChecked(loggedCall.id)} />
|
<Checkbox isChecked={isChecked} onChange={() => toggleChecked(loggedCall.id)} />
|
||||||
</Td>
|
</Td>
|
||||||
)}
|
)}
|
||||||
<Td>
|
{visibleColumns.has(StaticColumnKeys.SENT_AT) && (
|
||||||
<Tooltip label={fullTime} placement="top">
|
<Td>
|
||||||
<Box whiteSpace="nowrap" minW="120px">
|
<Tooltip label={fullTime} placement="top">
|
||||||
{timeAgo}
|
<Box whiteSpace="nowrap" minW="120px">
|
||||||
</Box>
|
{requestedAt}
|
||||||
</Tooltip>
|
</Box>
|
||||||
</Td>
|
</Tooltip>
|
||||||
<Td width="100%">
|
</Td>
|
||||||
<HStack justifyContent="flex-start">
|
)}
|
||||||
<Text
|
{visibleColumns.has(StaticColumnKeys.MODEL) && (
|
||||||
colorScheme="purple"
|
<Td>
|
||||||
color="purple.500"
|
<HStack justifyContent="flex-start">
|
||||||
borderColor="purple.500"
|
<Text
|
||||||
px={1}
|
colorScheme="purple"
|
||||||
borderRadius={4}
|
color="purple.500"
|
||||||
borderWidth={1}
|
borderColor="purple.500"
|
||||||
fontSize="xs"
|
px={1}
|
||||||
>
|
borderRadius={4}
|
||||||
{loggedCall.model}
|
borderWidth={1}
|
||||||
</Text>
|
fontSize="xs"
|
||||||
</HStack>
|
whiteSpace="nowrap"
|
||||||
</Td>
|
>
|
||||||
{durationCell}
|
{loggedCall.model}
|
||||||
<Td isNumeric>{loggedCall.modelResponse?.inputTokens}</Td>
|
</Text>
|
||||||
<Td isNumeric>{loggedCall.modelResponse?.outputTokens}</Td>
|
</HStack>
|
||||||
<Td sx={{ color: isError ? "red.500" : "green.500", fontWeight: "semibold" }} isNumeric>
|
</Td>
|
||||||
{loggedCall.modelResponse?.statusCode ?? "No response"}
|
)}
|
||||||
</Td>
|
{visibleTagNames.map((tagName) => (
|
||||||
|
<Td key={tagName}>{loggedCall.tags[tagName]}</Td>
|
||||||
|
))}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.DURATION) && (
|
||||||
|
<Td isNumeric>
|
||||||
|
{loggedCall.cacheHit ? (
|
||||||
|
<Text color="gray.500">Cached</Text>
|
||||||
|
) : (
|
||||||
|
((loggedCall.modelResponse?.durationMs ?? 0) / 1000).toFixed(2) + "s"
|
||||||
|
)}
|
||||||
|
</Td>
|
||||||
|
)}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.INPUT_TOKENS) && (
|
||||||
|
<Td isNumeric>{loggedCall.modelResponse?.inputTokens}</Td>
|
||||||
|
)}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.OUTPUT_TOKENS) && (
|
||||||
|
<Td isNumeric>{loggedCall.modelResponse?.outputTokens}</Td>
|
||||||
|
)}
|
||||||
|
{visibleColumns.has(StaticColumnKeys.STATUS_CODE) && (
|
||||||
|
<Td sx={{ color: isError ? "red.500" : "green.500", fontWeight: "semibold" }} isNumeric>
|
||||||
|
{loggedCall.modelResponse?.statusCode ?? "No response"}
|
||||||
|
</Td>
|
||||||
|
)}
|
||||||
</Tr>
|
</Tr>
|
||||||
<Tr>
|
<Tr>
|
||||||
<Td colSpan={8} p={0}>
|
<Td colSpan={visibleColumns.size + 1} w="full" p={0}>
|
||||||
<Collapse in={isExpanded} unmountOnExit={true}>
|
<Collapse in={isExpanded} unmountOnExit={true}>
|
||||||
<VStack p={4} align="stretch">
|
<VStack p={4} align="stretch">
|
||||||
<HStack align="stretch">
|
<HStack align="stretch">
|
||||||
|
|||||||
@@ -21,6 +21,19 @@ export const env = createEnv({
|
|||||||
ANTHROPIC_API_KEY: z.string().default("placeholder"),
|
ANTHROPIC_API_KEY: z.string().default("placeholder"),
|
||||||
SENTRY_AUTH_TOKEN: z.string().optional(),
|
SENTRY_AUTH_TOKEN: z.string().optional(),
|
||||||
OPENPIPE_API_KEY: z.string().optional(),
|
OPENPIPE_API_KEY: z.string().optional(),
|
||||||
|
SENDER_EMAIL: z.string().default("placeholder"),
|
||||||
|
SMTP_HOST: z.string().default("placeholder"),
|
||||||
|
SMTP_PORT: z.string().default("placeholder"),
|
||||||
|
SMTP_LOGIN: z.string().default("placeholder"),
|
||||||
|
SMTP_PASSWORD: z.string().default("placeholder"),
|
||||||
|
WORKER_CONCURRENCY: z
|
||||||
|
.string()
|
||||||
|
.default("10")
|
||||||
|
.transform((val) => parseInt(val)),
|
||||||
|
WORKER_MAX_POOL_SIZE: z
|
||||||
|
.string()
|
||||||
|
.default("10")
|
||||||
|
.transform((val) => parseInt(val)),
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -33,8 +46,6 @@ export const env = createEnv({
|
|||||||
NEXT_PUBLIC_SOCKET_URL: z.string().url().default("http://localhost:3318"),
|
NEXT_PUBLIC_SOCKET_URL: z.string().url().default("http://localhost:3318"),
|
||||||
NEXT_PUBLIC_HOST: z.string().url().default("http://localhost:3000"),
|
NEXT_PUBLIC_HOST: z.string().url().default("http://localhost:3000"),
|
||||||
NEXT_PUBLIC_SENTRY_DSN: z.string().optional(),
|
NEXT_PUBLIC_SENTRY_DSN: z.string().optional(),
|
||||||
NEXT_PUBLIC_SHOW_DATA: z.string().optional(),
|
|
||||||
NEXT_PUBLIC_FF_SHOW_LOGGED_CALLS: z.string().optional(),
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -49,7 +60,6 @@ export const env = createEnv({
|
|||||||
NEXT_PUBLIC_POSTHOG_KEY: process.env.NEXT_PUBLIC_POSTHOG_KEY,
|
NEXT_PUBLIC_POSTHOG_KEY: process.env.NEXT_PUBLIC_POSTHOG_KEY,
|
||||||
NEXT_PUBLIC_SOCKET_URL: process.env.NEXT_PUBLIC_SOCKET_URL,
|
NEXT_PUBLIC_SOCKET_URL: process.env.NEXT_PUBLIC_SOCKET_URL,
|
||||||
NEXT_PUBLIC_HOST: process.env.NEXT_PUBLIC_HOST,
|
NEXT_PUBLIC_HOST: process.env.NEXT_PUBLIC_HOST,
|
||||||
NEXT_PUBLIC_SHOW_DATA: process.env.NEXT_PUBLIC_SHOW_DATA,
|
|
||||||
GITHUB_CLIENT_ID: process.env.GITHUB_CLIENT_ID,
|
GITHUB_CLIENT_ID: process.env.GITHUB_CLIENT_ID,
|
||||||
GITHUB_CLIENT_SECRET: process.env.GITHUB_CLIENT_SECRET,
|
GITHUB_CLIENT_SECRET: process.env.GITHUB_CLIENT_SECRET,
|
||||||
REPLICATE_API_TOKEN: process.env.REPLICATE_API_TOKEN,
|
REPLICATE_API_TOKEN: process.env.REPLICATE_API_TOKEN,
|
||||||
@@ -57,7 +67,13 @@ export const env = createEnv({
|
|||||||
NEXT_PUBLIC_SENTRY_DSN: process.env.NEXT_PUBLIC_SENTRY_DSN,
|
NEXT_PUBLIC_SENTRY_DSN: process.env.NEXT_PUBLIC_SENTRY_DSN,
|
||||||
SENTRY_AUTH_TOKEN: process.env.SENTRY_AUTH_TOKEN,
|
SENTRY_AUTH_TOKEN: process.env.SENTRY_AUTH_TOKEN,
|
||||||
OPENPIPE_API_KEY: process.env.OPENPIPE_API_KEY,
|
OPENPIPE_API_KEY: process.env.OPENPIPE_API_KEY,
|
||||||
NEXT_PUBLIC_FF_SHOW_LOGGED_CALLS: process.env.NEXT_PUBLIC_FF_SHOW_LOGGED_CALLS,
|
SENDER_EMAIL: process.env.SENDER_EMAIL,
|
||||||
|
SMTP_HOST: process.env.SMTP_HOST,
|
||||||
|
SMTP_PORT: process.env.SMTP_PORT,
|
||||||
|
SMTP_LOGIN: process.env.SMTP_LOGIN,
|
||||||
|
SMTP_PASSWORD: process.env.SMTP_PASSWORD,
|
||||||
|
WORKER_CONCURRENCY: process.env.WORKER_CONCURRENCY,
|
||||||
|
WORKER_MAX_POOL_SIZE: process.env.WORKER_MAX_POOL_SIZE,
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
* Run `build` or `dev` with `SKIP_ENV_VALIDATION` to skip env validation.
|
* Run `build` or `dev` with `SKIP_ENV_VALIDATION` to skip env validation.
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import openaiChatCompletionFrontend from "./openai-ChatCompletion/frontend";
|
import openaiChatCompletionFrontend from "./openai-ChatCompletion/frontend";
|
||||||
import replicateLlama2Frontend from "./replicate-llama2/frontend";
|
import replicateLlama2Frontend from "./replicate-llama2/frontend";
|
||||||
import anthropicFrontend from "./anthropic-completion/frontend";
|
import anthropicFrontend from "./anthropic-completion/frontend";
|
||||||
|
import openpipeFrontend from "./openpipe-chat/frontend";
|
||||||
import { type SupportedProvider, type FrontendModelProvider } from "./types";
|
import { type SupportedProvider, type FrontendModelProvider } from "./types";
|
||||||
|
|
||||||
// Keep attributes here that need to be accessible from the frontend. We can't
|
// Keep attributes here that need to be accessible from the frontend. We can't
|
||||||
@@ -10,6 +11,7 @@ const frontendModelProviders: Record<SupportedProvider, FrontendModelProvider<an
|
|||||||
"openai/ChatCompletion": openaiChatCompletionFrontend,
|
"openai/ChatCompletion": openaiChatCompletionFrontend,
|
||||||
"replicate/llama2": replicateLlama2Frontend,
|
"replicate/llama2": replicateLlama2Frontend,
|
||||||
"anthropic/completion": anthropicFrontend,
|
"anthropic/completion": anthropicFrontend,
|
||||||
|
"openpipe/Chat": openpipeFrontend,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default frontendModelProviders;
|
export default frontendModelProviders;
|
||||||
|
|||||||
@@ -1,12 +1,14 @@
|
|||||||
import openaiChatCompletion from "./openai-ChatCompletion";
|
import openaiChatCompletion from "./openai-ChatCompletion";
|
||||||
import replicateLlama2 from "./replicate-llama2";
|
import replicateLlama2 from "./replicate-llama2";
|
||||||
import anthropicCompletion from "./anthropic-completion";
|
import anthropicCompletion from "./anthropic-completion";
|
||||||
|
import openpipeChatCompletion from "./openpipe-chat";
|
||||||
import { type SupportedProvider, type ModelProvider } from "./types";
|
import { type SupportedProvider, type ModelProvider } from "./types";
|
||||||
|
|
||||||
const modelProviders: Record<SupportedProvider, ModelProvider<any, any, any>> = {
|
const modelProviders: Record<SupportedProvider, ModelProvider<any, any, any>> = {
|
||||||
"openai/ChatCompletion": openaiChatCompletion,
|
"openai/ChatCompletion": openaiChatCompletion,
|
||||||
"replicate/llama2": replicateLlama2,
|
"replicate/llama2": replicateLlama2,
|
||||||
"anthropic/completion": anthropicCompletion,
|
"anthropic/completion": anthropicCompletion,
|
||||||
|
"openpipe/Chat": openpipeChatCompletion,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default modelProviders;
|
export default modelProviders;
|
||||||
|
|||||||
@@ -16,7 +16,16 @@ export async function getCompletion(
|
|||||||
try {
|
try {
|
||||||
if (onStream) {
|
if (onStream) {
|
||||||
const resp = await openai.chat.completions.create(
|
const resp = await openai.chat.completions.create(
|
||||||
{ ...input, stream: true },
|
{
|
||||||
|
...input,
|
||||||
|
stream: true,
|
||||||
|
openpipe: {
|
||||||
|
tags: {
|
||||||
|
prompt_id: "getCompletion",
|
||||||
|
stream: "true",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
maxRetries: 0,
|
maxRetries: 0,
|
||||||
},
|
},
|
||||||
@@ -34,7 +43,16 @@ export async function getCompletion(
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
const resp = await openai.chat.completions.create(
|
const resp = await openai.chat.completions.create(
|
||||||
{ ...input, stream: false },
|
{
|
||||||
|
...input,
|
||||||
|
stream: false,
|
||||||
|
openpipe: {
|
||||||
|
tags: {
|
||||||
|
prompt_id: "getCompletion",
|
||||||
|
stream: "false",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
maxRetries: 0,
|
maxRetries: 0,
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -12,7 +12,6 @@ export const refinementActions: Record<string, RefinementAction> = {
|
|||||||
|
|
||||||
definePrompt("openai/ChatCompletion", {
|
definePrompt("openai/ChatCompletion", {
|
||||||
model: "gpt-4",
|
model: "gpt-4",
|
||||||
stream: true,
|
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
role: "system",
|
role: "system",
|
||||||
@@ -29,7 +28,6 @@ export const refinementActions: Record<string, RefinementAction> = {
|
|||||||
|
|
||||||
definePrompt("openai/ChatCompletion", {
|
definePrompt("openai/ChatCompletion", {
|
||||||
model: "gpt-4",
|
model: "gpt-4",
|
||||||
stream: true,
|
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
role: "system",
|
role: "system",
|
||||||
@@ -126,7 +124,6 @@ export const refinementActions: Record<string, RefinementAction> = {
|
|||||||
|
|
||||||
definePrompt("openai/ChatCompletion", {
|
definePrompt("openai/ChatCompletion", {
|
||||||
model: "gpt-4",
|
model: "gpt-4",
|
||||||
stream: true,
|
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
role: "system",
|
role: "system",
|
||||||
@@ -143,7 +140,6 @@ export const refinementActions: Record<string, RefinementAction> = {
|
|||||||
|
|
||||||
definePrompt("openai/ChatCompletion", {
|
definePrompt("openai/ChatCompletion", {
|
||||||
model: "gpt-4",
|
model: "gpt-4",
|
||||||
stream: true,
|
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
role: "system",
|
role: "system",
|
||||||
@@ -237,7 +233,6 @@ export const refinementActions: Record<string, RefinementAction> = {
|
|||||||
|
|
||||||
definePrompt("openai/ChatCompletion", {
|
definePrompt("openai/ChatCompletion", {
|
||||||
model: "gpt-3.5-turbo",
|
model: "gpt-3.5-turbo",
|
||||||
stream: true,
|
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
role: "system",
|
role: "system",
|
||||||
|
|||||||
98
app/src/modelProviders/openpipe-chat/frontend.ts
Normal file
98
app/src/modelProviders/openpipe-chat/frontend.ts
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
import { type OpenpipeChatOutput, type SupportedModel } from ".";
|
||||||
|
import { type FrontendModelProvider } from "../types";
|
||||||
|
import { refinementActions } from "./refinementActions";
|
||||||
|
import {
|
||||||
|
templateOpenOrcaPrompt,
|
||||||
|
templateAlpacaInstructPrompt,
|
||||||
|
// templateSystemUserAssistantPrompt,
|
||||||
|
templateInstructionInputResponsePrompt,
|
||||||
|
templateAiroborosPrompt,
|
||||||
|
templateGryphePrompt,
|
||||||
|
templateVicunaPrompt,
|
||||||
|
} from "./templatePrompt";
|
||||||
|
|
||||||
|
const frontendModelProvider: FrontendModelProvider<SupportedModel, OpenpipeChatOutput> = {
|
||||||
|
name: "OpenAI ChatCompletion",
|
||||||
|
|
||||||
|
models: {
|
||||||
|
"Open-Orca/OpenOrcaxOpenChat-Preview2-13B": {
|
||||||
|
name: "OpenOrcaxOpenChat-Preview2-13B",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/Open-Orca/OpenOrcaxOpenChat-Preview2-13B",
|
||||||
|
templatePrompt: templateOpenOrcaPrompt,
|
||||||
|
},
|
||||||
|
"Open-Orca/OpenOrca-Platypus2-13B": {
|
||||||
|
name: "OpenOrca-Platypus2-13B",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/Open-Orca/OpenOrca-Platypus2-13B",
|
||||||
|
templatePrompt: templateAlpacaInstructPrompt,
|
||||||
|
defaultStopTokens: ["</s>"],
|
||||||
|
},
|
||||||
|
// "stabilityai/StableBeluga-13B": {
|
||||||
|
// name: "StableBeluga-13B",
|
||||||
|
// contextWindow: 4096,
|
||||||
|
// pricePerSecond: 0.0003,
|
||||||
|
// speed: "medium",
|
||||||
|
// provider: "openpipe/Chat",
|
||||||
|
// learnMoreUrl: "https://huggingface.co/stabilityai/StableBeluga-13B",
|
||||||
|
// templatePrompt: templateSystemUserAssistantPrompt,
|
||||||
|
// },
|
||||||
|
"NousResearch/Nous-Hermes-Llama2-13b": {
|
||||||
|
name: "Nous-Hermes-Llama2-13b",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/NousResearch/Nous-Hermes-Llama2-13b",
|
||||||
|
templatePrompt: templateInstructionInputResponsePrompt,
|
||||||
|
},
|
||||||
|
"jondurbin/airoboros-l2-13b-gpt4-2.0": {
|
||||||
|
name: "airoboros-l2-13b-gpt4-2.0",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/jondurbin/airoboros-l2-13b-gpt4-2.0",
|
||||||
|
templatePrompt: templateAiroborosPrompt,
|
||||||
|
},
|
||||||
|
"lmsys/vicuna-13b-v1.5": {
|
||||||
|
name: "vicuna-13b-v1.5",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/lmsys/vicuna-13b-v1.5",
|
||||||
|
templatePrompt: templateVicunaPrompt,
|
||||||
|
},
|
||||||
|
"Gryphe/MythoMax-L2-13b": {
|
||||||
|
name: "MythoMax-L2-13b",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/Gryphe/MythoMax-L2-13b",
|
||||||
|
templatePrompt: templateGryphePrompt,
|
||||||
|
},
|
||||||
|
"NousResearch/Nous-Hermes-llama-2-7b": {
|
||||||
|
name: "Nous-Hermes-llama-2-7b",
|
||||||
|
contextWindow: 4096,
|
||||||
|
pricePerSecond: 0.0003,
|
||||||
|
speed: "medium",
|
||||||
|
provider: "openpipe/Chat",
|
||||||
|
learnMoreUrl: "https://huggingface.co/NousResearch/Nous-Hermes-llama-2-7b",
|
||||||
|
templatePrompt: templateInstructionInputResponsePrompt,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
refinementActions,
|
||||||
|
|
||||||
|
normalizeOutput: (output) => ({ type: "text", value: output }),
|
||||||
|
};
|
||||||
|
|
||||||
|
export default frontendModelProvider;
|
||||||
134
app/src/modelProviders/openpipe-chat/getCompletion.ts
Normal file
134
app/src/modelProviders/openpipe-chat/getCompletion.ts
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-unsafe-call */
|
||||||
|
import { isArray, isString } from "lodash-es";
|
||||||
|
import OpenAI, { APIError } from "openai";
|
||||||
|
|
||||||
|
import { type CompletionResponse } from "../types";
|
||||||
|
import { type OpenpipeChatInput, type OpenpipeChatOutput } from ".";
|
||||||
|
import frontendModelProvider from "./frontend";
|
||||||
|
|
||||||
|
const modelEndpoints: Record<OpenpipeChatInput["model"], string> = {
|
||||||
|
"Open-Orca/OpenOrcaxOpenChat-Preview2-13B": "https://5ef82gjxk8kdys-8000.proxy.runpod.net/v1",
|
||||||
|
"Open-Orca/OpenOrca-Platypus2-13B": "https://lt5qlel6qcji8t-8000.proxy.runpod.net/v1",
|
||||||
|
// "stabilityai/StableBeluga-13B": "https://vcorl8mxni2ou1-8000.proxy.runpod.net/v1",
|
||||||
|
"NousResearch/Nous-Hermes-Llama2-13b": "https://ncv8pw3u0vb8j2-8000.proxy.runpod.net/v1",
|
||||||
|
"jondurbin/airoboros-l2-13b-gpt4-2.0": "https://9nrbx7oph4btou-8000.proxy.runpod.net/v1",
|
||||||
|
"lmsys/vicuna-13b-v1.5": "https://h88hkt3ux73rb7-8000.proxy.runpod.net/v1",
|
||||||
|
"Gryphe/MythoMax-L2-13b": "https://3l5jvhnxdgky3v-8000.proxy.runpod.net/v1",
|
||||||
|
"NousResearch/Nous-Hermes-llama-2-7b": "https://ua1bpc6kv3dgge-8000.proxy.runpod.net/v1",
|
||||||
|
};
|
||||||
|
|
||||||
|
const CUSTOM_MODELS_ENABLED = false;
|
||||||
|
|
||||||
|
export async function getCompletion(
|
||||||
|
input: OpenpipeChatInput,
|
||||||
|
onStream: ((partialOutput: OpenpipeChatOutput) => void) | null,
|
||||||
|
): Promise<CompletionResponse<OpenpipeChatOutput>> {
|
||||||
|
// Temporarily disable these models because of GPU constraints
|
||||||
|
|
||||||
|
if (!CUSTOM_MODELS_ENABLED) {
|
||||||
|
return {
|
||||||
|
type: "error",
|
||||||
|
message:
|
||||||
|
"We've disabled this model temporarily because of GPU capacity constraints. Check back later.",
|
||||||
|
autoRetry: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const { model, messages, ...rest } = input;
|
||||||
|
|
||||||
|
const templatedPrompt = frontendModelProvider.models[model].templatePrompt?.(messages);
|
||||||
|
|
||||||
|
if (!templatedPrompt) {
|
||||||
|
return {
|
||||||
|
type: "error",
|
||||||
|
message: "Failed to generate prompt",
|
||||||
|
autoRetry: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const openai = new OpenAI({
|
||||||
|
baseURL: modelEndpoints[model],
|
||||||
|
});
|
||||||
|
const start = Date.now();
|
||||||
|
let finalCompletion: OpenpipeChatOutput = "";
|
||||||
|
|
||||||
|
const completionParams = {
|
||||||
|
model,
|
||||||
|
prompt: templatedPrompt,
|
||||||
|
...rest,
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!completionParams.stop && frontendModelProvider.models[model].defaultStopTokens) {
|
||||||
|
completionParams.stop = frontendModelProvider.models[model].defaultStopTokens;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (onStream) {
|
||||||
|
const resp = await openai.completions.create(
|
||||||
|
{ ...completionParams, stream: true },
|
||||||
|
{
|
||||||
|
maxRetries: 0,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
for await (const part of resp) {
|
||||||
|
finalCompletion += part.choices[0]?.text;
|
||||||
|
onStream(finalCompletion);
|
||||||
|
}
|
||||||
|
if (!finalCompletion) {
|
||||||
|
return {
|
||||||
|
type: "error",
|
||||||
|
message: "Streaming failed to return a completion",
|
||||||
|
autoRetry: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const resp = await openai.completions.create(
|
||||||
|
{ ...completionParams, stream: false },
|
||||||
|
{
|
||||||
|
maxRetries: 0,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
finalCompletion = resp.choices[0]?.text || "";
|
||||||
|
if (!finalCompletion) {
|
||||||
|
return {
|
||||||
|
type: "error",
|
||||||
|
message: "Failed to return a completion",
|
||||||
|
autoRetry: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const timeToComplete = Date.now() - start;
|
||||||
|
|
||||||
|
return {
|
||||||
|
type: "success",
|
||||||
|
statusCode: 200,
|
||||||
|
value: finalCompletion,
|
||||||
|
timeToComplete,
|
||||||
|
};
|
||||||
|
} catch (error: unknown) {
|
||||||
|
if (error instanceof APIError) {
|
||||||
|
// The types from the sdk are wrong
|
||||||
|
const rawMessage = error.message as string | string[];
|
||||||
|
// If the message is not a string, stringify it
|
||||||
|
const message = isString(rawMessage)
|
||||||
|
? rawMessage
|
||||||
|
: isArray(rawMessage)
|
||||||
|
? rawMessage.map((m) => m.toString()).join("\n")
|
||||||
|
: (rawMessage as any).toString();
|
||||||
|
return {
|
||||||
|
type: "error",
|
||||||
|
message,
|
||||||
|
autoRetry: error.status === 429 || error.status === 503,
|
||||||
|
statusCode: error.status,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
console.error(error);
|
||||||
|
return {
|
||||||
|
type: "error",
|
||||||
|
message: (error as Error).message,
|
||||||
|
autoRetry: true,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
54
app/src/modelProviders/openpipe-chat/index.ts
Normal file
54
app/src/modelProviders/openpipe-chat/index.ts
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import { type JSONSchema4 } from "json-schema";
|
||||||
|
import { type ModelProvider } from "../types";
|
||||||
|
import inputSchema from "./input.schema.json";
|
||||||
|
import { getCompletion } from "./getCompletion";
|
||||||
|
import frontendModelProvider from "./frontend";
|
||||||
|
|
||||||
|
const supportedModels = [
|
||||||
|
"Open-Orca/OpenOrcaxOpenChat-Preview2-13B",
|
||||||
|
"Open-Orca/OpenOrca-Platypus2-13B",
|
||||||
|
// "stabilityai/StableBeluga-13B",
|
||||||
|
"NousResearch/Nous-Hermes-Llama2-13b",
|
||||||
|
"jondurbin/airoboros-l2-13b-gpt4-2.0",
|
||||||
|
"lmsys/vicuna-13b-v1.5",
|
||||||
|
"Gryphe/MythoMax-L2-13b",
|
||||||
|
"NousResearch/Nous-Hermes-llama-2-7b",
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
export type SupportedModel = (typeof supportedModels)[number];
|
||||||
|
|
||||||
|
export type OpenpipeChatInput = {
|
||||||
|
model: SupportedModel;
|
||||||
|
messages: {
|
||||||
|
role: "system" | "user" | "assistant";
|
||||||
|
content: string;
|
||||||
|
}[];
|
||||||
|
temperature?: number;
|
||||||
|
top_p?: number;
|
||||||
|
stop?: string[] | string;
|
||||||
|
max_tokens?: number;
|
||||||
|
presence_penalty?: number;
|
||||||
|
frequency_penalty?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type OpenpipeChatOutput = string;
|
||||||
|
|
||||||
|
export type OpenpipeChatModelProvider = ModelProvider<
|
||||||
|
SupportedModel,
|
||||||
|
OpenpipeChatInput,
|
||||||
|
OpenpipeChatOutput
|
||||||
|
>;
|
||||||
|
|
||||||
|
const modelProvider: OpenpipeChatModelProvider = {
|
||||||
|
getModel: (input) => input.model,
|
||||||
|
inputSchema: inputSchema as JSONSchema4,
|
||||||
|
canStream: true,
|
||||||
|
getCompletion,
|
||||||
|
getUsage: (input, output) => {
|
||||||
|
// TODO: Implement this
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
...frontendModelProvider,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default modelProvider;
|
||||||
96
app/src/modelProviders/openpipe-chat/input.schema.json
Normal file
96
app/src/modelProviders/openpipe-chat/input.schema.json
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
{
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"model": {
|
||||||
|
"description": "ID of the model to use.",
|
||||||
|
"example": "Open-Orca/OpenOrcaxOpenChat-Preview2-13B",
|
||||||
|
"type": "string",
|
||||||
|
"enum": [
|
||||||
|
"Open-Orca/OpenOrcaxOpenChat-Preview2-13B",
|
||||||
|
"Open-Orca/OpenOrca-Platypus2-13B",
|
||||||
|
"NousResearch/Nous-Hermes-Llama2-13b",
|
||||||
|
"jondurbin/airoboros-l2-13b-gpt4-2.0",
|
||||||
|
"lmsys/vicuna-13b-v1.5",
|
||||||
|
"Gryphe/MythoMax-L2-13b",
|
||||||
|
"NousResearch/Nous-Hermes-llama-2-7b"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"messages": {
|
||||||
|
"description": "A list of messages comprising the conversation so far.",
|
||||||
|
"type": "array",
|
||||||
|
"minItems": 1,
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"role": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["system", "user", "assistant"],
|
||||||
|
"description": "The role of the messages author. One of `system`, `user`, or `assistant`."
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"type": "string",
|
||||||
|
"description": "The contents of the message. `content` is required for all messages."
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["role", "content"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"temperature": {
|
||||||
|
"type": "number",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 2,
|
||||||
|
"default": 1,
|
||||||
|
"example": 1,
|
||||||
|
"nullable": true,
|
||||||
|
"description": "What sampling temperature to use, between 0 and 2. Higher values like 0.8 will make the output more random, while lower values like 0.2 will make it more focused and deterministic.\n\nWe generally recommend altering this or `top_p` but not both.\n"
|
||||||
|
},
|
||||||
|
"top_p": {
|
||||||
|
"type": "number",
|
||||||
|
"minimum": 0,
|
||||||
|
"maximum": 1,
|
||||||
|
"default": 1,
|
||||||
|
"example": 1,
|
||||||
|
"nullable": true,
|
||||||
|
"description": "An alternative to sampling with temperature, called nucleus sampling, where the model considers the results of the tokens with top_p probability mass. So 0.1 means only the tokens comprising the top 10% probability mass are considered.\n\nWe generally recommend altering this or `temperature` but not both.\n"
|
||||||
|
},
|
||||||
|
"stop": {
|
||||||
|
"description": "Up to 4 sequences where the API will stop generating further tokens.\n",
|
||||||
|
"default": null,
|
||||||
|
"oneOf": [
|
||||||
|
{
|
||||||
|
"type": "string",
|
||||||
|
"nullable": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "array",
|
||||||
|
"minItems": 1,
|
||||||
|
"maxItems": 4,
|
||||||
|
"items": {
|
||||||
|
"type": "string"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"max_tokens": {
|
||||||
|
"description": "The maximum number of [tokens](/tokenizer) to generate in the chat completion.\n\nThe total length of input tokens and generated tokens is limited by the model's context length. [Example Python code](https://github.com/openai/openai-cookbook/blob/main/examples/How_to_count_tokens_with_tiktoken.ipynb) for counting tokens.\n",
|
||||||
|
"type": "integer"
|
||||||
|
},
|
||||||
|
"presence_penalty": {
|
||||||
|
"type": "number",
|
||||||
|
"default": 0,
|
||||||
|
"minimum": -2,
|
||||||
|
"maximum": 2,
|
||||||
|
"nullable": true,
|
||||||
|
"description": "Number between -2.0 and 2.0. Positive values penalize new tokens based on whether they appear in the text so far, increasing the model's likelihood to talk about new topics.\n\n[See more information about frequency and presence penalties.](/docs/api-reference/parameter-details)\n"
|
||||||
|
},
|
||||||
|
"frequency_penalty": {
|
||||||
|
"type": "number",
|
||||||
|
"default": 0,
|
||||||
|
"minimum": -2,
|
||||||
|
"maximum": 2,
|
||||||
|
"nullable": true,
|
||||||
|
"description": "Number between -2.0 and 2.0. Positive values penalize new tokens based on their existing frequency in the text so far, decreasing the model's likelihood to repeat the same line verbatim.\n\n[See more information about frequency and presence penalties.](/docs/api-reference/parameter-details)\n"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"required": ["model", "messages"]
|
||||||
|
}
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
import { type RefinementAction } from "../types";
|
||||||
|
|
||||||
|
export const refinementActions: Record<string, RefinementAction> = {};
|
||||||
274
app/src/modelProviders/openpipe-chat/templatePrompt.ts
Normal file
274
app/src/modelProviders/openpipe-chat/templatePrompt.ts
Normal file
@@ -0,0 +1,274 @@
|
|||||||
|
import { type OpenpipeChatInput } from ".";
|
||||||
|
|
||||||
|
// User: Hello<|end_of_turn|>Assistant: Hi<|end_of_turn|>User: How are you today?<|end_of_turn|>Assistant:
|
||||||
|
export const templateOpenOrcaPrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = "<|end_of_turn|>";
|
||||||
|
|
||||||
|
const formattedMessages = messages.map((message) => {
|
||||||
|
if (message.role === "system" || message.role === "user") {
|
||||||
|
return "User: " + message.content;
|
||||||
|
} else {
|
||||||
|
return "Assistant: " + message.content;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let prompt = formattedMessages.join(splitter);
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastUserIndex = prompt.lastIndexOf("User:");
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf("Assistant:");
|
||||||
|
if (lastUserIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + "Assistant:";
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
};
|
||||||
|
|
||||||
|
// ### Instruction:
|
||||||
|
|
||||||
|
// <prompt> (without the <>)
|
||||||
|
|
||||||
|
// ### Response: (leave two newlines for model to respond)
|
||||||
|
export const templateAlpacaInstructPrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = "\n\n";
|
||||||
|
|
||||||
|
const userTag = "### Instruction:\n\n";
|
||||||
|
const assistantTag = "### Response:\n\n";
|
||||||
|
|
||||||
|
const formattedMessages = messages.map((message) => {
|
||||||
|
if (message.role === "system" || message.role === "user") {
|
||||||
|
return userTag + message.content;
|
||||||
|
} else {
|
||||||
|
return assistantTag + message.content;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let prompt = formattedMessages.join(splitter);
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastUserIndex = prompt.lastIndexOf(userTag);
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf(assistantTag);
|
||||||
|
if (lastUserIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + assistantTag;
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
};
|
||||||
|
|
||||||
|
// ### System:
|
||||||
|
// This is a system prompt, please behave and help the user.
|
||||||
|
|
||||||
|
// ### User:
|
||||||
|
// Your prompt here
|
||||||
|
|
||||||
|
// ### Assistant
|
||||||
|
// The output of Stable Beluga 13B
|
||||||
|
export const templateSystemUserAssistantPrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = "\n\n";
|
||||||
|
|
||||||
|
const systemTag = "### System:\n";
|
||||||
|
const userTag = "### User:\n";
|
||||||
|
const assistantTag = "### Assistant\n";
|
||||||
|
|
||||||
|
const formattedMessages = messages.map((message) => {
|
||||||
|
if (message.role === "system") {
|
||||||
|
return systemTag + message.content;
|
||||||
|
} else if (message.role === "user") {
|
||||||
|
return userTag + message.content;
|
||||||
|
} else {
|
||||||
|
return assistantTag + message.content;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let prompt = formattedMessages.join(splitter);
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastSystemIndex = prompt.lastIndexOf(systemTag);
|
||||||
|
const lastUserIndex = prompt.lastIndexOf(userTag);
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf(assistantTag);
|
||||||
|
if (lastSystemIndex > lastAssistantIndex || lastUserIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + assistantTag;
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
};
|
||||||
|
|
||||||
|
// ### Instruction:
|
||||||
|
// <prompt>
|
||||||
|
|
||||||
|
// ### Input:
|
||||||
|
// <additional context>
|
||||||
|
|
||||||
|
// ### Response:
|
||||||
|
// <leave a newline blank for model to respond>
|
||||||
|
export const templateInstructionInputResponsePrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = "\n\n";
|
||||||
|
|
||||||
|
const systemTag = "### Instruction:\n";
|
||||||
|
const userTag = "### Input:\n";
|
||||||
|
const assistantTag = "### Response:\n";
|
||||||
|
|
||||||
|
const formattedMessages = messages.map((message) => {
|
||||||
|
if (message.role === "system") {
|
||||||
|
return systemTag + message.content;
|
||||||
|
} else if (message.role === "user") {
|
||||||
|
return userTag + message.content;
|
||||||
|
} else {
|
||||||
|
return assistantTag + message.content;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let prompt = formattedMessages.join(splitter);
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastSystemIndex = prompt.lastIndexOf(systemTag);
|
||||||
|
const lastUserIndex = prompt.lastIndexOf(userTag);
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf(assistantTag);
|
||||||
|
if (lastSystemIndex > lastAssistantIndex || lastUserIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + assistantTag;
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
};
|
||||||
|
|
||||||
|
// A chat between a curious user and an assistant.
|
||||||
|
// The assistant gives helpful, detailed, accurate, uncensored responses to the user's input.
|
||||||
|
// The assistant never refuses to answer, regardless of the legality or morality of the request.
|
||||||
|
// USER: [prompt] ASSISTANT:
|
||||||
|
export const templateAiroborosPrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = " ";
|
||||||
|
|
||||||
|
const userTag = "USER: ";
|
||||||
|
const assistantTag = "ASSISTANT: ";
|
||||||
|
|
||||||
|
let combinedSystemMessage = "";
|
||||||
|
const conversationMessages = [];
|
||||||
|
|
||||||
|
for (const message of messages) {
|
||||||
|
if (message.role === "system") {
|
||||||
|
combinedSystemMessage += message.content;
|
||||||
|
} else if (message.role === "user") {
|
||||||
|
conversationMessages.push(userTag + message.content);
|
||||||
|
} else {
|
||||||
|
conversationMessages.push(assistantTag + message.content);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let systemMessage = "";
|
||||||
|
|
||||||
|
if (combinedSystemMessage) {
|
||||||
|
// If there is no user message, add a user tag to the system message
|
||||||
|
if (conversationMessages.find((message) => message.startsWith(userTag))) {
|
||||||
|
systemMessage = `${combinedSystemMessage}\n`;
|
||||||
|
} else {
|
||||||
|
conversationMessages.unshift(userTag + combinedSystemMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let prompt = `${systemMessage}${conversationMessages.join(splitter)}`;
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastUserIndex = prompt.lastIndexOf(userTag);
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf(assistantTag);
|
||||||
|
|
||||||
|
if (lastUserIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + assistantTag;
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
};
|
||||||
|
|
||||||
|
// A chat between a curious user and an artificial intelligence assistant. The assistant gives helpful, detailed, and polite answers to the user's questions.
|
||||||
|
|
||||||
|
// USER: {prompt}
|
||||||
|
// ASSISTANT:
|
||||||
|
export const templateVicunaPrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = "\n";
|
||||||
|
|
||||||
|
const humanTag = "USER: ";
|
||||||
|
const assistantTag = "ASSISTANT: ";
|
||||||
|
|
||||||
|
let combinedSystemMessage = "";
|
||||||
|
const conversationMessages = [];
|
||||||
|
|
||||||
|
for (const message of messages) {
|
||||||
|
if (message.role === "system") {
|
||||||
|
combinedSystemMessage += message.content;
|
||||||
|
} else if (message.role === "user") {
|
||||||
|
conversationMessages.push(humanTag + message.content);
|
||||||
|
} else {
|
||||||
|
conversationMessages.push(assistantTag + message.content);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let systemMessage = "";
|
||||||
|
|
||||||
|
if (combinedSystemMessage) {
|
||||||
|
// If there is no user message, add a user tag to the system message
|
||||||
|
if (conversationMessages.find((message) => message.startsWith(humanTag))) {
|
||||||
|
systemMessage = `${combinedSystemMessage}\n\n`;
|
||||||
|
} else {
|
||||||
|
conversationMessages.unshift(humanTag + combinedSystemMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let prompt = `${systemMessage}${conversationMessages.join(splitter)}`;
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastHumanIndex = prompt.lastIndexOf(humanTag);
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf(assistantTag);
|
||||||
|
if (lastHumanIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + assistantTag;
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt.trim();
|
||||||
|
};
|
||||||
|
|
||||||
|
// <System prompt/Character Card>
|
||||||
|
|
||||||
|
// ### Instruction:
|
||||||
|
// Your instruction or question here.
|
||||||
|
// For roleplay purposes, I suggest the following - Write <CHAR NAME>'s next reply in a chat between <YOUR NAME> and <CHAR NAME>. Write a single reply only.
|
||||||
|
|
||||||
|
// ### Response:
|
||||||
|
export const templateGryphePrompt = (messages: OpenpipeChatInput["messages"]) => {
|
||||||
|
const splitter = "\n\n";
|
||||||
|
|
||||||
|
const instructionTag = "### Instruction:\n";
|
||||||
|
const responseTag = "### Response:\n";
|
||||||
|
|
||||||
|
let combinedSystemMessage = "";
|
||||||
|
const conversationMessages = [];
|
||||||
|
|
||||||
|
for (const message of messages) {
|
||||||
|
if (message.role === "system") {
|
||||||
|
combinedSystemMessage += message.content;
|
||||||
|
} else if (message.role === "user") {
|
||||||
|
conversationMessages.push(instructionTag + message.content);
|
||||||
|
} else {
|
||||||
|
conversationMessages.push(responseTag + message.content);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let systemMessage = "";
|
||||||
|
|
||||||
|
if (combinedSystemMessage) {
|
||||||
|
// If there is no user message, add a user tag to the system message
|
||||||
|
if (conversationMessages.find((message) => message.startsWith(instructionTag))) {
|
||||||
|
systemMessage = `${combinedSystemMessage}\n\n`;
|
||||||
|
} else {
|
||||||
|
conversationMessages.unshift(instructionTag + combinedSystemMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let prompt = `${systemMessage}${conversationMessages.join(splitter)}`;
|
||||||
|
|
||||||
|
// Ensure that the prompt ends with an assistant message
|
||||||
|
const lastInstructionIndex = prompt.lastIndexOf(instructionTag);
|
||||||
|
const lastAssistantIndex = prompt.lastIndexOf(responseTag);
|
||||||
|
if (lastInstructionIndex > lastAssistantIndex) {
|
||||||
|
prompt += splitter + responseTag;
|
||||||
|
}
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
};
|
||||||
@@ -8,7 +8,7 @@ const replicate = new Replicate({
|
|||||||
});
|
});
|
||||||
|
|
||||||
const modelIds: Record<ReplicateLlama2Input["model"], string> = {
|
const modelIds: Record<ReplicateLlama2Input["model"], string> = {
|
||||||
"7b-chat": "4f0b260b6a13eb53a6b1891f089d57c08f41003ae79458be5011303d81a394dc",
|
"7b-chat": "7b0bfc9aff140d5b75bacbed23e91fd3c34b01a1e958d32132de6e0a19796e2c",
|
||||||
"13b-chat": "2a7f981751ec7fdf87b5b91ad4db53683a98082e9ff7bfd12c8cd5ea85980a52",
|
"13b-chat": "2a7f981751ec7fdf87b5b91ad4db53683a98082e9ff7bfd12c8cd5ea85980a52",
|
||||||
"70b-chat": "2c1608e18606fad2812020dc541930f2d0495ce32eee50074220b87300bc16e1",
|
"70b-chat": "2c1608e18606fad2812020dc541930f2d0495ce32eee50074220b87300bc16e1",
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,11 +2,13 @@ import { type JSONSchema4 } from "json-schema";
|
|||||||
import { type IconType } from "react-icons";
|
import { type IconType } from "react-icons";
|
||||||
import { type JsonValue } from "type-fest";
|
import { type JsonValue } from "type-fest";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
import { type OpenpipeChatInput } from "./openpipe-chat";
|
||||||
|
|
||||||
export const ZodSupportedProvider = z.union([
|
export const ZodSupportedProvider = z.union([
|
||||||
z.literal("openai/ChatCompletion"),
|
z.literal("openai/ChatCompletion"),
|
||||||
z.literal("replicate/llama2"),
|
z.literal("replicate/llama2"),
|
||||||
z.literal("anthropic/completion"),
|
z.literal("anthropic/completion"),
|
||||||
|
z.literal("openpipe/Chat"),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
export type SupportedProvider = z.infer<typeof ZodSupportedProvider>;
|
export type SupportedProvider = z.infer<typeof ZodSupportedProvider>;
|
||||||
@@ -22,6 +24,8 @@ export type Model = {
|
|||||||
description?: string;
|
description?: string;
|
||||||
learnMoreUrl?: string;
|
learnMoreUrl?: string;
|
||||||
apiDocsUrl?: string;
|
apiDocsUrl?: string;
|
||||||
|
templatePrompt?: (initialPrompt: OpenpipeChatInput["messages"]) => string;
|
||||||
|
defaultStopTokens?: string[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export type ProviderModel = { provider: z.infer<typeof ZodSupportedProvider>; model: string };
|
export type ProviderModel = { provider: z.infer<typeof ZodSupportedProvider>; model: string };
|
||||||
|
|||||||
54
app/src/pages/admin/jobs/index.tsx
Normal file
54
app/src/pages/admin/jobs/index.tsx
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import { Card, Table, Tbody, Td, Th, Thead, Tr } from "@chakra-ui/react";
|
||||||
|
import dayjs from "dayjs";
|
||||||
|
import { isDate, isObject, isString } from "lodash-es";
|
||||||
|
import AppShell from "~/components/nav/AppShell";
|
||||||
|
import { type RouterOutputs, api } from "~/utils/api";
|
||||||
|
|
||||||
|
const fieldsToShow: (keyof RouterOutputs["adminJobs"]["list"][0])[] = [
|
||||||
|
"id",
|
||||||
|
"queue_name",
|
||||||
|
"payload",
|
||||||
|
"priority",
|
||||||
|
"attempts",
|
||||||
|
"last_error",
|
||||||
|
"created_at",
|
||||||
|
"key",
|
||||||
|
"locked_at",
|
||||||
|
"run_at",
|
||||||
|
];
|
||||||
|
|
||||||
|
export default function Jobs() {
|
||||||
|
const jobs = api.adminJobs.list.useQuery({});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<AppShell title="Admin Jobs">
|
||||||
|
<Card m={4} overflowX="auto">
|
||||||
|
<Table>
|
||||||
|
<Thead>
|
||||||
|
<Tr>
|
||||||
|
{fieldsToShow.map((field) => (
|
||||||
|
<Th key={field}>{field}</Th>
|
||||||
|
))}
|
||||||
|
</Tr>
|
||||||
|
</Thead>
|
||||||
|
<Tbody>
|
||||||
|
{jobs.data?.map((job) => (
|
||||||
|
<Tr key={job.id}>
|
||||||
|
{fieldsToShow.map((field) => {
|
||||||
|
// Check if object
|
||||||
|
let value = job[field];
|
||||||
|
if (isDate(value)) {
|
||||||
|
value = dayjs(value).format("YYYY-MM-DD HH:mm:ss");
|
||||||
|
} else if (isObject(value) && !isString(value)) {
|
||||||
|
value = JSON.stringify(value);
|
||||||
|
} // check if date
|
||||||
|
return <Td key={field}>{value}</Td>;
|
||||||
|
})}
|
||||||
|
</Tr>
|
||||||
|
))}
|
||||||
|
</Tbody>
|
||||||
|
</Table>
|
||||||
|
</Card>
|
||||||
|
</AppShell>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -33,7 +33,7 @@ export default function Dashboard() {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AppShell title="Dashboard" requireAuth>
|
<AppShell title="Dashboard" requireAuth requireBeta>
|
||||||
<VStack px={8} py={8} alignItems="flex-start" spacing={4}>
|
<VStack px={8} py={8} alignItems="flex-start" spacing={4}>
|
||||||
<Text fontSize="2xl" fontWeight="bold">
|
<Text fontSize="2xl" fontWeight="bold">
|
||||||
Dashboard
|
Dashboard
|
||||||
|
|||||||
@@ -1,97 +0,0 @@
|
|||||||
import {
|
|
||||||
Box,
|
|
||||||
Breadcrumb,
|
|
||||||
BreadcrumbItem,
|
|
||||||
Center,
|
|
||||||
Flex,
|
|
||||||
Icon,
|
|
||||||
Input,
|
|
||||||
VStack,
|
|
||||||
} from "@chakra-ui/react";
|
|
||||||
import Link from "next/link";
|
|
||||||
|
|
||||||
import { useRouter } from "next/router";
|
|
||||||
import { useState, useEffect } from "react";
|
|
||||||
import { RiDatabase2Line } from "react-icons/ri";
|
|
||||||
import AppShell from "~/components/nav/AppShell";
|
|
||||||
import { api } from "~/utils/api";
|
|
||||||
import { useDataset, useHandledAsyncCallback } from "~/utils/hooks";
|
|
||||||
import DatasetEntriesTable from "~/components/datasets/DatasetEntriesTable";
|
|
||||||
import { DatasetHeaderButtons } from "~/components/datasets/DatasetHeaderButtons/DatasetHeaderButtons";
|
|
||||||
import PageHeaderContainer from "~/components/nav/PageHeaderContainer";
|
|
||||||
import ProjectBreadcrumbContents from "~/components/nav/ProjectBreadcrumbContents";
|
|
||||||
|
|
||||||
export default function Dataset() {
|
|
||||||
const router = useRouter();
|
|
||||||
const utils = api.useContext();
|
|
||||||
|
|
||||||
const dataset = useDataset();
|
|
||||||
const datasetId = router.query.id as string;
|
|
||||||
|
|
||||||
const [name, setName] = useState(dataset.data?.name || "");
|
|
||||||
useEffect(() => {
|
|
||||||
setName(dataset.data?.name || "");
|
|
||||||
}, [dataset.data?.name]);
|
|
||||||
|
|
||||||
const updateMutation = api.datasets.update.useMutation();
|
|
||||||
const [onSaveName] = useHandledAsyncCallback(async () => {
|
|
||||||
if (name && name !== dataset.data?.name && dataset.data?.id) {
|
|
||||||
await updateMutation.mutateAsync({
|
|
||||||
id: dataset.data.id,
|
|
||||||
updates: { name: name },
|
|
||||||
});
|
|
||||||
await Promise.all([utils.datasets.list.invalidate(), utils.datasets.get.invalidate()]);
|
|
||||||
}
|
|
||||||
}, [updateMutation, dataset.data?.id, dataset.data?.name, name]);
|
|
||||||
|
|
||||||
if (!dataset.isLoading && !dataset.data) {
|
|
||||||
return (
|
|
||||||
<AppShell title="Dataset not found">
|
|
||||||
<Center h="100%">
|
|
||||||
<div>Dataset not found 😕</div>
|
|
||||||
</Center>
|
|
||||||
</AppShell>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<AppShell title={dataset.data?.name}>
|
|
||||||
<VStack h="full">
|
|
||||||
<PageHeaderContainer>
|
|
||||||
<Breadcrumb>
|
|
||||||
<BreadcrumbItem>
|
|
||||||
<ProjectBreadcrumbContents projectName={dataset.data?.project?.name} />
|
|
||||||
</BreadcrumbItem>
|
|
||||||
<BreadcrumbItem>
|
|
||||||
<Link href="/data">
|
|
||||||
<Flex alignItems="center" _hover={{ textDecoration: "underline" }}>
|
|
||||||
<Icon as={RiDatabase2Line} boxSize={4} mr={2} /> Datasets
|
|
||||||
</Flex>
|
|
||||||
</Link>
|
|
||||||
</BreadcrumbItem>
|
|
||||||
<BreadcrumbItem isCurrentPage>
|
|
||||||
<Input
|
|
||||||
size="sm"
|
|
||||||
value={name}
|
|
||||||
onChange={(e) => setName(e.target.value)}
|
|
||||||
onBlur={onSaveName}
|
|
||||||
borderWidth={1}
|
|
||||||
borderColor="transparent"
|
|
||||||
fontSize={16}
|
|
||||||
px={0}
|
|
||||||
minW={{ base: 100, lg: 300 }}
|
|
||||||
flex={1}
|
|
||||||
_hover={{ borderColor: "gray.300" }}
|
|
||||||
_focus={{ borderColor: "blue.500", outline: "none" }}
|
|
||||||
/>
|
|
||||||
</BreadcrumbItem>
|
|
||||||
</Breadcrumb>
|
|
||||||
<DatasetHeaderButtons />
|
|
||||||
</PageHeaderContainer>
|
|
||||||
<Box w="full" overflowX="auto" flex={1} px={8} pt={8} pb={16}>
|
|
||||||
{datasetId && <DatasetEntriesTable />}
|
|
||||||
</Box>
|
|
||||||
</VStack>
|
|
||||||
</AppShell>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
import { SimpleGrid, Icon, Breadcrumb, BreadcrumbItem, Flex } from "@chakra-ui/react";
|
|
||||||
import AppShell from "~/components/nav/AppShell";
|
|
||||||
import { RiDatabase2Line } from "react-icons/ri";
|
|
||||||
import {
|
|
||||||
DatasetCard,
|
|
||||||
DatasetCardSkeleton,
|
|
||||||
NewDatasetCard,
|
|
||||||
} from "~/components/datasets/DatasetCard";
|
|
||||||
import PageHeaderContainer from "~/components/nav/PageHeaderContainer";
|
|
||||||
import ProjectBreadcrumbContents from "~/components/nav/ProjectBreadcrumbContents";
|
|
||||||
import { useDatasets } from "~/utils/hooks";
|
|
||||||
|
|
||||||
export default function DatasetsPage() {
|
|
||||||
const datasets = useDatasets();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<AppShell title="Data" requireAuth>
|
|
||||||
<PageHeaderContainer>
|
|
||||||
<Breadcrumb>
|
|
||||||
<BreadcrumbItem>
|
|
||||||
<ProjectBreadcrumbContents />
|
|
||||||
</BreadcrumbItem>
|
|
||||||
<BreadcrumbItem minH={8}>
|
|
||||||
<Flex alignItems="center">
|
|
||||||
<Icon as={RiDatabase2Line} boxSize={4} mr={2} /> Datasets
|
|
||||||
</Flex>
|
|
||||||
</BreadcrumbItem>
|
|
||||||
</Breadcrumb>
|
|
||||||
</PageHeaderContainer>
|
|
||||||
<SimpleGrid w="full" columns={{ base: 1, md: 2, lg: 3, xl: 4 }} spacing={8} py={4} px={8}>
|
|
||||||
<NewDatasetCard />
|
|
||||||
{datasets.data && !datasets.isLoading ? (
|
|
||||||
datasets?.data?.map((dataset) => (
|
|
||||||
<DatasetCard
|
|
||||||
key={dataset.id}
|
|
||||||
dataset={{ ...dataset, numEntries: dataset._count.datasetEntries }}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<DatasetCardSkeleton />
|
|
||||||
<DatasetCardSkeleton />
|
|
||||||
<DatasetCardSkeleton />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</SimpleGrid>
|
|
||||||
</AppShell>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -26,26 +26,6 @@ import Head from "next/head";
|
|||||||
import PageHeaderContainer from "~/components/nav/PageHeaderContainer";
|
import PageHeaderContainer from "~/components/nav/PageHeaderContainer";
|
||||||
import ProjectBreadcrumbContents from "~/components/nav/ProjectBreadcrumbContents";
|
import ProjectBreadcrumbContents from "~/components/nav/ProjectBreadcrumbContents";
|
||||||
|
|
||||||
// TODO: import less to fix deployment with server side props
|
|
||||||
// export const getServerSideProps = async (context: GetServerSidePropsContext<{ id: string }>) => {
|
|
||||||
// const experimentId = context.params?.id as string;
|
|
||||||
|
|
||||||
// const helpers = createServerSideHelpers({
|
|
||||||
// router: appRouter,
|
|
||||||
// ctx: createInnerTRPCContext({ session: null }),
|
|
||||||
// transformer: superjson, // optional - adds superjson serialization
|
|
||||||
// });
|
|
||||||
|
|
||||||
// // prefetch query
|
|
||||||
// await helpers.experiments.stats.prefetch({ id: experimentId });
|
|
||||||
|
|
||||||
// return {
|
|
||||||
// props: {
|
|
||||||
// trpcState: helpers.dehydrate(),
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
// };
|
|
||||||
|
|
||||||
export default function Experiment() {
|
export default function Experiment() {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const utils = api.useContext();
|
const utils = api.useContext();
|
||||||
@@ -53,9 +33,9 @@ export default function Experiment() {
|
|||||||
|
|
||||||
const experiment = useExperiment();
|
const experiment = useExperiment();
|
||||||
const experimentStats = api.experiments.stats.useQuery(
|
const experimentStats = api.experiments.stats.useQuery(
|
||||||
{ id: router.query.id as string },
|
{ id: experiment.data?.id as string },
|
||||||
{
|
{
|
||||||
enabled: !!router.query.id,
|
enabled: !!experiment.data?.id,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
const stats = experimentStats.data;
|
const stats = experimentStats.data;
|
||||||
@@ -144,8 +124,8 @@ export default function Experiment() {
|
|||||||
<ExperimentHeaderButtons />
|
<ExperimentHeaderButtons />
|
||||||
</PageHeaderContainer>
|
</PageHeaderContainer>
|
||||||
<ExperimentSettingsDrawer />
|
<ExperimentSettingsDrawer />
|
||||||
<Box w="100%" overflowX="auto" flex={1}>
|
<Box w="100%" overflowX="auto" flex={1} id="output-container">
|
||||||
<OutputsTable experimentId={router.query.id as string | undefined} />
|
<OutputsTable experimentId={experiment.data?.id} />
|
||||||
</Box>
|
</Box>
|
||||||
</VStack>
|
</VStack>
|
||||||
</AppShell>
|
</AppShell>
|
||||||
18
app/src/pages/fine-tunes/index.tsx
Normal file
18
app/src/pages/fine-tunes/index.tsx
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
import { Text, VStack, Divider } from "@chakra-ui/react";
|
||||||
|
import FineTunesTable from "~/components/fineTunes/FineTunesTable";
|
||||||
|
|
||||||
|
import AppShell from "~/components/nav/AppShell";
|
||||||
|
|
||||||
|
export default function FineTunes() {
|
||||||
|
return (
|
||||||
|
<AppShell title="Fine Tunes" requireAuth requireBeta>
|
||||||
|
<VStack px={8} py={8} alignItems="flex-start" spacing={4} w="full">
|
||||||
|
<Text fontSize="2xl" fontWeight="bold">
|
||||||
|
Fine Tunes
|
||||||
|
</Text>
|
||||||
|
<Divider />
|
||||||
|
<FineTunesTable />
|
||||||
|
</VStack>
|
||||||
|
</AppShell>
|
||||||
|
);
|
||||||
|
}
|
||||||
110
app/src/pages/invitations/[invitationToken].tsx
Normal file
110
app/src/pages/invitations/[invitationToken].tsx
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
import { Center, Text, VStack, HStack, Button, Card } from "@chakra-ui/react";
|
||||||
|
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import AppShell from "~/components/nav/AppShell";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { useSyncVariantEditor } from "~/state/sync";
|
||||||
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
|
|
||||||
|
export default function Invitation() {
|
||||||
|
const router = useRouter();
|
||||||
|
const utils = api.useContext();
|
||||||
|
useSyncVariantEditor();
|
||||||
|
|
||||||
|
const setSelectedProjectId = useAppStore((state) => state.setSelectedProjectId);
|
||||||
|
|
||||||
|
const invitationToken = router.query.invitationToken as string | undefined;
|
||||||
|
|
||||||
|
const invitation = api.users.getProjectInvitation.useQuery(
|
||||||
|
{ invitationToken: invitationToken as string },
|
||||||
|
{ enabled: !!invitationToken },
|
||||||
|
);
|
||||||
|
|
||||||
|
const cancelMutation = api.users.cancelProjectInvitation.useMutation();
|
||||||
|
const [declineInvitation, isDeclining] = useHandledAsyncCallback(async () => {
|
||||||
|
if (invitationToken) {
|
||||||
|
await cancelMutation.mutateAsync({
|
||||||
|
invitationToken,
|
||||||
|
});
|
||||||
|
await router.replace("/");
|
||||||
|
}
|
||||||
|
}, [cancelMutation, invitationToken]);
|
||||||
|
|
||||||
|
const acceptMutation = api.users.acceptProjectInvitation.useMutation();
|
||||||
|
const [acceptInvitation, isAccepting] = useHandledAsyncCallback(async () => {
|
||||||
|
if (invitationToken) {
|
||||||
|
const resp = await acceptMutation.mutateAsync({
|
||||||
|
invitationToken,
|
||||||
|
});
|
||||||
|
if (!maybeReportError(resp) && resp) {
|
||||||
|
await utils.projects.list.invalidate();
|
||||||
|
setSelectedProjectId(resp.payload);
|
||||||
|
}
|
||||||
|
await router.replace("/");
|
||||||
|
}
|
||||||
|
}, [acceptMutation, invitationToken]);
|
||||||
|
|
||||||
|
if (invitation.isLoading) {
|
||||||
|
return (
|
||||||
|
<AppShell requireAuth title="Loading...">
|
||||||
|
<Center h="full">
|
||||||
|
<Text>Loading...</Text>
|
||||||
|
</Center>
|
||||||
|
</AppShell>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!invitationToken || !invitation.data) {
|
||||||
|
return (
|
||||||
|
<AppShell requireAuth title="Invalid invitation token">
|
||||||
|
<Center h="full">
|
||||||
|
<Text>
|
||||||
|
The invitation you've received is invalid or expired. Please ask your project admin for
|
||||||
|
a new token.
|
||||||
|
</Text>
|
||||||
|
</Center>
|
||||||
|
</AppShell>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<AppShell requireAuth title="Invitation">
|
||||||
|
<Center h="full">
|
||||||
|
<Card>
|
||||||
|
<VStack
|
||||||
|
spacing={8}
|
||||||
|
w="full"
|
||||||
|
maxW="2xl"
|
||||||
|
p={16}
|
||||||
|
borderWidth={1}
|
||||||
|
borderRadius={8}
|
||||||
|
bgColor="white"
|
||||||
|
>
|
||||||
|
<Text fontSize="lg" fontWeight="bold">
|
||||||
|
You're invited! 🎉
|
||||||
|
</Text>
|
||||||
|
<Text textAlign="center">
|
||||||
|
You've been invited to join <b>{invitation.data.project.name}</b> by{" "}
|
||||||
|
<b>
|
||||||
|
{invitation.data.sender.name} ({invitation.data.sender.email})
|
||||||
|
</b>
|
||||||
|
.
|
||||||
|
</Text>
|
||||||
|
<HStack spacing={4}>
|
||||||
|
<Button colorScheme="gray" isLoading={isDeclining} onClick={declineInvitation}>
|
||||||
|
Decline
|
||||||
|
</Button>
|
||||||
|
<Button colorScheme="orange" isLoading={isAccepting} onClick={acceptInvitation}>
|
||||||
|
Accept
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</VStack>
|
||||||
|
</Card>
|
||||||
|
</Center>
|
||||||
|
</AppShell>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -9,9 +9,11 @@ import {
|
|||||||
Divider,
|
Divider,
|
||||||
Icon,
|
Icon,
|
||||||
useDisclosure,
|
useDisclosure,
|
||||||
|
Box,
|
||||||
|
Tooltip,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { BsTrash } from "react-icons/bs";
|
import { BsPlus, BsTrash } from "react-icons/bs";
|
||||||
|
|
||||||
import AppShell from "~/components/nav/AppShell";
|
import AppShell from "~/components/nav/AppShell";
|
||||||
import PageHeaderContainer from "~/components/nav/PageHeaderContainer";
|
import PageHeaderContainer from "~/components/nav/PageHeaderContainer";
|
||||||
@@ -21,6 +23,8 @@ import ProjectBreadcrumbContents from "~/components/nav/ProjectBreadcrumbContent
|
|||||||
import CopiableCode from "~/components/CopiableCode";
|
import CopiableCode from "~/components/CopiableCode";
|
||||||
import { DeleteProjectDialog } from "~/components/projectSettings/DeleteProjectDialog";
|
import { DeleteProjectDialog } from "~/components/projectSettings/DeleteProjectDialog";
|
||||||
import AutoResizeTextArea from "~/components/AutoResizeTextArea";
|
import AutoResizeTextArea from "~/components/AutoResizeTextArea";
|
||||||
|
import MemberTable from "~/components/projectSettings/MemberTable";
|
||||||
|
import { InviteMemberModal } from "~/components/projectSettings/InviteMemberModal";
|
||||||
|
|
||||||
export default function Settings() {
|
export default function Settings() {
|
||||||
const utils = api.useContext();
|
const utils = api.useContext();
|
||||||
@@ -50,12 +54,13 @@ export default function Settings() {
|
|||||||
setName(selectedProject?.name);
|
setName(selectedProject?.name);
|
||||||
}, [selectedProject?.name]);
|
}, [selectedProject?.name]);
|
||||||
|
|
||||||
const deleteProjectOpen = useDisclosure();
|
const inviteMemberModal = useDisclosure();
|
||||||
|
const deleteProjectDialog = useDisclosure();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<AppShell>
|
<AppShell requireAuth>
|
||||||
<PageHeaderContainer>
|
<PageHeaderContainer px={{ base: 4, md: 8 }}>
|
||||||
<Breadcrumb>
|
<Breadcrumb>
|
||||||
<BreadcrumbItem>
|
<BreadcrumbItem>
|
||||||
<ProjectBreadcrumbContents />
|
<ProjectBreadcrumbContents />
|
||||||
@@ -65,7 +70,7 @@ export default function Settings() {
|
|||||||
</BreadcrumbItem>
|
</BreadcrumbItem>
|
||||||
</Breadcrumb>
|
</Breadcrumb>
|
||||||
</PageHeaderContainer>
|
</PageHeaderContainer>
|
||||||
<VStack px={8} py={4} alignItems="flex-start" spacing={4}>
|
<VStack px={{ base: 4, md: 8 }} py={4} alignItems="flex-start" spacing={4}>
|
||||||
<VStack spacing={0} alignItems="flex-start">
|
<VStack spacing={0} alignItems="flex-start">
|
||||||
<Text fontSize="2xl" fontWeight="bold">
|
<Text fontSize="2xl" fontWeight="bold">
|
||||||
Project Settings
|
Project Settings
|
||||||
@@ -109,6 +114,37 @@ export default function Settings() {
|
|||||||
</Button>
|
</Button>
|
||||||
</VStack>
|
</VStack>
|
||||||
<Divider backgroundColor="gray.300" />
|
<Divider backgroundColor="gray.300" />
|
||||||
|
<VStack w="full" alignItems="flex-start">
|
||||||
|
<Subtitle>Project Members</Subtitle>
|
||||||
|
|
||||||
|
<Text fontSize="sm">
|
||||||
|
Add members to your project to allow them to view and edit your project's data.
|
||||||
|
</Text>
|
||||||
|
<Box mt={4} w="full">
|
||||||
|
<MemberTable />
|
||||||
|
</Box>
|
||||||
|
<Tooltip
|
||||||
|
isDisabled={selectedProject?.role === "ADMIN"}
|
||||||
|
label="Only admins can invite new members"
|
||||||
|
hasArrow
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
colorScheme="orange"
|
||||||
|
borderRadius={4}
|
||||||
|
onClick={inviteMemberModal.onOpen}
|
||||||
|
mt={2}
|
||||||
|
_disabled={{
|
||||||
|
opacity: 0.6,
|
||||||
|
}}
|
||||||
|
isDisabled={selectedProject?.role !== "ADMIN"}
|
||||||
|
>
|
||||||
|
<Icon as={BsPlus} boxSize={5} />
|
||||||
|
<Text>Invite New Member</Text>
|
||||||
|
</Button>
|
||||||
|
</Tooltip>
|
||||||
|
</VStack>
|
||||||
|
<Divider backgroundColor="gray.300" />
|
||||||
<VStack alignItems="flex-start">
|
<VStack alignItems="flex-start">
|
||||||
<Subtitle>Project API Key</Subtitle>
|
<Subtitle>Project API Key</Subtitle>
|
||||||
<Text fontSize="sm">
|
<Text fontSize="sm">
|
||||||
@@ -141,7 +177,7 @@ export default function Settings() {
|
|||||||
borderRadius={4}
|
borderRadius={4}
|
||||||
mt={2}
|
mt={2}
|
||||||
height="auto"
|
height="auto"
|
||||||
onClick={deleteProjectOpen.onOpen}
|
onClick={deleteProjectDialog.onOpen}
|
||||||
>
|
>
|
||||||
<Icon as={BsTrash} />
|
<Icon as={BsTrash} />
|
||||||
<Text overflowWrap="break-word" whiteSpace="normal" py={2}>
|
<Text overflowWrap="break-word" whiteSpace="normal" py={2}>
|
||||||
@@ -153,7 +189,11 @@ export default function Settings() {
|
|||||||
</VStack>
|
</VStack>
|
||||||
</VStack>
|
</VStack>
|
||||||
</AppShell>
|
</AppShell>
|
||||||
<DeleteProjectDialog isOpen={deleteProjectOpen.isOpen} onClose={deleteProjectOpen.onClose} />
|
<InviteMemberModal isOpen={inviteMemberModal.isOpen} onClose={inviteMemberModal.onClose} />
|
||||||
|
<DeleteProjectDialog
|
||||||
|
isOpen={deleteProjectDialog.isOpen}
|
||||||
|
onClose={deleteProjectDialog.onClose}
|
||||||
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { Text, VStack, Divider, HStack } from "@chakra-ui/react";
|
import { useState } from "react";
|
||||||
|
import { Text, VStack, Divider, HStack, Box } from "@chakra-ui/react";
|
||||||
|
|
||||||
import AppShell from "~/components/nav/AppShell";
|
import AppShell from "~/components/nav/AppShell";
|
||||||
import LoggedCallTable from "~/components/requestLogs/LoggedCallsTable";
|
import LoggedCallTable from "~/components/requestLogs/LoggedCallsTable";
|
||||||
@@ -6,29 +7,50 @@ import LoggedCallsPaginator from "~/components/requestLogs/LoggedCallsPaginator"
|
|||||||
import ActionButton from "~/components/requestLogs/ActionButton";
|
import ActionButton from "~/components/requestLogs/ActionButton";
|
||||||
import { useAppStore } from "~/state/store";
|
import { useAppStore } from "~/state/store";
|
||||||
import { RiFlaskLine } from "react-icons/ri";
|
import { RiFlaskLine } from "react-icons/ri";
|
||||||
|
import { FiFilter } from "react-icons/fi";
|
||||||
|
import LogFilters from "~/components/requestLogs/LogFilters/LogFilters";
|
||||||
|
import ColumnVisiblityDropdown from "~/components/requestLogs/ColumnVisiblityDropdown";
|
||||||
|
import FineTuneButton from "~/components/requestLogs/FineTuneButton";
|
||||||
|
import ExportButton from "~/components/requestLogs/ExportButton";
|
||||||
|
|
||||||
export default function LoggedCalls() {
|
export default function LoggedCalls() {
|
||||||
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
const selectedLogIds = useAppStore((s) => s.selectedLogs.selectedLogIds);
|
||||||
|
|
||||||
|
const [filtersShown, setFiltersShown] = useState(true);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AppShell title="Request Logs" requireAuth>
|
<AppShell title="Request Logs" requireAuth requireBeta>
|
||||||
<VStack px={8} py={8} alignItems="flex-start" spacing={4} w="full">
|
<Box h="100vh" overflowY="scroll">
|
||||||
<Text fontSize="2xl" fontWeight="bold">
|
<VStack px={8} py={8} alignItems="flex-start" spacing={4} w="full">
|
||||||
Request Logs
|
<Text fontSize="2xl" fontWeight="bold">
|
||||||
</Text>
|
Request Logs
|
||||||
<Divider />
|
</Text>
|
||||||
<HStack w="full" justifyContent="flex-end">
|
<Divider />
|
||||||
<ActionButton
|
<HStack w="full" justifyContent="flex-end">
|
||||||
onClick={() => {
|
<FineTuneButton />
|
||||||
console.log("experimenting with these ids", selectedLogIds);
|
<ActionButton
|
||||||
}}
|
onClick={() => {
|
||||||
label="Experiment"
|
console.log("experimenting with these ids", selectedLogIds);
|
||||||
icon={RiFlaskLine}
|
}}
|
||||||
isDisabled={selectedLogIds.size === 0}
|
label="Experiment"
|
||||||
/>
|
icon={RiFlaskLine}
|
||||||
</HStack>
|
isDisabled={selectedLogIds.size === 0}
|
||||||
<LoggedCallTable />
|
/>
|
||||||
<LoggedCallsPaginator />
|
<ExportButton />
|
||||||
</VStack>
|
<ColumnVisiblityDropdown />
|
||||||
|
<ActionButton
|
||||||
|
onClick={() => {
|
||||||
|
setFiltersShown(!filtersShown);
|
||||||
|
}}
|
||||||
|
label={filtersShown ? "Hide Filters" : "Show Filters"}
|
||||||
|
icon={FiFilter}
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
|
{filtersShown && <LogFilters />}
|
||||||
|
<LoggedCallTable />
|
||||||
|
<LoggedCallsPaginator />
|
||||||
|
</VStack>
|
||||||
|
</Box>
|
||||||
</AppShell>
|
</AppShell>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,108 +0,0 @@
|
|||||||
import { type ChatCompletion } from "openai/resources/chat";
|
|
||||||
import { openai } from "../../utils/openai";
|
|
||||||
import { isAxiosError } from "./utils";
|
|
||||||
import { type APIResponse } from "openai/core";
|
|
||||||
import { sleep } from "~/server/utils/sleep";
|
|
||||||
|
|
||||||
const MAX_AUTO_RETRIES = 50;
|
|
||||||
const MIN_DELAY = 500; // milliseconds
|
|
||||||
const MAX_DELAY = 15000; // milliseconds
|
|
||||||
|
|
||||||
function calculateDelay(numPreviousTries: number): number {
|
|
||||||
const baseDelay = Math.min(MAX_DELAY, MIN_DELAY * Math.pow(2, numPreviousTries));
|
|
||||||
const jitter = Math.random() * baseDelay;
|
|
||||||
return baseDelay + jitter;
|
|
||||||
}
|
|
||||||
|
|
||||||
const getCompletionWithBackoff = async (
|
|
||||||
getCompletion: () => Promise<APIResponse<ChatCompletion>>,
|
|
||||||
) => {
|
|
||||||
let completion;
|
|
||||||
let tries = 0;
|
|
||||||
while (tries < MAX_AUTO_RETRIES) {
|
|
||||||
try {
|
|
||||||
completion = await getCompletion();
|
|
||||||
break;
|
|
||||||
} catch (e) {
|
|
||||||
if (isAxiosError(e)) {
|
|
||||||
console.error(e?.response?.data?.error?.message);
|
|
||||||
} else {
|
|
||||||
await sleep(calculateDelay(tries));
|
|
||||||
console.error(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
tries++;
|
|
||||||
}
|
|
||||||
return completion;
|
|
||||||
};
|
|
||||||
// TODO: Add seeds to ensure batches don't contain duplicate data
|
|
||||||
const MAX_BATCH_SIZE = 5;
|
|
||||||
|
|
||||||
export const autogenerateDatasetEntries = async (
|
|
||||||
numToGenerate: number,
|
|
||||||
inputDescription: string,
|
|
||||||
outputDescription: string,
|
|
||||||
): Promise<{ input: string; output: string }[]> => {
|
|
||||||
const batchSizes = Array.from({ length: Math.ceil(numToGenerate / MAX_BATCH_SIZE) }, (_, i) =>
|
|
||||||
i === Math.ceil(numToGenerate / MAX_BATCH_SIZE) - 1 && numToGenerate % MAX_BATCH_SIZE
|
|
||||||
? numToGenerate % MAX_BATCH_SIZE
|
|
||||||
: MAX_BATCH_SIZE,
|
|
||||||
);
|
|
||||||
|
|
||||||
const getCompletion = (batchSize: number) =>
|
|
||||||
openai.chat.completions.create({
|
|
||||||
model: "gpt-4",
|
|
||||||
messages: [
|
|
||||||
{
|
|
||||||
role: "system",
|
|
||||||
content: `The user needs ${batchSize} rows of data, each with an input and an output.\n---\n The input should follow these requirements: ${inputDescription}\n---\n The output should follow these requirements: ${outputDescription}`,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
functions: [
|
|
||||||
{
|
|
||||||
name: "add_list_of_data",
|
|
||||||
description: "Add a list of data to the database",
|
|
||||||
parameters: {
|
|
||||||
type: "object",
|
|
||||||
properties: {
|
|
||||||
rows: {
|
|
||||||
type: "array",
|
|
||||||
description: "The rows of data that match the description",
|
|
||||||
items: {
|
|
||||||
type: "object",
|
|
||||||
properties: {
|
|
||||||
input: {
|
|
||||||
type: "string",
|
|
||||||
description: "The input for this row",
|
|
||||||
},
|
|
||||||
output: {
|
|
||||||
type: "string",
|
|
||||||
description: "The output for this row",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
function_call: { name: "add_list_of_data" },
|
|
||||||
temperature: 0.5,
|
|
||||||
});
|
|
||||||
|
|
||||||
const completionCallbacks = batchSizes.map((batchSize) =>
|
|
||||||
getCompletionWithBackoff(() => getCompletion(batchSize)),
|
|
||||||
);
|
|
||||||
|
|
||||||
const completions = await Promise.all(completionCallbacks);
|
|
||||||
|
|
||||||
const rows = completions.flatMap((completion) => {
|
|
||||||
const parsed = JSON.parse(
|
|
||||||
completion?.choices[0]?.message?.function_call?.arguments ?? "{rows: []}",
|
|
||||||
) as { rows: { input: string; output: string }[] };
|
|
||||||
return parsed.rows;
|
|
||||||
});
|
|
||||||
|
|
||||||
return rows;
|
|
||||||
};
|
|
||||||
@@ -98,6 +98,11 @@ export const autogenerateScenarioValues = async (
|
|||||||
|
|
||||||
function_call: { name: "add_scenario" },
|
function_call: { name: "add_scenario" },
|
||||||
temperature: 0.5,
|
temperature: 0.5,
|
||||||
|
openpipe: {
|
||||||
|
tags: {
|
||||||
|
prompt_id: "autogenerateScenarioValues",
|
||||||
|
},
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const parsed = JSON.parse(
|
const parsed = JSON.parse(
|
||||||
|
|||||||
14
app/src/server/api/external/v1Api.router.ts
vendored
14
app/src/server/api/external/v1Api.router.ts
vendored
@@ -66,7 +66,7 @@ export const v1ApiRouter = createOpenApiRouter({
|
|||||||
|
|
||||||
if (!existingResponse) return { respPayload: null };
|
if (!existingResponse) return { respPayload: null };
|
||||||
|
|
||||||
await prisma.loggedCall.create({
|
const newCall = await prisma.loggedCall.create({
|
||||||
data: {
|
data: {
|
||||||
projectId: ctx.key.projectId,
|
projectId: ctx.key.projectId,
|
||||||
requestedAt: new Date(input.requestedAt),
|
requestedAt: new Date(input.requestedAt),
|
||||||
@@ -75,7 +75,7 @@ export const v1ApiRouter = createOpenApiRouter({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await createTags(existingResponse.originalLoggedCallId, input.tags);
|
await createTags(newCall.projectId, newCall.id, input.tags);
|
||||||
return {
|
return {
|
||||||
respPayload: existingResponse.respPayload,
|
respPayload: existingResponse.respPayload,
|
||||||
};
|
};
|
||||||
@@ -107,7 +107,7 @@ export const v1ApiRouter = createOpenApiRouter({
|
|||||||
.default({}),
|
.default({}),
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
.output(z.object({ status: z.literal("ok") }))
|
.output(z.object({ status: z.union([z.literal("ok"), z.literal("error")]) }))
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
const reqPayload = await reqValidator.spa(input.reqPayload);
|
const reqPayload = await reqValidator.spa(input.reqPayload);
|
||||||
const respPayload = await respValidator.spa(input.respPayload);
|
const respPayload = await respValidator.spa(input.respPayload);
|
||||||
@@ -165,7 +165,7 @@ export const v1ApiRouter = createOpenApiRouter({
|
|||||||
}),
|
}),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
await createTags(newLoggedCallId, input.tags);
|
await createTags(ctx.key.projectId, newLoggedCallId, input.tags);
|
||||||
return { status: "ok" };
|
return { status: "ok" };
|
||||||
}),
|
}),
|
||||||
localTestingOnlyGetLatestLoggedCall: openApiProtectedProc
|
localTestingOnlyGetLatestLoggedCall: openApiProtectedProc
|
||||||
@@ -208,6 +208,7 @@ export const v1ApiRouter = createOpenApiRouter({
|
|||||||
createdAt: true,
|
createdAt: true,
|
||||||
cacheHit: true,
|
cacheHit: true,
|
||||||
tags: true,
|
tags: true,
|
||||||
|
id: true,
|
||||||
modelResponse: {
|
modelResponse: {
|
||||||
select: {
|
select: {
|
||||||
id: true,
|
id: true,
|
||||||
@@ -229,10 +230,11 @@ export const v1ApiRouter = createOpenApiRouter({
|
|||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
async function createTags(loggedCallId: string, tags: Record<string, string>) {
|
async function createTags(projectId: string, loggedCallId: string, tags: Record<string, string>) {
|
||||||
const tagsToCreate = Object.entries(tags).map(([name, value]) => ({
|
const tagsToCreate = Object.entries(tags).map(([name, value]) => ({
|
||||||
|
projectId,
|
||||||
loggedCallId,
|
loggedCallId,
|
||||||
name: name.replaceAll(/[^a-zA-Z0-9_$]/g, "_"),
|
name: name.replaceAll(/[^a-zA-Z0-9_$.]/g, "_"),
|
||||||
value,
|
value,
|
||||||
}));
|
}));
|
||||||
await prisma.loggedCallTag.createMany({
|
await prisma.loggedCallTag.createMany({
|
||||||
|
|||||||
@@ -6,11 +6,12 @@ import { scenarioVariantCellsRouter } from "./routers/scenarioVariantCells.route
|
|||||||
import { scenarioVarsRouter } from "./routers/scenarioVariables.router";
|
import { scenarioVarsRouter } from "./routers/scenarioVariables.router";
|
||||||
import { evaluationsRouter } from "./routers/evaluations.router";
|
import { evaluationsRouter } from "./routers/evaluations.router";
|
||||||
import { worldChampsRouter } from "./routers/worldChamps.router";
|
import { worldChampsRouter } from "./routers/worldChamps.router";
|
||||||
import { datasetsRouter } from "./routers/datasets.router";
|
|
||||||
import { datasetEntries } from "./routers/datasetEntries.router";
|
|
||||||
import { projectsRouter } from "./routers/projects.router";
|
import { projectsRouter } from "./routers/projects.router";
|
||||||
import { dashboardRouter } from "./routers/dashboard.router";
|
import { dashboardRouter } from "./routers/dashboard.router";
|
||||||
import { loggedCallsRouter } from "./routers/loggedCalls.router";
|
import { loggedCallsRouter } from "./routers/loggedCalls.router";
|
||||||
|
import { fineTunesRouter } from "./routers/fineTunes.router";
|
||||||
|
import { usersRouter } from "./routers/users.router";
|
||||||
|
import { adminJobsRouter } from "./routers/adminJobs.router";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is the primary router for your server.
|
* This is the primary router for your server.
|
||||||
@@ -25,11 +26,12 @@ export const appRouter = createTRPCRouter({
|
|||||||
scenarioVars: scenarioVarsRouter,
|
scenarioVars: scenarioVarsRouter,
|
||||||
evaluations: evaluationsRouter,
|
evaluations: evaluationsRouter,
|
||||||
worldChamps: worldChampsRouter,
|
worldChamps: worldChampsRouter,
|
||||||
datasets: datasetsRouter,
|
|
||||||
datasetEntries: datasetEntries,
|
|
||||||
projects: projectsRouter,
|
projects: projectsRouter,
|
||||||
dashboard: dashboardRouter,
|
dashboard: dashboardRouter,
|
||||||
loggedCalls: loggedCallsRouter,
|
loggedCalls: loggedCallsRouter,
|
||||||
|
fineTunes: fineTunesRouter,
|
||||||
|
users: usersRouter,
|
||||||
|
adminJobs: adminJobsRouter,
|
||||||
});
|
});
|
||||||
|
|
||||||
// export type definition of API
|
// export type definition of API
|
||||||
|
|||||||
18
app/src/server/api/routers/adminJobs.router.ts
Normal file
18
app/src/server/api/routers/adminJobs.router.ts
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
import { createTRPCRouter, protectedProcedure } from "~/server/api/trpc";
|
||||||
|
import { kysely } from "~/server/db";
|
||||||
|
import { requireIsAdmin } from "~/utils/accessControl";
|
||||||
|
|
||||||
|
export const adminJobsRouter = createTRPCRouter({
|
||||||
|
list: protectedProcedure.input(z.object({})).query(async ({ ctx }) => {
|
||||||
|
await requireIsAdmin(ctx);
|
||||||
|
|
||||||
|
return await kysely
|
||||||
|
.selectFrom("graphile_worker.jobs")
|
||||||
|
.limit(100)
|
||||||
|
.selectAll()
|
||||||
|
.orderBy("created_at", "desc")
|
||||||
|
.execute();
|
||||||
|
}),
|
||||||
|
});
|
||||||
@@ -1,145 +0,0 @@
|
|||||||
import { z } from "zod";
|
|
||||||
import { createTRPCRouter, protectedProcedure } from "~/server/api/trpc";
|
|
||||||
import { prisma } from "~/server/db";
|
|
||||||
import { requireCanModifyDataset, requireCanViewDataset } from "~/utils/accessControl";
|
|
||||||
import { autogenerateDatasetEntries } from "../autogenerate/autogenerateDatasetEntries";
|
|
||||||
|
|
||||||
export const datasetEntries = createTRPCRouter({
|
|
||||||
list: protectedProcedure
|
|
||||||
.input(z.object({ datasetId: z.string(), page: z.number(), pageSize: z.number() }))
|
|
||||||
.query(async ({ input, ctx }) => {
|
|
||||||
await requireCanViewDataset(input.datasetId, ctx);
|
|
||||||
|
|
||||||
const { datasetId, page, pageSize } = input;
|
|
||||||
|
|
||||||
const entries = await prisma.datasetEntry.findMany({
|
|
||||||
where: {
|
|
||||||
datasetId,
|
|
||||||
},
|
|
||||||
orderBy: { createdAt: "desc" },
|
|
||||||
skip: (page - 1) * pageSize,
|
|
||||||
take: pageSize,
|
|
||||||
});
|
|
||||||
|
|
||||||
const count = await prisma.datasetEntry.count({
|
|
||||||
where: {
|
|
||||||
datasetId,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return {
|
|
||||||
entries,
|
|
||||||
count,
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
createOne: protectedProcedure
|
|
||||||
.input(
|
|
||||||
z.object({
|
|
||||||
datasetId: z.string(),
|
|
||||||
input: z.string(),
|
|
||||||
output: z.string().optional(),
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
await requireCanModifyDataset(input.datasetId, ctx);
|
|
||||||
|
|
||||||
return await prisma.datasetEntry.create({
|
|
||||||
data: {
|
|
||||||
datasetId: input.datasetId,
|
|
||||||
input: input.input,
|
|
||||||
output: input.output,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
|
|
||||||
autogenerateEntries: protectedProcedure
|
|
||||||
.input(
|
|
||||||
z.object({
|
|
||||||
datasetId: z.string(),
|
|
||||||
numToGenerate: z.number(),
|
|
||||||
inputDescription: z.string(),
|
|
||||||
outputDescription: z.string(),
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
await requireCanModifyDataset(input.datasetId, ctx);
|
|
||||||
|
|
||||||
const dataset = await prisma.dataset.findUnique({
|
|
||||||
where: {
|
|
||||||
id: input.datasetId,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!dataset) {
|
|
||||||
throw new Error(`Dataset with id ${input.datasetId} does not exist`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const entries = await autogenerateDatasetEntries(
|
|
||||||
input.numToGenerate,
|
|
||||||
input.inputDescription,
|
|
||||||
input.outputDescription,
|
|
||||||
);
|
|
||||||
|
|
||||||
const createdEntries = await prisma.datasetEntry.createMany({
|
|
||||||
data: entries.map((entry) => ({
|
|
||||||
datasetId: input.datasetId,
|
|
||||||
input: entry.input,
|
|
||||||
output: entry.output,
|
|
||||||
})),
|
|
||||||
});
|
|
||||||
|
|
||||||
return createdEntries;
|
|
||||||
}),
|
|
||||||
|
|
||||||
delete: protectedProcedure
|
|
||||||
.input(z.object({ id: z.string() }))
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
const datasetId = (
|
|
||||||
await prisma.datasetEntry.findUniqueOrThrow({
|
|
||||||
where: { id: input.id },
|
|
||||||
})
|
|
||||||
).datasetId;
|
|
||||||
|
|
||||||
await requireCanModifyDataset(datasetId, ctx);
|
|
||||||
|
|
||||||
return await prisma.datasetEntry.delete({
|
|
||||||
where: {
|
|
||||||
id: input.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
|
|
||||||
update: protectedProcedure
|
|
||||||
.input(
|
|
||||||
z.object({
|
|
||||||
id: z.string(),
|
|
||||||
updates: z.object({
|
|
||||||
input: z.string(),
|
|
||||||
output: z.string().optional(),
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
const existing = await prisma.datasetEntry.findUnique({
|
|
||||||
where: {
|
|
||||||
id: input.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!existing) {
|
|
||||||
throw new Error(`dataEntry with id ${input.id} does not exist`);
|
|
||||||
}
|
|
||||||
|
|
||||||
await requireCanModifyDataset(existing.datasetId, ctx);
|
|
||||||
|
|
||||||
return await prisma.datasetEntry.update({
|
|
||||||
where: {
|
|
||||||
id: input.id,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
input: input.updates.input,
|
|
||||||
output: input.updates.output,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
@@ -1,88 +0,0 @@
|
|||||||
import { z } from "zod";
|
|
||||||
import { createTRPCRouter, protectedProcedure, publicProcedure } from "~/server/api/trpc";
|
|
||||||
import { prisma } from "~/server/db";
|
|
||||||
import {
|
|
||||||
requireCanModifyDataset,
|
|
||||||
requireCanModifyProject,
|
|
||||||
requireCanViewDataset,
|
|
||||||
requireCanViewProject,
|
|
||||||
} from "~/utils/accessControl";
|
|
||||||
|
|
||||||
export const datasetsRouter = createTRPCRouter({
|
|
||||||
list: protectedProcedure
|
|
||||||
.input(z.object({ projectId: z.string() }))
|
|
||||||
.query(async ({ input, ctx }) => {
|
|
||||||
await requireCanViewProject(input.projectId, ctx);
|
|
||||||
|
|
||||||
const datasets = await prisma.dataset.findMany({
|
|
||||||
where: {
|
|
||||||
projectId: input.projectId,
|
|
||||||
},
|
|
||||||
orderBy: {
|
|
||||||
createdAt: "desc",
|
|
||||||
},
|
|
||||||
include: {
|
|
||||||
_count: {
|
|
||||||
select: { datasetEntries: true },
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return datasets;
|
|
||||||
}),
|
|
||||||
|
|
||||||
get: publicProcedure.input(z.object({ id: z.string() })).query(async ({ input, ctx }) => {
|
|
||||||
await requireCanViewDataset(input.id, ctx);
|
|
||||||
return await prisma.dataset.findFirstOrThrow({
|
|
||||||
where: { id: input.id },
|
|
||||||
include: {
|
|
||||||
project: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
|
|
||||||
create: protectedProcedure
|
|
||||||
.input(z.object({ projectId: z.string() }))
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
await requireCanModifyProject(input.projectId, ctx);
|
|
||||||
|
|
||||||
const numDatasets = await prisma.dataset.count({
|
|
||||||
where: {
|
|
||||||
projectId: input.projectId,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return await prisma.dataset.create({
|
|
||||||
data: {
|
|
||||||
name: `Dataset ${numDatasets + 1}`,
|
|
||||||
projectId: input.projectId,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
|
|
||||||
update: protectedProcedure
|
|
||||||
.input(z.object({ id: z.string(), updates: z.object({ name: z.string() }) }))
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
await requireCanModifyDataset(input.id, ctx);
|
|
||||||
return await prisma.dataset.update({
|
|
||||||
where: {
|
|
||||||
id: input.id,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
name: input.updates.name,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
|
|
||||||
delete: protectedProcedure
|
|
||||||
.input(z.object({ id: z.string() }))
|
|
||||||
.mutation(async ({ input, ctx }) => {
|
|
||||||
await requireCanModifyDataset(input.id, ctx);
|
|
||||||
|
|
||||||
await prisma.dataset.delete({
|
|
||||||
where: {
|
|
||||||
id: input.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}),
|
|
||||||
});
|
|
||||||
@@ -85,15 +85,16 @@ export const experimentsRouter = createTRPCRouter({
|
|||||||
return experimentsWithCounts;
|
return experimentsWithCounts;
|
||||||
}),
|
}),
|
||||||
|
|
||||||
get: publicProcedure.input(z.object({ id: z.string() })).query(async ({ input, ctx }) => {
|
get: publicProcedure.input(z.object({ slug: z.string() })).query(async ({ input, ctx }) => {
|
||||||
await requireCanViewExperiment(input.id, ctx);
|
|
||||||
const experiment = await prisma.experiment.findFirstOrThrow({
|
const experiment = await prisma.experiment.findFirstOrThrow({
|
||||||
where: { id: input.id },
|
where: { slug: input.slug },
|
||||||
include: {
|
include: {
|
||||||
project: true,
|
project: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await requireCanViewExperiment(experiment.id, ctx);
|
||||||
|
|
||||||
const canModify = ctx.session?.user.id
|
const canModify = ctx.session?.user.id
|
||||||
? await canModifyExperiment(experiment.id, ctx.session?.user.id)
|
? await canModifyExperiment(experiment.id, ctx.session?.user.id)
|
||||||
: false;
|
: false;
|
||||||
@@ -177,6 +178,7 @@ export const experimentsRouter = createTRPCRouter({
|
|||||||
existingToNewVariantIds.set(variant.id, newVariantId);
|
existingToNewVariantIds.set(variant.id, newVariantId);
|
||||||
variantsToCreate.push({
|
variantsToCreate.push({
|
||||||
...variant,
|
...variant,
|
||||||
|
uiId: uuidv4(),
|
||||||
id: newVariantId,
|
id: newVariantId,
|
||||||
experimentId: newExperimentId,
|
experimentId: newExperimentId,
|
||||||
});
|
});
|
||||||
@@ -190,6 +192,7 @@ export const experimentsRouter = createTRPCRouter({
|
|||||||
scenariosToCreate.push({
|
scenariosToCreate.push({
|
||||||
...scenario,
|
...scenario,
|
||||||
id: newScenarioId,
|
id: newScenarioId,
|
||||||
|
uiId: uuidv4(),
|
||||||
experimentId: newExperimentId,
|
experimentId: newExperimentId,
|
||||||
variableValues: scenario.variableValues as Prisma.InputJsonValue,
|
variableValues: scenario.variableValues as Prisma.InputJsonValue,
|
||||||
});
|
});
|
||||||
@@ -290,7 +293,10 @@ export const experimentsRouter = createTRPCRouter({
|
|||||||
}),
|
}),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
return newExperimentId;
|
const newExperiment = await prisma.experiment.findUniqueOrThrow({
|
||||||
|
where: { id: newExperimentId },
|
||||||
|
});
|
||||||
|
return newExperiment;
|
||||||
}),
|
}),
|
||||||
|
|
||||||
create: protectedProcedure
|
create: protectedProcedure
|
||||||
@@ -335,7 +341,6 @@ export const experimentsRouter = createTRPCRouter({
|
|||||||
|
|
||||||
definePrompt("openai/ChatCompletion", {
|
definePrompt("openai/ChatCompletion", {
|
||||||
model: "gpt-3.5-turbo-0613",
|
model: "gpt-3.5-turbo-0613",
|
||||||
stream: true,
|
|
||||||
messages: [
|
messages: [
|
||||||
{
|
{
|
||||||
role: "system",
|
role: "system",
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user