Compare commits
180 Commits
python-sdk
...
upload-jso
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ffeed77ab4 | ||
|
|
5de1d293d6 | ||
|
|
54b4b1edab | ||
|
|
93a967c6ac | ||
|
|
ed25dcce39 | ||
|
|
c9a53c2b94 | ||
|
|
91f1b2a8ec | ||
|
|
ba9221c093 | ||
|
|
071ce47411 | ||
|
|
4fddc80dc5 | ||
|
|
5aadf3c2ba | ||
|
|
7c4ab151a4 | ||
|
|
c5c8dbf65e | ||
|
|
82b94657b1 | ||
|
|
0a642fac2a | ||
|
|
4d90ff68c8 | ||
|
|
6ac554f7e1 | ||
|
|
422a6ff4c6 | ||
|
|
6153ebda41 | ||
|
|
b682bd6b78 | ||
|
|
43a22865fd | ||
|
|
5b8113d8e7 | ||
|
|
96a589e401 | ||
|
|
16354d83df | ||
|
|
6a5afd0c9b | ||
|
|
1684663ddc | ||
|
|
70fae68225 | ||
|
|
518c8620d0 | ||
|
|
ab87794192 | ||
|
|
48aa697002 | ||
|
|
55f2be861e | ||
|
|
fa87887e91 | ||
|
|
28713fb3ef | ||
|
|
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 | ||
|
|
8f4e7f7e2e | ||
|
|
634739c045 | ||
|
|
9a9cbe8fd4 | ||
|
|
649dc3376b | ||
|
|
05e774d021 | ||
|
|
0e328b13dc | ||
|
|
0a18ca9cd6 | ||
|
|
a5fe35912e | ||
|
|
3d3ddbe7a9 | ||
|
|
d8a5617dee | ||
|
|
5da62fdc29 | ||
|
|
754e273049 | ||
|
|
2863dc2f89 | ||
|
|
c4cef35717 | ||
|
|
8552baf632 | ||
|
|
f41e2229ca | ||
|
|
e649f42c9c | ||
|
|
99f305483b | ||
|
|
b28f4cad57 | ||
|
|
df4a3a0950 | ||
|
|
e423ad656a | ||
|
|
7d0d94de3a | ||
|
|
344b257db4 | ||
|
|
28b43b6e6d | ||
|
|
8d373ec9b5 | ||
|
|
537525667d | ||
|
|
519367c553 | ||
|
|
1a338ec863 | ||
|
|
01d0b8f778 | ||
|
|
d99836ec30 | ||
|
|
33751c12d2 | ||
|
|
89815e1f7f | ||
|
|
5fa5109f34 | ||
|
|
b06ab2cbf9 | ||
|
|
35fb554038 | ||
|
|
f238177277 | ||
|
|
723c0f7505 | ||
|
|
ce6936f753 | ||
|
|
2a80cbf74a | ||
|
|
098805ef25 | ||
|
|
ed90bc5a99 | ||
|
|
de9be8c7ce | ||
|
|
3e02bcf9b8 | ||
|
|
cef2ee31fb | ||
|
|
228c547839 | ||
|
|
e1fcc8fb38 | ||
|
|
3a908d51aa |
5
.dockerignore
Normal file
5
.dockerignore
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
**/node_modules/
|
||||||
|
.git
|
||||||
|
**/.venv/
|
||||||
|
**/.env*
|
||||||
|
**/.next/
|
||||||
14
.github/ISSUE_TEMPLATE/sweep-fast-template.yml
vendored
Normal file
14
.github/ISSUE_TEMPLATE/sweep-fast-template.yml
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
name: Sweep Fast Issue
|
||||||
|
title: 'Sweep (fast): '
|
||||||
|
description: For few-line fixes to be handled by Sweep, an AI-powered junior developer. Sweep will use GPT-3.5 to quickly create a PR for very small changes.
|
||||||
|
labels: sweep
|
||||||
|
body:
|
||||||
|
- type: textarea
|
||||||
|
id: description
|
||||||
|
attributes:
|
||||||
|
label: Details
|
||||||
|
description: Tell Sweep where and what to edit and provide enough context for a new developer to the codebase
|
||||||
|
placeholder: |
|
||||||
|
Bugs: The bug might be in ... file. Here are the logs: ...
|
||||||
|
Features: the new endpoint should use the ... class from ... file because it contains ... logic.
|
||||||
|
Refactors: We are migrating this function to ... version because ...
|
||||||
14
.github/ISSUE_TEMPLATE/sweep-slow-template.yml
vendored
Normal file
14
.github/ISSUE_TEMPLATE/sweep-slow-template.yml
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
name: Sweep Slow Issue
|
||||||
|
title: 'Sweep (slow): '
|
||||||
|
description: For larger bugs, features, refactors, and tests to be handled by Sweep, an AI-powered junior developer. Sweep will perform a deeper search and more self-reviews but will take longer.
|
||||||
|
labels: sweep
|
||||||
|
body:
|
||||||
|
- type: textarea
|
||||||
|
id: description
|
||||||
|
attributes:
|
||||||
|
label: Details
|
||||||
|
description: Tell Sweep where and what to edit and provide enough context for a new developer to the codebase
|
||||||
|
placeholder: |
|
||||||
|
Bugs: The bug might be in ... file. Here are the logs: ...
|
||||||
|
Features: the new endpoint should use the ... class from ... file because it contains ... logic.
|
||||||
|
Refactors: We are migrating this function to ... version because ...
|
||||||
14
.github/ISSUE_TEMPLATE/sweep-template.yml
vendored
Normal file
14
.github/ISSUE_TEMPLATE/sweep-template.yml
vendored
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
name: Sweep Issue
|
||||||
|
title: 'Sweep: '
|
||||||
|
description: For small bugs, features, refactors, and tests to be handled by Sweep, an AI-powered junior developer.
|
||||||
|
labels: sweep
|
||||||
|
body:
|
||||||
|
- type: textarea
|
||||||
|
id: description
|
||||||
|
attributes:
|
||||||
|
label: Details
|
||||||
|
description: Tell Sweep where and what to edit and provide enough context for a new developer to the codebase
|
||||||
|
placeholder: |
|
||||||
|
Bugs: The bug might be in ... file. Here are the logs: ...
|
||||||
|
Features: the new endpoint should use the ... class from ... file because it contains ... logic.
|
||||||
|
Refactors: We are migrating this function to ... version because ...
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,3 +1,6 @@
|
|||||||
.env
|
.env
|
||||||
.venv/
|
.venv/
|
||||||
*.pyc
|
*.pyc
|
||||||
|
node_modules/
|
||||||
|
*.tsbuildinfo
|
||||||
|
dist/
|
||||||
2
.prettierignore
Normal file
2
.prettierignore
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
*.schema.json
|
||||||
|
app/pnpm-lock.yaml
|
||||||
105
README.md
105
README.md
@@ -1,16 +1,53 @@
|
|||||||
<!-- <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>
|
||||||
|
<img src="https://img.shields.io/badge/Y%20Combinator-S23-orange?style=flat-square" alt="Y Combinator S23">
|
||||||
|
</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 +56,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
|
||||||
|
|
||||||
|
|||||||
@@ -32,5 +32,16 @@ NEXT_PUBLIC_HOST="http://localhost:3000"
|
|||||||
GITHUB_CLIENT_ID="your_client_id"
|
GITHUB_CLIENT_ID="your_client_id"
|
||||||
GITHUB_CLIENT_SECRET="your_secret"
|
GITHUB_CLIENT_SECRET="your_secret"
|
||||||
|
|
||||||
OPENPIPE_BASE_URL="http://localhost:3000/api"
|
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"
|
||||||
|
|
||||||
|
# Azure credentials are necessary for uploading large training data files
|
||||||
|
AZURE_STORAGE_ACCOUNT_NAME="placeholder"
|
||||||
|
AZURE_STORAGE_ACCOUNT_KEY="placeholder"
|
||||||
|
AZURE_STORAGE_CONTAINER_NAME="placeholder"
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ const config = {
|
|||||||
overrides: [
|
overrides: [
|
||||||
{
|
{
|
||||||
extends: ["plugin:@typescript-eslint/recommended-requiring-type-checking"],
|
extends: ["plugin:@typescript-eslint/recommended-requiring-type-checking"],
|
||||||
files: ["*.ts", "*.tsx"],
|
files: ["*.mts", "*.ts", "*.tsx"],
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
project: path.join(__dirname, "tsconfig.json"),
|
project: path.join(__dirname, "tsconfig.json"),
|
||||||
},
|
},
|
||||||
|
|||||||
3
app/.gitignore
vendored
3
app/.gitignore
vendored
@@ -44,3 +44,6 @@ yarn-error.log*
|
|||||||
|
|
||||||
# Sentry Auth Token
|
# Sentry Auth Token
|
||||||
.sentryclirc
|
.sentryclirc
|
||||||
|
|
||||||
|
# custom openai intialization
|
||||||
|
src/server/utils/openaiCustomConfig.json
|
||||||
|
|||||||
@@ -1,2 +0,0 @@
|
|||||||
*.schema.json
|
|
||||||
pnpm-lock.yaml
|
|
||||||
17
app/@types/nextjs-routes.d.ts
vendored
17
app/@types/nextjs-routes.d.ts
vendored
@@ -12,19 +12,22 @@ declare module "nextjs-routes" {
|
|||||||
|
|
||||||
export type Route =
|
export type Route =
|
||||||
| StaticRoute<"/account/signin">
|
| StaticRoute<"/account/signin">
|
||||||
| DynamicRoute<"/api/[...trpc]", { "trpc": string[] }>
|
| 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">
|
||||||
| StaticRoute<"/api/openapi">
|
|
||||||
| StaticRoute<"/api/sentry-example-api">
|
|
||||||
| DynamicRoute<"/api/trpc/[trpc]", { "trpc": string }>
|
| DynamicRoute<"/api/trpc/[trpc]", { "trpc": string }>
|
||||||
| DynamicRoute<"/data/[id]", { "id": string }>
|
| DynamicRoute<"/api/v1/[...trpc]", { "trpc": string[] }>
|
||||||
| StaticRoute<"/data">
|
| StaticRoute<"/api/v1/openapi">
|
||||||
| DynamicRoute<"/experiments/[id]", { "id": string }>
|
| StaticRoute<"/dashboard">
|
||||||
|
| DynamicRoute<"/datasets/[id]", { "id": string }>
|
||||||
|
| StaticRoute<"/datasets">
|
||||||
|
| DynamicRoute<"/experiments/[experimentSlug]", { "experimentSlug": string }>
|
||||||
| StaticRoute<"/experiments">
|
| StaticRoute<"/experiments">
|
||||||
|
| StaticRoute<"/fine-tunes">
|
||||||
| StaticRoute<"/">
|
| StaticRoute<"/">
|
||||||
| StaticRoute<"/logged-calls">
|
| DynamicRoute<"/invitations/[invitationToken]", { "invitationToken": string }>
|
||||||
| StaticRoute<"/project/settings">
|
| StaticRoute<"/project/settings">
|
||||||
|
| StaticRoute<"/request-logs">
|
||||||
| StaticRoute<"/sentry-example-page">
|
| StaticRoute<"/sentry-example-page">
|
||||||
| StaticRoute<"/world-champs">
|
| StaticRoute<"/world-champs">
|
||||||
| StaticRoute<"/world-champs/signup">;
|
| StaticRoute<"/world-champs/signup">;
|
||||||
|
|||||||
@@ -6,13 +6,13 @@ RUN yarn global add pnpm
|
|||||||
# DEPS
|
# DEPS
|
||||||
FROM base as deps
|
FROM base as deps
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /code
|
||||||
|
|
||||||
COPY prisma ./
|
COPY app/prisma app/package.json ./app/
|
||||||
|
COPY client-libs/typescript/package.json ./client-libs/typescript/
|
||||||
|
COPY pnpm-lock.yaml pnpm-workspace.yaml ./
|
||||||
|
|
||||||
COPY package.json pnpm-lock.yaml ./
|
RUN cd app && pnpm install --frozen-lockfile
|
||||||
|
|
||||||
RUN pnpm install --frozen-lockfile
|
|
||||||
|
|
||||||
# BUILDER
|
# BUILDER
|
||||||
FROM base as builder
|
FROM base as builder
|
||||||
@@ -23,24 +23,25 @@ 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 /app
|
WORKDIR /code
|
||||||
COPY --from=deps /app/node_modules ./node_modules
|
COPY --from=deps /code/node_modules ./node_modules
|
||||||
|
COPY --from=deps /code/app/node_modules ./app/node_modules
|
||||||
|
COPY --from=deps /code/client-libs/typescript/node_modules ./client-libs/typescript/node_modules
|
||||||
COPY . .
|
COPY . .
|
||||||
RUN SKIP_ENV_VALIDATION=1 pnpm build
|
RUN cd app && SKIP_ENV_VALIDATION=1 pnpm build
|
||||||
|
|
||||||
# RUNNER
|
# RUNNER
|
||||||
FROM base as runner
|
FROM base as runner
|
||||||
WORKDIR /app
|
WORKDIR /code/app
|
||||||
|
|
||||||
ENV NODE_ENV production
|
ENV NODE_ENV production
|
||||||
ENV NEXT_TELEMETRY_DISABLED 1
|
ENV NEXT_TELEMETRY_DISABLED 1
|
||||||
|
|
||||||
COPY --from=builder /app/ ./
|
COPY --from=builder /code/ /code/
|
||||||
|
|
||||||
EXPOSE 3000
|
EXPOSE 3000
|
||||||
ENV PORT 3000
|
ENV PORT 3000
|
||||||
|
|
||||||
# Run the "run-prod.sh" script
|
# Run the "run-prod.sh" script
|
||||||
CMD /app/run-prod.sh
|
CMD /code/app/scripts/run-prod.sh
|
||||||
@@ -36,6 +36,8 @@ let config = {
|
|||||||
});
|
});
|
||||||
return config;
|
return config;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
transpilePackages: ["openpipe"],
|
||||||
};
|
};
|
||||||
|
|
||||||
config = nextRoutes()(config);
|
config = nextRoutes()(config);
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "openpipe",
|
"name": "openpipe-app",
|
||||||
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"version": "0.1.0",
|
"version": "0.1.0",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
@@ -9,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": "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"
|
||||||
@@ -24,7 +26,8 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@anthropic-ai/sdk": "^0.5.8",
|
"@anthropic-ai/sdk": "^0.5.8",
|
||||||
"@apidevtools/json-schema-ref-parser": "^10.1.0",
|
"@apidevtools/json-schema-ref-parser": "^10.1.0",
|
||||||
"@babel/preset-typescript": "^7.22.5",
|
"@azure/identity": "^3.3.0",
|
||||||
|
"@azure/storage-blob": "12.15.0",
|
||||||
"@babel/standalone": "^7.22.9",
|
"@babel/standalone": "^7.22.9",
|
||||||
"@chakra-ui/anatomy": "^2.2.0",
|
"@chakra-ui/anatomy": "^2.2.0",
|
||||||
"@chakra-ui/next-js": "^2.1.4",
|
"@chakra-ui/next-js": "^2.1.4",
|
||||||
@@ -37,6 +40,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 +50,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,20 +63,27 @@
|
|||||||
"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",
|
||||||
|
"llama-tokenizer-js": "^1.1.3",
|
||||||
"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": "^0.3.0",
|
||||||
|
"openpipe-dev": "workspace:^",
|
||||||
"pg": "^8.11.2",
|
"pg": "^8.11.2",
|
||||||
"pluralize": "^8.0.0",
|
"pluralize": "^8.0.0",
|
||||||
"posthog-js": "^1.75.3",
|
"posthog-js": "^1.75.3",
|
||||||
@@ -92,6 +104,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",
|
||||||
@@ -104,6 +117,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",
|
||||||
@@ -113,12 +127,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",
|
||||||
@@ -128,6 +144,7 @@
|
|||||||
"eslint-plugin-unused-imports": "^2.0.0",
|
"eslint-plugin-unused-imports": "^2.0.0",
|
||||||
"monaco-editor": "^0.40.0",
|
"monaco-editor": "^0.40.0",
|
||||||
"openapi-typescript": "^6.3.4",
|
"openapi-typescript": "^6.3.4",
|
||||||
|
"openapi-typescript-codegen": "^0.25.0",
|
||||||
"prisma": "^4.14.0",
|
"prisma": "^4.14.0",
|
||||||
"raw-loader": "^4.0.2",
|
"raw-loader": "^4.0.2",
|
||||||
"typescript": "^5.0.4",
|
"typescript": "^5.0.4",
|
||||||
|
|||||||
12
app/prisma/deleteOneFineTune.ts
Normal file
12
app/prisma/deleteOneFineTune.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { prisma } from "~/server/db";
|
||||||
|
|
||||||
|
// delete most recent fineTune
|
||||||
|
const mostRecentFineTune = await prisma.fineTune.findFirst({
|
||||||
|
orderBy: { createdAt: "desc" },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (mostRecentFineTune) {
|
||||||
|
await prisma.fineTune.delete({
|
||||||
|
where: { id: mostRecentFineTune.id },
|
||||||
|
});
|
||||||
|
}
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "LoggedCall" ADD COLUMN "model" TEXT;
|
||||||
@@ -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;
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
/*
|
||||||
|
Warnings:
|
||||||
|
|
||||||
|
- Added the required column `inputTokens` to the `DatasetEntry` table without a default value. This is not possible if the table is not empty.
|
||||||
|
- Added the required column `outputTokens` to the `DatasetEntry` table without a default value. This is not possible if the table is not empty.
|
||||||
|
- Added the required column `type` to the `DatasetEntry` table without a default value. This is not possible if the table is not empty.
|
||||||
|
|
||||||
|
*/
|
||||||
|
-- CreateEnum
|
||||||
|
CREATE TYPE "DatasetEntryType" AS ENUM ('TRAIN', 'TEST');
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "Dataset" ADD COLUMN "trainingRatio" DOUBLE PRECISION NOT NULL DEFAULT 0.8;
|
||||||
|
|
||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "DatasetEntry" ADD COLUMN "input" JSONB NOT NULL DEFAULT '[]',
|
||||||
|
ADD COLUMN "inputTokens" INTEGER NOT NULL DEFAULT 0,
|
||||||
|
ADD COLUMN "output" JSONB,
|
||||||
|
ADD COLUMN "outputTokens" INTEGER NOT NULL DEFAULT 0,
|
||||||
|
ADD COLUMN "type" "DatasetEntryType" NOT NULL DEFAULT 'TRAIN';
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "DatasetEntry_datasetId_createdAt_id_idx" ON "DatasetEntry"("datasetId", "createdAt", "id");
|
||||||
|
|
||||||
|
-- CreateIndex
|
||||||
|
CREATE INDEX "DatasetEntry_datasetId_type_idx" ON "DatasetEntry"("datasetId", "type");
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
-- AlterTable
|
||||||
|
ALTER TABLE "DatasetEntry" ALTER COLUMN "loggedCallId" DROP NOT NULL,
|
||||||
|
ALTER COLUMN "inputTokens" DROP DEFAULT,
|
||||||
|
ALTER COLUMN "outputTokens" DROP DEFAULT,
|
||||||
|
ALTER COLUMN "type" DROP DEFAULT;
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
-- CreateEnum
|
||||||
|
CREATE TYPE "DatasetFileUploadStatus" AS ENUM ('PENDING', 'DOWNLOADING', 'PROCESSING', 'SAVING', 'COMPLETE', 'ERROR');
|
||||||
|
|
||||||
|
-- CreateTable
|
||||||
|
CREATE TABLE "DatasetFileUpload" (
|
||||||
|
"id" UUID NOT NULL,
|
||||||
|
"datasetId" UUID NOT NULL,
|
||||||
|
"blobName" TEXT NOT NULL,
|
||||||
|
"fileName" TEXT NOT NULL,
|
||||||
|
"fileSize" INTEGER NOT NULL,
|
||||||
|
"progress" INTEGER NOT NULL DEFAULT 0,
|
||||||
|
"status" "DatasetFileUploadStatus" NOT NULL DEFAULT 'PENDING',
|
||||||
|
"uploadedAt" TIMESTAMP(3) NOT NULL,
|
||||||
|
"visible" BOOLEAN NOT NULL DEFAULT true,
|
||||||
|
"errorMessage" TEXT,
|
||||||
|
"createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
"updatedAt" TIMESTAMP(3) NOT NULL,
|
||||||
|
|
||||||
|
CONSTRAINT "DatasetFileUpload_pkey" PRIMARY KEY ("id")
|
||||||
|
);
|
||||||
|
|
||||||
|
-- AddForeignKey
|
||||||
|
ALTER TABLE "DatasetFileUpload" ADD CONSTRAINT "DatasetFileUpload_datasetId_fkey" FOREIGN KEY ("datasetId") REFERENCES "Dataset"("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)
|
||||||
@@ -112,17 +114,17 @@ model ScenarioVariantCell {
|
|||||||
model ModelResponse {
|
model ModelResponse {
|
||||||
id String @id @default(uuid()) @db.Uuid
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
cacheKey String
|
cacheKey String
|
||||||
requestedAt DateTime?
|
requestedAt DateTime?
|
||||||
receivedAt DateTime?
|
receivedAt DateTime?
|
||||||
respPayload Json?
|
respPayload Json?
|
||||||
cost Float?
|
cost Float?
|
||||||
inputTokens Int?
|
inputTokens Int?
|
||||||
outputTokens Int?
|
outputTokens Int?
|
||||||
statusCode Int?
|
statusCode Int?
|
||||||
errorMessage String?
|
errorMessage String?
|
||||||
retryTime DateTime?
|
retryTime DateTime?
|
||||||
outdated Boolean @default(false)
|
outdated Boolean @default(false)
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
@@ -174,11 +176,42 @@ model OutputEvaluation {
|
|||||||
@@unique([modelResponseId, evaluationId])
|
@@unique([modelResponseId, evaluationId])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
enum DatasetFileUploadStatus {
|
||||||
|
PENDING
|
||||||
|
DOWNLOADING
|
||||||
|
PROCESSING
|
||||||
|
SAVING
|
||||||
|
COMPLETE
|
||||||
|
ERROR
|
||||||
|
}
|
||||||
|
|
||||||
|
model DatasetFileUpload {
|
||||||
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
|
datasetId String @db.Uuid
|
||||||
|
dataset Dataset @relation(fields: [datasetId], references: [id], onDelete: Cascade)
|
||||||
|
blobName String
|
||||||
|
fileName String
|
||||||
|
fileSize Int
|
||||||
|
progress Int @default(0) // Percentage
|
||||||
|
status DatasetFileUploadStatus @default(PENDING)
|
||||||
|
uploadedAt DateTime
|
||||||
|
visible Boolean @default(true)
|
||||||
|
errorMessage String?
|
||||||
|
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
}
|
||||||
|
|
||||||
model Dataset {
|
model Dataset {
|
||||||
id String @id @default(uuid()) @db.Uuid
|
id String @id @default(uuid()) @db.Uuid
|
||||||
|
|
||||||
name String
|
name String
|
||||||
datasetEntries DatasetEntry[]
|
datasetEntries DatasetEntry[]
|
||||||
|
fineTunes FineTune[]
|
||||||
|
datasetFileUploads DatasetFileUpload[]
|
||||||
|
trainingRatio Float @default(0.8)
|
||||||
|
|
||||||
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)
|
||||||
@@ -187,17 +220,32 @@ model Dataset {
|
|||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum DatasetEntryType {
|
||||||
|
TRAIN
|
||||||
|
TEST
|
||||||
|
}
|
||||||
|
|
||||||
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)
|
||||||
|
|
||||||
|
input Json @default("[]")
|
||||||
|
output Json?
|
||||||
|
inputTokens Int
|
||||||
|
outputTokens Int
|
||||||
|
|
||||||
|
type DatasetEntryType
|
||||||
|
|
||||||
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)
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
|
|
||||||
|
@@index([datasetId, createdAt, id])
|
||||||
|
@@index([datasetId, type])
|
||||||
}
|
}
|
||||||
|
|
||||||
model Project {
|
model Project {
|
||||||
@@ -207,13 +255,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,7 +323,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)
|
||||||
|
|
||||||
tags LoggedCallTag[]
|
model String?
|
||||||
|
tags LoggedCallTag[]
|
||||||
|
datasetEntries DatasetEntry[]
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
@@ -294,7 +346,7 @@ model LoggedCallModelResponse {
|
|||||||
errorMessage String?
|
errorMessage String?
|
||||||
|
|
||||||
requestedAt DateTime
|
requestedAt DateTime
|
||||||
receivedAt DateTime
|
receivedAt DateTime
|
||||||
|
|
||||||
// Note: the function to calculate the cacheKey should include the project
|
// Note: the function to calculate the cacheKey should include the project
|
||||||
// ID so we don't share cached responses between projects, which could be an
|
// ID so we don't share cached responses between projects, which could be an
|
||||||
@@ -308,7 +360,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
|
||||||
@@ -322,15 +374,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 {
|
||||||
@@ -339,8 +393,8 @@ model ApiKey {
|
|||||||
name String
|
name String
|
||||||
apiKey String @unique
|
apiKey String @unique
|
||||||
|
|
||||||
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)
|
||||||
|
|
||||||
createdAt DateTime @default(now())
|
createdAt DateTime @default(now())
|
||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
@@ -387,16 +441,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
|
||||||
@@ -404,3 +475,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
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { prisma } from "~/server/db";
|
|||||||
import dedent from "dedent";
|
import dedent from "dedent";
|
||||||
import { generateNewCell } from "~/server/utils/generateNewCell";
|
import { generateNewCell } from "~/server/utils/generateNewCell";
|
||||||
import { promptConstructorVersion } from "~/promptConstructor/version";
|
import { promptConstructorVersion } from "~/promptConstructor/version";
|
||||||
|
import { env } from "~/env.mjs";
|
||||||
|
|
||||||
const defaultId = "11111111-1111-1111-1111-111111111111";
|
const defaultId = "11111111-1111-1111-1111-111111111111";
|
||||||
|
|
||||||
@@ -9,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({})) ??
|
||||||
@@ -16,6 +25,20 @@ const project =
|
|||||||
data: { id: defaultId },
|
data: { id: defaultId },
|
||||||
}));
|
}));
|
||||||
|
|
||||||
|
if (env.OPENPIPE_API_KEY) {
|
||||||
|
await prisma.apiKey.upsert({
|
||||||
|
where: {
|
||||||
|
apiKey: env.OPENPIPE_API_KEY,
|
||||||
|
},
|
||||||
|
create: {
|
||||||
|
projectId: project.id,
|
||||||
|
name: "Default API Key",
|
||||||
|
apiKey: env.OPENPIPE_API_KEY,
|
||||||
|
},
|
||||||
|
update: {},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
await prisma.experiment.deleteMany({
|
await prisma.experiment.deleteMany({
|
||||||
where: {
|
where: {
|
||||||
id: defaultId,
|
id: defaultId,
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import { prisma } from "~/server/db";
|
import { prisma } from "~/server/db";
|
||||||
import { generateNewCell } from "~/server/utils/generateNewCell";
|
|
||||||
import dedent from "dedent";
|
import dedent from "dedent";
|
||||||
import { execSync } from "child_process";
|
import { execSync } from "child_process";
|
||||||
import fs from "fs";
|
import fs from "fs";
|
||||||
|
|||||||
@@ -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: {
|
||||||
@@ -79,7 +80,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
},
|
},
|
||||||
respStatus: 200,
|
respStatus: 200,
|
||||||
respPayload: {
|
respPayload: {
|
||||||
id: "chatcmpl-7lNspqePJWVyXwXebupxb1eMozo6Q",
|
id: "chatcmpl-7",
|
||||||
model: "gpt-3.5-turbo-0613",
|
model: "gpt-3.5-turbo-0613",
|
||||||
usage: {
|
usage: {
|
||||||
total_tokens: 241,
|
total_tokens: 241,
|
||||||
@@ -107,6 +108,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: 236,
|
inputTokens: 236,
|
||||||
outputTokens: 5,
|
outputTokens: 5,
|
||||||
finishReason: "stop",
|
finishReason: "stop",
|
||||||
|
tags: [{ name: "prompt_id", value: "add_scenario" }],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -165,7 +167,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
},
|
},
|
||||||
respStatus: 200,
|
respStatus: 200,
|
||||||
respPayload: {
|
respPayload: {
|
||||||
id: "chatcmpl-7lNifmc5AncyAvleZRDBhAcLFYBIT",
|
id: "chatcmpl-7",
|
||||||
model: "gpt-3.5-turbo-0613",
|
model: "gpt-3.5-turbo-0613",
|
||||||
usage: {
|
usage: {
|
||||||
total_tokens: 227,
|
total_tokens: 227,
|
||||||
@@ -193,6 +195,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
inputTokens: 222,
|
inputTokens: 222,
|
||||||
outputTokens: 5,
|
outputTokens: 5,
|
||||||
finishReason: "stop",
|
finishReason: "stop",
|
||||||
|
tags: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -207,7 +210,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
},
|
},
|
||||||
respStatus: 200,
|
respStatus: 200,
|
||||||
respPayload: {
|
respPayload: {
|
||||||
id: "chatcmpl-7lNh1TtrsJVgz3Nj70bKkZZk7xPi7",
|
id: "chatcmpl-7",
|
||||||
model: "gpt-3.5-turbo-0613",
|
model: "gpt-3.5-turbo-0613",
|
||||||
usage: {
|
usage: {
|
||||||
total_tokens: 21,
|
total_tokens: 21,
|
||||||
@@ -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: "translate_text" }],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
reqPayload: {
|
reqPayload: {
|
||||||
@@ -277,7 +281,7 @@ const MODEL_RESPONSE_TEMPLATES: {
|
|||||||
},
|
},
|
||||||
respStatus: 200,
|
respStatus: 200,
|
||||||
respPayload: {
|
respPayload: {
|
||||||
id: "chatcmpl-7lQS3MktOT8BTgNEytl9dkyssCQqL",
|
id: "chatcmpl-7",
|
||||||
model: "gpt-4-0613",
|
model: "gpt-4-0613",
|
||||||
usage: {
|
usage: {
|
||||||
total_tokens: 2910,
|
total_tokens: 2910,
|
||||||
@@ -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: "define_func" },
|
||||||
|
{ name: "some_other_tag", value: "some_other_value" },
|
||||||
|
],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -331,7 +339,7 @@ const loggedCallsToCreate: Prisma.LoggedCallCreateManyInput[] = [];
|
|||||||
const loggedCallModelResponsesToCreate: Prisma.LoggedCallModelResponseCreateManyInput[] = [];
|
const loggedCallModelResponsesToCreate: Prisma.LoggedCallModelResponseCreateManyInput[] = [];
|
||||||
const loggedCallsToUpdate: Prisma.LoggedCallUpdateArgs[] = [];
|
const loggedCallsToUpdate: Prisma.LoggedCallUpdateArgs[] = [];
|
||||||
const loggedCallTagsToCreate: Prisma.LoggedCallTagCreateManyInput[] = [];
|
const loggedCallTagsToCreate: Prisma.LoggedCallTagCreateManyInput[] = [];
|
||||||
for (let i = 0; i < 1437; i++) {
|
for (let i = 0; i < 11437; i++) {
|
||||||
const loggedCallId = uuidv4();
|
const loggedCallId = uuidv4();
|
||||||
const loggedCallModelResponseId = uuidv4();
|
const loggedCallModelResponseId = uuidv4();
|
||||||
const template =
|
const template =
|
||||||
@@ -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);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
55
app/src/components/ActionButton.tsx
Normal file
55
app/src/components/ActionButton.tsx
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
|
||||||
|
import { Button, HStack, type ButtonProps, Icon, Text } from "@chakra-ui/react";
|
||||||
|
import { type IconType } from "react-icons";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { BetaModal } from "./BetaModal";
|
||||||
|
|
||||||
|
const ActionButton = ({
|
||||||
|
icon,
|
||||||
|
iconBoxSize = 3.5,
|
||||||
|
label,
|
||||||
|
requireBeta = false,
|
||||||
|
onClick,
|
||||||
|
...buttonProps
|
||||||
|
}: {
|
||||||
|
icon: IconType;
|
||||||
|
iconBoxSize?: number;
|
||||||
|
label: string;
|
||||||
|
requireBeta?: boolean;
|
||||||
|
onClick?: () => void;
|
||||||
|
} & ButtonProps) => {
|
||||||
|
const flags = useAppStore((s) => s.featureFlags.featureFlags);
|
||||||
|
const flagsLoaded = useAppStore((s) => s.featureFlags.flagsLoaded);
|
||||||
|
|
||||||
|
const [betaModalOpen, setBetaModalOpen] = useState(false);
|
||||||
|
|
||||||
|
const isBetaBlocked = requireBeta && flagsLoaded && !flags.betaAccess;
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Button
|
||||||
|
colorScheme="blue"
|
||||||
|
color="black"
|
||||||
|
bgColor="white"
|
||||||
|
borderColor="gray.300"
|
||||||
|
borderRadius={4}
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
fontSize="sm"
|
||||||
|
fontWeight="normal"
|
||||||
|
onClick={isBetaBlocked ? () => setBetaModalOpen(true) : onClick}
|
||||||
|
{...buttonProps}
|
||||||
|
>
|
||||||
|
<HStack spacing={1}>
|
||||||
|
{icon && (
|
||||||
|
<Icon as={icon} boxSize={iconBoxSize} color={requireBeta ? "orange.400" : undefined} />
|
||||||
|
)}
|
||||||
|
<Text display={{ base: "none", md: "flex" }}>{label}</Text>
|
||||||
|
</HStack>
|
||||||
|
</Button>
|
||||||
|
<BetaModal isOpen={betaModalOpen} onClose={() => setBetaModalOpen(false)} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ActionButton;
|
||||||
@@ -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
|
||||||
|
|||||||
65
app/src/components/BetaModal.tsx
Normal file
65
app/src/components/BetaModal.tsx
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
import {
|
||||||
|
Button,
|
||||||
|
Modal,
|
||||||
|
ModalBody,
|
||||||
|
ModalContent,
|
||||||
|
ModalFooter,
|
||||||
|
ModalHeader,
|
||||||
|
ModalOverlay,
|
||||||
|
VStack,
|
||||||
|
Text,
|
||||||
|
HStack,
|
||||||
|
Icon,
|
||||||
|
Link,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { BsStars } from "react-icons/bs";
|
||||||
|
import { useSession } from "next-auth/react";
|
||||||
|
|
||||||
|
export const BetaModal = ({ isOpen, onClose }: { isOpen: boolean; onClose: () => void }) => {
|
||||||
|
const session = useSession();
|
||||||
|
|
||||||
|
const email = session.data?.user.email ?? "";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal
|
||||||
|
isOpen={isOpen}
|
||||||
|
onClose={onClose}
|
||||||
|
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={onClose}>
|
||||||
|
Done
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { useState, useMemo, useCallback } from "react";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
HStack,
|
HStack,
|
||||||
@@ -14,16 +15,18 @@ import {
|
|||||||
VStack,
|
VStack,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { type PromptVariant } from "@prisma/client";
|
import { type PromptVariant } from "@prisma/client";
|
||||||
import { isObject, isString } from "lodash-es";
|
import { isString } from "lodash-es";
|
||||||
import { useState } from "react";
|
|
||||||
import { RiExchangeFundsFill } from "react-icons/ri";
|
import { RiExchangeFundsFill } from "react-icons/ri";
|
||||||
|
|
||||||
import { type ProviderModel } from "~/modelProviders/types";
|
import { type ProviderModel } from "~/modelProviders/types";
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import { useExperiment, useHandledAsyncCallback, useVisibleScenarioIds } from "~/utils/hooks";
|
import { useExperiment, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
import { lookupModel, modelLabel } from "~/utils/utils";
|
import { lookupModel, modelLabel } from "~/utils/utils";
|
||||||
import CompareFunctions from "../RefinePromptModal/CompareFunctions";
|
import CompareFunctions from "../RefinePromptModal/CompareFunctions";
|
||||||
import { ModelSearch } from "./ModelSearch";
|
import { ModelSearch } from "./ModelSearch";
|
||||||
import { ModelStatsCard } from "./ModelStatsCard";
|
import { ModelStatsCard } from "./ModelStatsCard";
|
||||||
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
|
||||||
export const ChangeModelModal = ({
|
export const ChangeModelModal = ({
|
||||||
variant,
|
variant,
|
||||||
@@ -32,48 +35,43 @@ export const ChangeModelModal = ({
|
|||||||
variant: PromptVariant;
|
variant: PromptVariant;
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
|
const editorOptionsMap = useAppStore((s) => s.sharedVariantEditor.editorOptionsMap);
|
||||||
|
const originalPromptFn = useMemo(
|
||||||
|
() => editorOptionsMap[variant.uiId]?.getContent() || "",
|
||||||
|
[editorOptionsMap, variant.uiId],
|
||||||
|
);
|
||||||
|
|
||||||
const originalModel = lookupModel(variant.modelProvider, variant.model);
|
const originalModel = lookupModel(variant.modelProvider, variant.model);
|
||||||
const [selectedModel, setSelectedModel] = useState({
|
const [selectedModel, setSelectedModel] = useState({
|
||||||
provider: variant.modelProvider,
|
provider: variant.modelProvider,
|
||||||
model: variant.model,
|
model: variant.model,
|
||||||
} as ProviderModel);
|
} as ProviderModel);
|
||||||
const [convertedModel, setConvertedModel] = useState<ProviderModel | undefined>();
|
const [convertedModel, setConvertedModel] = useState<ProviderModel | undefined>();
|
||||||
const visibleScenarios = useVisibleScenarioIds();
|
const [modifiedPromptFn, setModifiedPromptFn] = useState<string>();
|
||||||
|
|
||||||
const utils = api.useContext();
|
|
||||||
|
|
||||||
const experiment = useExperiment();
|
const experiment = useExperiment();
|
||||||
|
|
||||||
const { mutateAsync: getModifiedPromptMutateAsync, data: modifiedPromptFn } =
|
const { mutateAsync: getModifiedPromptMutateAsync } =
|
||||||
api.promptVariants.getModifiedPromptFn.useMutation();
|
api.promptVariants.getModifiedPromptFn.useMutation();
|
||||||
|
|
||||||
const [getModifiedPromptFn, modificationInProgress] = useHandledAsyncCallback(async () => {
|
const [getModifiedPromptFn, modificationInProgress] = useHandledAsyncCallback(async () => {
|
||||||
if (!experiment) return;
|
if (!experiment) return;
|
||||||
|
|
||||||
await getModifiedPromptMutateAsync({
|
const resp = await getModifiedPromptMutateAsync({
|
||||||
id: variant.id,
|
id: variant.id,
|
||||||
|
originalPromptFn,
|
||||||
newModel: selectedModel,
|
newModel: selectedModel,
|
||||||
});
|
});
|
||||||
|
if (maybeReportError(resp)) return;
|
||||||
|
setModifiedPromptFn(resp.payload);
|
||||||
setConvertedModel(selectedModel);
|
setConvertedModel(selectedModel);
|
||||||
}, [getModifiedPromptMutateAsync, onClose, experiment, variant, selectedModel]);
|
}, [getModifiedPromptMutateAsync, onClose, experiment, variant, selectedModel]);
|
||||||
|
|
||||||
const replaceVariantMutation = api.promptVariants.replaceVariant.useMutation();
|
const replaceVariant = useCallback(() => {
|
||||||
|
if (!modifiedPromptFn) return;
|
||||||
const [replaceVariant, replacementInProgress] = useHandledAsyncCallback(async () => {
|
editorOptionsMap[variant.uiId]?.setContent(modifiedPromptFn);
|
||||||
if (
|
|
||||||
!variant.experimentId ||
|
|
||||||
!modifiedPromptFn ||
|
|
||||||
(isObject(modifiedPromptFn) && "status" in modifiedPromptFn)
|
|
||||||
)
|
|
||||||
return;
|
|
||||||
await replaceVariantMutation.mutateAsync({
|
|
||||||
id: variant.id,
|
|
||||||
promptConstructor: modifiedPromptFn,
|
|
||||||
streamScenarios: visibleScenarios,
|
|
||||||
});
|
|
||||||
await utils.promptVariants.list.invalidate();
|
|
||||||
onClose();
|
onClose();
|
||||||
}, [replaceVariantMutation, variant, onClose, modifiedPromptFn]);
|
}, [variant.uiId, editorOptionsMap, onClose, modifiedPromptFn]);
|
||||||
|
|
||||||
const originalLabel = modelLabel(variant.modelProvider, variant.model);
|
const originalLabel = modelLabel(variant.modelProvider, variant.model);
|
||||||
const selectedLabel = modelLabel(selectedModel.provider, selectedModel.model);
|
const selectedLabel = modelLabel(selectedModel.provider, selectedModel.model);
|
||||||
@@ -130,9 +128,9 @@ export const ChangeModelModal = ({
|
|||||||
colorScheme="blue"
|
colorScheme="blue"
|
||||||
onClick={replaceVariant}
|
onClick={replaceVariant}
|
||||||
minW={24}
|
minW={24}
|
||||||
isDisabled={!convertedModel || modificationInProgress || replacementInProgress}
|
isDisabled={!convertedModel || modificationInProgress}
|
||||||
>
|
>
|
||||||
{replacementInProgress ? <Spinner boxSize={4} /> : <Text>Accept</Text>}
|
Accept
|
||||||
</Button>
|
</Button>
|
||||||
</HStack>
|
</HStack>
|
||||||
</ModalFooter>
|
</ModalFooter>
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -1,74 +0,0 @@
|
|||||||
import {
|
|
||||||
Button,
|
|
||||||
Icon,
|
|
||||||
AlertDialog,
|
|
||||||
AlertDialogBody,
|
|
||||||
AlertDialogFooter,
|
|
||||||
AlertDialogHeader,
|
|
||||||
AlertDialogContent,
|
|
||||||
AlertDialogOverlay,
|
|
||||||
useDisclosure,
|
|
||||||
Text,
|
|
||||||
} from "@chakra-ui/react";
|
|
||||||
|
|
||||||
import { useRouter } from "next/router";
|
|
||||||
import { useRef } from "react";
|
|
||||||
import { BsTrash } from "react-icons/bs";
|
|
||||||
import { useAppStore } from "~/state/store";
|
|
||||||
import { api } from "~/utils/api";
|
|
||||||
import { useExperiment, useHandledAsyncCallback } from "~/utils/hooks";
|
|
||||||
|
|
||||||
export const DeleteButton = () => {
|
|
||||||
const experiment = useExperiment();
|
|
||||||
const mutation = api.experiments.delete.useMutation();
|
|
||||||
const utils = api.useContext();
|
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
const closeDrawer = useAppStore((s) => s.closeDrawer);
|
|
||||||
|
|
||||||
const { isOpen, onOpen, onClose } = useDisclosure();
|
|
||||||
const cancelRef = useRef<HTMLButtonElement>(null);
|
|
||||||
|
|
||||||
const [onDeleteConfirm] = useHandledAsyncCallback(async () => {
|
|
||||||
if (!experiment.data?.id) return;
|
|
||||||
await mutation.mutateAsync({ id: experiment.data.id });
|
|
||||||
await utils.experiments.list.invalidate();
|
|
||||||
await router.push({ pathname: "/experiments" });
|
|
||||||
closeDrawer();
|
|
||||||
|
|
||||||
onClose();
|
|
||||||
}, [mutation, experiment.data?.id, router]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Button size="sm" variant="ghost" colorScheme="red" fontWeight="normal" onClick={onOpen}>
|
|
||||||
<Icon as={BsTrash} boxSize={4} />
|
|
||||||
<Text ml={2}>Delete Experiment</Text>
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
<AlertDialog isOpen={isOpen} leastDestructiveRef={cancelRef} onClose={onClose}>
|
|
||||||
<AlertDialogOverlay>
|
|
||||||
<AlertDialogContent>
|
|
||||||
<AlertDialogHeader fontSize="lg" fontWeight="bold">
|
|
||||||
Delete Experiment
|
|
||||||
</AlertDialogHeader>
|
|
||||||
|
|
||||||
<AlertDialogBody>
|
|
||||||
If you delete this experiment all the associated prompts and scenarios will be deleted
|
|
||||||
as well. Are you sure?
|
|
||||||
</AlertDialogBody>
|
|
||||||
|
|
||||||
<AlertDialogFooter>
|
|
||||||
<Button ref={cancelRef} onClick={onClose}>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<Button colorScheme="red" onClick={onDeleteConfirm} ml={3}>
|
|
||||||
Delete
|
|
||||||
</Button>
|
|
||||||
</AlertDialogFooter>
|
|
||||||
</AlertDialogContent>
|
|
||||||
</AlertDialogOverlay>
|
|
||||||
</AlertDialog>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
55
app/src/components/FormattedJson.tsx
Normal file
55
app/src/components/FormattedJson.tsx
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import { Box, IconButton, useToast } from "@chakra-ui/react";
|
||||||
|
import { CopyIcon } from "lucide-react";
|
||||||
|
import SyntaxHighlighter from "react-syntax-highlighter";
|
||||||
|
import { atelierCaveLight } from "react-syntax-highlighter/dist/cjs/styles/hljs";
|
||||||
|
import stringify from "json-stringify-pretty-compact";
|
||||||
|
|
||||||
|
const FormattedJson = ({ json }: { json: any }) => {
|
||||||
|
const jsonString = stringify(json, { maxLength: 40 });
|
||||||
|
const toast = useToast();
|
||||||
|
|
||||||
|
const copyToClipboard = async (text: string) => {
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(text);
|
||||||
|
toast({
|
||||||
|
title: "Copied to clipboard",
|
||||||
|
status: "success",
|
||||||
|
duration: 2000,
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
toast({
|
||||||
|
title: "Failed to copy to clipboard",
|
||||||
|
status: "error",
|
||||||
|
duration: 2000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box position="relative" fontSize="sm" borderRadius="md" overflow="hidden">
|
||||||
|
<SyntaxHighlighter
|
||||||
|
customStyle={{ overflowX: "unset" }}
|
||||||
|
language="json"
|
||||||
|
style={atelierCaveLight}
|
||||||
|
lineProps={{
|
||||||
|
style: { wordBreak: "break-all", whiteSpace: "pre-wrap" },
|
||||||
|
}}
|
||||||
|
wrapLines
|
||||||
|
>
|
||||||
|
{jsonString}
|
||||||
|
</SyntaxHighlighter>
|
||||||
|
<IconButton
|
||||||
|
aria-label="Copy"
|
||||||
|
icon={<CopyIcon />}
|
||||||
|
position="absolute"
|
||||||
|
top={1}
|
||||||
|
right={1}
|
||||||
|
size="xs"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => void copyToClipboard(jsonString)}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export { FormattedJson };
|
||||||
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;
|
||||||
100
app/src/components/InputDropdown.tsx
Normal file
100
app/src/components/InputDropdown.tsx
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
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";
|
||||||
|
import { isEqual } from "lodash-es";
|
||||||
|
import React from "react";
|
||||||
|
|
||||||
|
type InputDropdownProps<T> = {
|
||||||
|
options: ReadonlyArray<T>;
|
||||||
|
selectedOption: T;
|
||||||
|
onSelect: (option: T) => void;
|
||||||
|
inputGroupProps?: InputGroupProps;
|
||||||
|
getDisplayLabel?: (option: T) => string;
|
||||||
|
isDisabled?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
const InputDropdown = <T,>({
|
||||||
|
options,
|
||||||
|
selectedOption,
|
||||||
|
onSelect,
|
||||||
|
inputGroupProps,
|
||||||
|
getDisplayLabel = (option) => option as string,
|
||||||
|
isDisabled,
|
||||||
|
}: InputDropdownProps<T>) => {
|
||||||
|
const { onOpen, ...popover } = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Popover placement="bottom-start" onOpen={isDisabled ? undefined : onOpen} {...popover}>
|
||||||
|
<PopoverTrigger>
|
||||||
|
<InputGroup
|
||||||
|
cursor="pointer"
|
||||||
|
w={getDisplayLabel(selectedOption).length * 14 + 180}
|
||||||
|
{...inputGroupProps}
|
||||||
|
>
|
||||||
|
<Input
|
||||||
|
value={getDisplayLabel(selectedOption)}
|
||||||
|
// 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();
|
||||||
|
}}
|
||||||
|
isDisabled={isDisabled}
|
||||||
|
/>
|
||||||
|
<InputRightElement>
|
||||||
|
<Icon as={FiChevronDown} color={isDisabled ? "gray.300" : undefined} />
|
||||||
|
</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}>{getDisplayLabel(option)}</Text>
|
||||||
|
{isEqual(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() {
|
||||||
@@ -33,25 +33,11 @@ export default function AddVariantButton() {
|
|||||||
<Flex w="100%" justifyContent="flex-end">
|
<Flex w="100%" justifyContent="flex-end">
|
||||||
<ActionButton
|
<ActionButton
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
py={5}
|
py={7}
|
||||||
leftIcon={<Icon as={loading ? Spinner : BsPlus} boxSize={6} mr={loading ? 1 : 0} />}
|
leftIcon={<Icon as={loading ? Spinner : BsPlus} boxSize={6} mr={loading ? 1 : 0} />}
|
||||||
>
|
>
|
||||||
<Text display={{ base: "none", md: "flex" }}>Add Variant</Text>
|
<Text display={{ base: "none", md: "flex" }}>Add Variant</Text>
|
||||||
</ActionButton>
|
</ActionButton>
|
||||||
{/* <Button
|
|
||||||
alignItems="center"
|
|
||||||
justifyContent="center"
|
|
||||||
fontWeight="normal"
|
|
||||||
bgColor="transparent"
|
|
||||||
_hover={{ bgColor: "gray.100" }}
|
|
||||||
px={cellPadding.x}
|
|
||||||
onClick={onClick}
|
|
||||||
height="unset"
|
|
||||||
minH={headerMinHeight}
|
|
||||||
>
|
|
||||||
<Icon as={loading ? Spinner : BsPlus} boxSize={6} mr={loading ? 1 : 0} />
|
|
||||||
<Text display={{ base: "none", md: "flex" }}>Add Variant</Text>
|
|
||||||
</Button> */}
|
|
||||||
</Flex>
|
</Flex>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|
||||||
@@ -33,7 +32,7 @@ export default function OutputCell({
|
|||||||
|
|
||||||
if (!templateHasVariables) disabledReason = "Add a value to the scenario variables to see output";
|
if (!templateHasVariables) disabledReason = "Add a value to the scenario variables to see output";
|
||||||
|
|
||||||
const [refetchInterval, setRefetchInterval] = useState(0);
|
const [refetchInterval, setRefetchInterval] = useState<number | false>(false);
|
||||||
const { data: cell, isLoading: queryLoading } = api.scenarioVariantCells.get.useQuery(
|
const { data: cell, isLoading: queryLoading } = api.scenarioVariantCells.get.useQuery(
|
||||||
{ scenarioId: scenario.id, variantId: variant.id },
|
{ scenarioId: scenario.id, variantId: variant.id },
|
||||||
{ refetchInterval },
|
{ refetchInterval },
|
||||||
@@ -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({
|
||||||
@@ -64,42 +63,34 @@ export default function OutputCell({
|
|||||||
cell.retrievalStatus === "PENDING" ||
|
cell.retrievalStatus === "PENDING" ||
|
||||||
cell.retrievalStatus === "IN_PROGRESS" ||
|
cell.retrievalStatus === "IN_PROGRESS" ||
|
||||||
hardRefetching;
|
hardRefetching;
|
||||||
useEffect(() => setRefetchInterval(awaitingOutput ? 1000 : 0), [awaitingOutput]);
|
|
||||||
|
useEffect(() => setRefetchInterval(awaitingOutput ? 1000 : false), [awaitingOutput]);
|
||||||
|
|
||||||
// TODO: disconnect from socket if we're not streaming anymore
|
// TODO: disconnect from socket if we're not streaming anymore
|
||||||
const streamedMessage = useSocket<OutputSchema>(cell?.id);
|
const streamedMessage = useSocket<OutputSchema>(cell?.id);
|
||||||
|
|
||||||
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>
|
||||||
);
|
);
|
||||||
@@ -111,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) => {
|
||||||
@@ -120,8 +116,13 @@ export default function OutputCell({
|
|||||||
? response.receivedAt.getTime()
|
? response.receivedAt.getTime()
|
||||||
: Date.now();
|
: Date.now();
|
||||||
if (response.requestedAt) {
|
if (response.requestedAt) {
|
||||||
numWaitingMessages = Math.floor(
|
numWaitingMessages = Math.min(
|
||||||
(relativeWaitingTime - response.requestedAt.getTime()) / WAITING_MESSAGE_INTERVAL,
|
Math.floor(
|
||||||
|
(relativeWaitingTime - response.requestedAt.getTime()) / WAITING_MESSAGE_INTERVAL,
|
||||||
|
),
|
||||||
|
// Don't try to render more than 15, it'll use too much CPU and
|
||||||
|
// break the page
|
||||||
|
15,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
@@ -146,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>
|
||||||
@@ -168,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"
|
||||||
@@ -187,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}>
|
||||||
|
|||||||
@@ -1,21 +1,16 @@
|
|||||||
|
import { type StackProps } from "@chakra-ui/react";
|
||||||
|
|
||||||
import { useScenarios } from "~/utils/hooks";
|
import { useScenarios } from "~/utils/hooks";
|
||||||
import Paginator from "../Paginator";
|
import Paginator from "../Paginator";
|
||||||
|
|
||||||
const ScenarioPaginator = () => {
|
const ScenarioPaginator = (props: StackProps) => {
|
||||||
const { data } = useScenarios();
|
const { data } = useScenarios();
|
||||||
|
|
||||||
if (!data) return null;
|
if (!data) return null;
|
||||||
|
|
||||||
const { scenarios, startIndex, lastPage, count } = data;
|
const { count } = data;
|
||||||
|
|
||||||
return (
|
return <Paginator count={count} condense {...props} />;
|
||||||
<Paginator
|
|
||||||
numItemsLoaded={scenarios.length}
|
|
||||||
startIndex={startIndex}
|
|
||||||
lastPage={lastPage}
|
|
||||||
count={count}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default ScenarioPaginator;
|
export default ScenarioPaginator;
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ const ScenarioRow = (props: {
|
|||||||
variants: PromptVariant[];
|
variants: PromptVariant[];
|
||||||
canHide: boolean;
|
canHide: boolean;
|
||||||
rowStart: number;
|
rowStart: number;
|
||||||
|
isFirst: boolean;
|
||||||
|
isLast: boolean;
|
||||||
}) => {
|
}) => {
|
||||||
const [isHovered, setIsHovered] = useState(false);
|
const [isHovered, setIsHovered] = useState(false);
|
||||||
|
|
||||||
@@ -21,10 +23,14 @@ const ScenarioRow = (props: {
|
|||||||
onMouseEnter={() => setIsHovered(true)}
|
onMouseEnter={() => setIsHovered(true)}
|
||||||
onMouseLeave={() => setIsHovered(false)}
|
onMouseLeave={() => setIsHovered(false)}
|
||||||
sx={isHovered ? highlightStyle : undefined}
|
sx={isHovered ? highlightStyle : undefined}
|
||||||
borderLeftWidth={1}
|
bgColor="white"
|
||||||
{...borders}
|
|
||||||
rowStart={props.rowStart}
|
rowStart={props.rowStart}
|
||||||
colStart={1}
|
colStart={1}
|
||||||
|
borderLeftWidth={1}
|
||||||
|
borderTopWidth={props.isFirst ? 1 : 0}
|
||||||
|
borderTopLeftRadius={props.isFirst ? 8 : 0}
|
||||||
|
borderBottomLeftRadius={props.isLast ? 8 : 0}
|
||||||
|
{...borders}
|
||||||
>
|
>
|
||||||
<ScenarioEditor scenario={props.scenario} hovered={isHovered} canHide={props.canHide} />
|
<ScenarioEditor scenario={props.scenario} hovered={isHovered} canHide={props.canHide} />
|
||||||
</GridItem>
|
</GridItem>
|
||||||
@@ -34,8 +40,12 @@ const ScenarioRow = (props: {
|
|||||||
onMouseEnter={() => setIsHovered(true)}
|
onMouseEnter={() => setIsHovered(true)}
|
||||||
onMouseLeave={() => setIsHovered(false)}
|
onMouseLeave={() => setIsHovered(false)}
|
||||||
sx={isHovered ? highlightStyle : undefined}
|
sx={isHovered ? highlightStyle : undefined}
|
||||||
|
bgColor="white"
|
||||||
rowStart={props.rowStart}
|
rowStart={props.rowStart}
|
||||||
colStart={i + 2}
|
colStart={i + 2}
|
||||||
|
borderTopWidth={props.isFirst ? 1 : 0}
|
||||||
|
borderTopRightRadius={props.isFirst && i === props.variants.length - 1 ? 8 : 0}
|
||||||
|
borderBottomRightRadius={props.isLast && i === props.variants.length - 1 ? 8 : 0}
|
||||||
{...borders}
|
{...borders}
|
||||||
>
|
>
|
||||||
<OutputCell key={variant.id} scenario={props.scenario} variant={variant} />
|
<OutputCell key={variant.id} scenario={props.scenario} variant={variant} />
|
||||||
|
|||||||
@@ -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,
|
||||||
@@ -19,15 +19,13 @@ import {
|
|||||||
useScenarios,
|
useScenarios,
|
||||||
} from "~/utils/hooks";
|
} from "~/utils/hooks";
|
||||||
import { BsGear, BsPencil, BsPlus, BsStars } from "react-icons/bs";
|
import { BsGear, BsPencil, BsPlus, BsStars } from "react-icons/bs";
|
||||||
import { useAppStore } from "~/state/store";
|
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
|
|
||||||
export const ActionButton = (props: ButtonProps) => (
|
export const ActionButton = (props: ButtonProps) => (
|
||||||
<Button size="sm" variant="ghost" color="gray.600" {...props} />
|
<Button size="sm" variant="ghost" color="gray.600" {...props} />
|
||||||
);
|
);
|
||||||
|
|
||||||
export const ScenariosHeader = () => {
|
export const ScenariosHeader = ({ openDrawer }: { openDrawer: () => void }) => {
|
||||||
const openDrawer = useAppStore((s) => s.openDrawer);
|
|
||||||
const { canModify } = useExperimentAccess();
|
const { canModify } = useExperimentAccess();
|
||||||
const scenarios = useScenarios();
|
const scenarios = useScenarios();
|
||||||
|
|
||||||
@@ -48,7 +46,7 @@ export const ScenariosHeader = () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HStack w="100%" pb={cellPadding.y} pt={0} align="center" spacing={0}>
|
<HStack w="100%" py={cellPadding.y} px={cellPadding.x} align="center" spacing={0}>
|
||||||
<Text fontSize={16} fontWeight="bold">
|
<Text fontSize={16} fontWeight="bold">
|
||||||
Scenarios ({scenarios.data?.count})
|
Scenarios ({scenarios.data?.count})
|
||||||
</Text>
|
</Text>
|
||||||
@@ -57,11 +55,16 @@ export const ScenariosHeader = () => {
|
|||||||
<MenuButton
|
<MenuButton
|
||||||
as={IconButton}
|
as={IconButton}
|
||||||
mt={1}
|
mt={1}
|
||||||
|
ml={2}
|
||||||
variant="ghost"
|
variant="ghost"
|
||||||
aria-label="Edit Scenarios"
|
aria-label="Edit Scenarios"
|
||||||
icon={<Icon as={loading ? Spinner : BsGear} />}
|
icon={<Icon as={loading ? Spinner : BsGear} />}
|
||||||
|
maxW={8}
|
||||||
|
minW={8}
|
||||||
|
minH={8}
|
||||||
|
maxH={8}
|
||||||
/>
|
/>
|
||||||
<MenuList fontSize="md" zIndex="dropdown" mt={-3}>
|
<MenuList fontSize="md" zIndex="dropdown" mt={-1}>
|
||||||
<MenuItem
|
<MenuItem
|
||||||
icon={<Icon as={BsPlus} boxSize={6} mx="-5px" />}
|
icon={<Icon as={BsPlus} boxSize={6} mx="-5px" />}
|
||||||
onClick={() => onAddScenario(false)}
|
onClick={() => onAddScenario(false)}
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import {
|
|||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { useCallback, useEffect, useRef, useState } from "react";
|
import { useCallback, useEffect, useRef, useState } from "react";
|
||||||
import { FiMaximize, FiMinimize } from "react-icons/fi";
|
import { FiMaximize, FiMinimize } from "react-icons/fi";
|
||||||
import { editorBackground } from "~/state/sharedVariantEditor.slice";
|
import { type CreatedEditor, editorBackground } from "~/state/sharedVariantEditor.slice";
|
||||||
import { useAppStore } from "~/state/store";
|
import { useAppStore } from "~/state/store";
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import {
|
import {
|
||||||
@@ -24,8 +24,10 @@ import { type PromptVariant } from "./types";
|
|||||||
export default function VariantEditor(props: { variant: PromptVariant }) {
|
export default function VariantEditor(props: { variant: PromptVariant }) {
|
||||||
const { canModify } = useExperimentAccess();
|
const { canModify } = useExperimentAccess();
|
||||||
const monaco = useAppStore.use.sharedVariantEditor.monaco();
|
const monaco = useAppStore.use.sharedVariantEditor.monaco();
|
||||||
const editorRef = useRef<ReturnType<NonNullable<typeof monaco>["editor"]["create"]> | null>(null);
|
const updateOptionsForEditor = useAppStore.use.sharedVariantEditor.updateOptionsForEditor();
|
||||||
|
const editorRef = useRef<CreatedEditor | null>(null);
|
||||||
const containerRef = useRef<HTMLDivElement | null>(null);
|
const containerRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
const lastSavedFnRef = useRef(props.variant.promptConstructor);
|
||||||
const [editorId] = useState(() => `editor_${Math.random().toString(36).substring(7)}`);
|
const [editorId] = useState(() => `editor_${Math.random().toString(36).substring(7)}`);
|
||||||
const [isChanged, setIsChanged] = useState(false);
|
const [isChanged, setIsChanged] = useState(false);
|
||||||
|
|
||||||
@@ -48,22 +50,18 @@ export default function VariantEditor(props: { variant: PromptVariant }) {
|
|||||||
}, [isFullscreen, toggleFullscreen]);
|
}, [isFullscreen, toggleFullscreen]);
|
||||||
|
|
||||||
const lastSavedFn = props.variant.promptConstructor;
|
const lastSavedFn = props.variant.promptConstructor;
|
||||||
|
useEffect(() => {
|
||||||
|
// Store in ref so that we can access it dynamically
|
||||||
|
lastSavedFnRef.current = lastSavedFn;
|
||||||
|
}, [lastSavedFn]);
|
||||||
|
|
||||||
const modifierKey = useModifierKeyLabel();
|
const modifierKey = useModifierKeyLabel();
|
||||||
|
|
||||||
const checkForChanges = useCallback(() => {
|
const checkForChanges = useCallback(() => {
|
||||||
if (!editorRef.current) return;
|
if (!editorRef.current) return;
|
||||||
const currentFn = editorRef.current.getValue();
|
const currentFn = editorRef.current.getValue();
|
||||||
setIsChanged(currentFn.length > 0 && currentFn !== lastSavedFn);
|
setIsChanged(currentFn.length > 0 && currentFn !== lastSavedFnRef.current);
|
||||||
}, [lastSavedFn]);
|
}, [editorRef]);
|
||||||
|
|
||||||
const matchUpdatedSavedFn = useCallback(() => {
|
|
||||||
if (!editorRef.current) return;
|
|
||||||
editorRef.current.setValue(lastSavedFn);
|
|
||||||
setIsChanged(false);
|
|
||||||
}, [lastSavedFn]);
|
|
||||||
|
|
||||||
useEffect(matchUpdatedSavedFn, [matchUpdatedSavedFn, lastSavedFn]);
|
|
||||||
|
|
||||||
const replaceVariant = api.promptVariants.replaceVariant.useMutation();
|
const replaceVariant = api.promptVariants.replaceVariant.useMutation();
|
||||||
const utils = api.useContext();
|
const utils = api.useContext();
|
||||||
@@ -110,7 +108,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) {
|
||||||
@@ -136,6 +134,11 @@ export default function VariantEditor(props: { variant: PromptVariant }) {
|
|||||||
readOnly: !canModify,
|
readOnly: !canModify,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
updateOptionsForEditor(props.variant.uiId, {
|
||||||
|
getContent: () => editorRef.current?.getValue() || "",
|
||||||
|
setContent: (content) => editorRef.current?.setValue(content),
|
||||||
|
});
|
||||||
|
|
||||||
// Workaround because otherwise the commands only work on whatever
|
// Workaround because otherwise the commands only work on whatever
|
||||||
// editor was loaded on the page last.
|
// editor was loaded on the page last.
|
||||||
// https://github.com/microsoft/monaco-editor/issues/2947#issuecomment-1422265201
|
// https://github.com/microsoft/monaco-editor/issues/2947#issuecomment-1422265201
|
||||||
@@ -155,7 +158,7 @@ export default function VariantEditor(props: { variant: PromptVariant }) {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
editorRef.current.onDidChangeModelContent(checkForChanges);
|
const checkForChangesListener = editorRef.current.onDidChangeModelContent(checkForChanges);
|
||||||
|
|
||||||
const resizeObserver = new ResizeObserver(() => {
|
const resizeObserver = new ResizeObserver(() => {
|
||||||
editorRef.current?.layout();
|
editorRef.current?.layout();
|
||||||
@@ -164,6 +167,7 @@ export default function VariantEditor(props: { variant: PromptVariant }) {
|
|||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
resizeObserver.disconnect();
|
resizeObserver.disconnect();
|
||||||
|
checkForChangesListener.dispose();
|
||||||
editorRef.current?.dispose();
|
editorRef.current?.dispose();
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -171,7 +175,7 @@ export default function VariantEditor(props: { variant: PromptVariant }) {
|
|||||||
// We intentionally skip the onSave and props.savedConfig dependencies here because
|
// We intentionally skip the onSave and props.savedConfig dependencies here because
|
||||||
// we don't want to re-render the editor from scratch
|
// we don't want to re-render the editor from scratch
|
||||||
/* eslint-disable-next-line react-hooks/exhaustive-deps */
|
/* eslint-disable-next-line react-hooks/exhaustive-deps */
|
||||||
}, [monaco, editorId]);
|
}, [monaco, editorId, updateOptionsForEditor]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!editorRef.current) return;
|
if (!editorRef.current) return;
|
||||||
|
|||||||
@@ -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(
|
||||||
@@ -84,6 +84,7 @@ export default function VariantHeader(
|
|||||||
>
|
>
|
||||||
<HStack
|
<HStack
|
||||||
spacing={2}
|
spacing={2}
|
||||||
|
py={2}
|
||||||
alignItems="flex-start"
|
alignItems="flex-start"
|
||||||
minH={headerMinHeight}
|
minH={headerMinHeight}
|
||||||
draggable={!isInputHovered}
|
draggable={!isInputHovered}
|
||||||
@@ -102,7 +103,9 @@ export default function VariantHeader(
|
|||||||
setIsDragTarget(false);
|
setIsDragTarget(false);
|
||||||
}}
|
}}
|
||||||
onDrop={onReorder}
|
onDrop={onReorder}
|
||||||
backgroundColor={isDragTarget ? "gray.200" : "gray.100"}
|
backgroundColor={isDragTarget ? "gray.200" : "white"}
|
||||||
|
borderTopLeftRadius={gridItemProps.borderTopLeftRadius}
|
||||||
|
borderTopRightRadius={gridItemProps.borderTopRightRadius}
|
||||||
h="full"
|
h="full"
|
||||||
>
|
>
|
||||||
<Icon
|
<Icon
|
||||||
@@ -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";
|
||||||
@@ -20,15 +20,19 @@ export default function VariantStats(props: { variant: PromptVariant }) {
|
|||||||
inputTokens: 0,
|
inputTokens: 0,
|
||||||
outputTokens: 0,
|
outputTokens: 0,
|
||||||
scenarioCount: 0,
|
scenarioCount: 0,
|
||||||
outputCount: 0,
|
finishedCount: 0,
|
||||||
|
awaitingCompletions: false,
|
||||||
awaitingEvals: false,
|
awaitingEvals: false,
|
||||||
},
|
},
|
||||||
refetchInterval,
|
refetchInterval,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
// Poll every two seconds while we are waiting for LLM retrievals to finish
|
// Poll every five seconds while we are waiting for LLM retrievals to finish
|
||||||
useEffect(() => setRefetchInterval(data.awaitingEvals ? 5000 : 0), [data.awaitingEvals]);
|
useEffect(
|
||||||
|
() => setRefetchInterval(data.awaitingCompletions || data.awaitingEvals ? 5000 : 0),
|
||||||
|
[data.awaitingCompletions, data.awaitingEvals],
|
||||||
|
);
|
||||||
|
|
||||||
const [passColor, neutralColor, failColor] = useToken("colors", [
|
const [passColor, neutralColor, failColor] = useToken("colors", [
|
||||||
"green.500",
|
"green.500",
|
||||||
@@ -38,7 +42,7 @@ export default function VariantStats(props: { variant: PromptVariant }) {
|
|||||||
|
|
||||||
const scale = chroma.scale([failColor, neutralColor, passColor]).domain([0, 0.5, 1]);
|
const scale = chroma.scale([failColor, neutralColor, passColor]).domain([0, 0.5, 1]);
|
||||||
|
|
||||||
const showNumFinished = data.scenarioCount > 0 && data.scenarioCount !== data.outputCount;
|
const showNumFinished = data.scenarioCount > 0 && data.scenarioCount !== data.finishedCount;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<HStack
|
<HStack
|
||||||
@@ -51,7 +55,7 @@ export default function VariantStats(props: { variant: PromptVariant }) {
|
|||||||
<HStack px={cellPadding.x} flexWrap="wrap">
|
<HStack px={cellPadding.x} flexWrap="wrap">
|
||||||
{showNumFinished && (
|
{showNumFinished && (
|
||||||
<Text>
|
<Text>
|
||||||
{data.outputCount} / {data.scenarioCount}
|
{data.finishedCount} / {data.scenarioCount}
|
||||||
</Text>
|
</Text>
|
||||||
)}
|
)}
|
||||||
{data.evalResults.map((result) => {
|
{data.evalResults.map((result) => {
|
||||||
|
|||||||
@@ -3,21 +3,29 @@ 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,
|
||||||
|
openDrawer,
|
||||||
|
}: {
|
||||||
|
experimentId: string | undefined;
|
||||||
|
openDrawer: () => void;
|
||||||
|
}) {
|
||||||
const variants = api.promptVariants.list.useQuery(
|
const variants = api.promptVariants.list.useQuery(
|
||||||
{ experimentId: experimentId as string },
|
{ experimentId: experimentId as string },
|
||||||
{ enabled: !!experimentId },
|
{ enabled: !!experimentId },
|
||||||
);
|
);
|
||||||
|
|
||||||
const scenarios = useScenarios();
|
const scenarios = useScenarios();
|
||||||
|
const shouldFlattenHeader = useScrolledPast(50);
|
||||||
|
|
||||||
if (!variants.data || !scenarios.data) return null;
|
if (!variants.data || !scenarios.data) return null;
|
||||||
|
|
||||||
@@ -53,20 +61,30 @@ export default function OutputsTable({ experimentId }: { experimentId: string |
|
|||||||
colStart: i + 2,
|
colStart: i + 2,
|
||||||
borderLeftWidth: i === 0 ? 1 : 0,
|
borderLeftWidth: i === 0 ? 1 : 0,
|
||||||
marginLeft: i === 0 ? "-1px" : 0,
|
marginLeft: i === 0 ? "-1px" : 0,
|
||||||
backgroundColor: "gray.100",
|
backgroundColor: "white",
|
||||||
};
|
};
|
||||||
|
const isFirst = i === 0;
|
||||||
|
const isLast = i === variants.data.length - 1;
|
||||||
return (
|
return (
|
||||||
<Fragment key={variant.uiId}>
|
<Fragment key={variant.uiId}>
|
||||||
<VariantHeader
|
<VariantHeader
|
||||||
variant={variant}
|
variant={variant}
|
||||||
canHide={variants.data.length > 1}
|
canHide={variants.data.length > 1}
|
||||||
rowStart={1}
|
rowStart={1}
|
||||||
|
borderTopLeftRadius={isFirst && !shouldFlattenHeader ? 8 : 0}
|
||||||
|
borderTopRightRadius={isLast && !shouldFlattenHeader ? 8 : 0}
|
||||||
{...sharedProps}
|
{...sharedProps}
|
||||||
/>
|
/>
|
||||||
<GridItem rowStart={2} {...sharedProps}>
|
<GridItem rowStart={2} {...sharedProps}>
|
||||||
<VariantEditor variant={variant} />
|
<VariantEditor variant={variant} />
|
||||||
</GridItem>
|
</GridItem>
|
||||||
<GridItem rowStart={3} {...sharedProps}>
|
<GridItem
|
||||||
|
rowStart={3}
|
||||||
|
{...sharedProps}
|
||||||
|
borderBottomLeftRadius={isFirst ? 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>
|
||||||
</Fragment>
|
</Fragment>
|
||||||
@@ -77,10 +95,9 @@ export default function OutputsTable({ experimentId }: { experimentId: string |
|
|||||||
colSpan={allCols - 1}
|
colSpan={allCols - 1}
|
||||||
rowStart={variantHeaderRows + 1}
|
rowStart={variantHeaderRows + 1}
|
||||||
colStart={1}
|
colStart={1}
|
||||||
{...borders}
|
|
||||||
borderRightWidth={0}
|
borderRightWidth={0}
|
||||||
>
|
>
|
||||||
<ScenariosHeader />
|
<ScenariosHeader openDrawer={openDrawer} />
|
||||||
</GridItem>
|
</GridItem>
|
||||||
|
|
||||||
{scenarios.data.scenarios.map((scenario, i) => (
|
{scenarios.data.scenarios.map((scenario, i) => (
|
||||||
@@ -90,6 +107,8 @@ export default function OutputsTable({ experimentId }: { experimentId: string |
|
|||||||
scenario={scenario}
|
scenario={scenario}
|
||||||
variants={variants.data}
|
variants={variants.data}
|
||||||
canHide={visibleScenariosCount > 1}
|
canHide={visibleScenariosCount > 1}
|
||||||
|
isFirst={i === 0}
|
||||||
|
isLast={i === visibleScenariosCount - 1}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
<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,77 +1,126 @@
|
|||||||
import { Box, HStack, IconButton } from "@chakra-ui/react";
|
|
||||||
import {
|
import {
|
||||||
BsChevronDoubleLeft,
|
HStack,
|
||||||
BsChevronDoubleRight,
|
IconButton,
|
||||||
BsChevronLeft,
|
Text,
|
||||||
BsChevronRight,
|
Select,
|
||||||
} from "react-icons/bs";
|
type StackProps,
|
||||||
import { usePage } from "~/utils/hooks";
|
Icon,
|
||||||
|
useBreakpointValue,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import React, { useCallback } from "react";
|
||||||
|
import { FiChevronsLeft, FiChevronsRight, FiChevronLeft, FiChevronRight } from "react-icons/fi";
|
||||||
|
import { usePageParams } from "~/utils/hooks";
|
||||||
|
|
||||||
const Paginator = ({
|
const pageSizeOptions = [10, 25, 50, 100];
|
||||||
numItemsLoaded,
|
|
||||||
startIndex,
|
const Paginator = ({ count, ...props }: { count: number; condense?: boolean } & StackProps) => {
|
||||||
lastPage,
|
const { page, pageSize, setPageParams } = usePageParams();
|
||||||
count,
|
|
||||||
}: {
|
const lastPage = Math.ceil(count / pageSize);
|
||||||
numItemsLoaded: number;
|
|
||||||
startIndex: number;
|
const updatePageSize = useCallback(
|
||||||
lastPage: number;
|
(newPageSize: number) => {
|
||||||
count: number;
|
const newPage = Math.floor(((page - 1) * pageSize) / newPageSize) + 1;
|
||||||
}) => {
|
setPageParams({ page: newPage, pageSize: newPageSize }, "replace");
|
||||||
const [page, setPage] = usePage();
|
},
|
||||||
|
[page, pageSize, setPageParams],
|
||||||
|
);
|
||||||
|
|
||||||
const nextPage = () => {
|
const nextPage = () => {
|
||||||
if (page < lastPage) {
|
if (page < lastPage) {
|
||||||
setPage(page + 1, "replace");
|
setPageParams({ page: page + 1 }, "replace");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const prevPage = () => {
|
const prevPage = () => {
|
||||||
if (page > 1) {
|
if (page > 1) {
|
||||||
setPage(page - 1, "replace");
|
setPageParams({ page: page - 1 }, "replace");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const goToLastPage = () => setPage(lastPage, "replace");
|
const goToLastPage = () => setPageParams({ page: lastPage }, "replace");
|
||||||
const goToFirstPage = () => setPage(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 pt={4}>
|
<HStack
|
||||||
<IconButton
|
pt={4}
|
||||||
variant="ghost"
|
spacing={8}
|
||||||
size="sm"
|
justifyContent={condense ? "flex-start" : "space-between"}
|
||||||
onClick={goToFirstPage}
|
alignItems="center"
|
||||||
isDisabled={page === 1}
|
w="full"
|
||||||
aria-label="Go to first page"
|
{...props}
|
||||||
icon={<BsChevronDoubleLeft />}
|
>
|
||||||
/>
|
{!condense && (
|
||||||
<IconButton
|
<>
|
||||||
variant="ghost"
|
<HStack>
|
||||||
size="sm"
|
<Text>Rows</Text>
|
||||||
onClick={prevPage}
|
<Select
|
||||||
isDisabled={page === 1}
|
value={pageSize}
|
||||||
aria-label="Previous page"
|
onChange={(e) => updatePageSize(parseInt(e.target.value))}
|
||||||
icon={<BsChevronLeft />}
|
w={20}
|
||||||
/>
|
backgroundColor="white"
|
||||||
<Box>
|
>
|
||||||
{startIndex}-{startIndex + numItemsLoaded - 1} / {count}
|
{pageSizeOptions.map((option) => (
|
||||||
</Box>
|
<option key={option} value={option}>
|
||||||
<IconButton
|
{option}
|
||||||
variant="ghost"
|
</option>
|
||||||
size="sm"
|
))}
|
||||||
onClick={nextPage}
|
</Select>
|
||||||
isDisabled={page === lastPage}
|
</HStack>
|
||||||
aria-label="Next page"
|
<Text>
|
||||||
icon={<BsChevronRight />}
|
Page {page} of {lastPage}
|
||||||
/>
|
</Text>
|
||||||
<IconButton
|
</>
|
||||||
variant="ghost"
|
)}
|
||||||
size="sm"
|
|
||||||
onClick={goToLastPage}
|
<HStack>
|
||||||
isDisabled={page === lastPage}
|
<IconButton
|
||||||
aria-label="Go to last page"
|
variant="outline"
|
||||||
icon={<BsChevronDoubleRight />}
|
size="sm"
|
||||||
/>
|
onClick={goToFirstPage}
|
||||||
|
isDisabled={page === 1}
|
||||||
|
aria-label="Go to first page"
|
||||||
|
icon={<Icon as={FiChevronsLeft} boxSize={5} strokeWidth={1.5} />}
|
||||||
|
bgColor="white"
|
||||||
|
/>
|
||||||
|
<IconButton
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={prevPage}
|
||||||
|
isDisabled={page === 1}
|
||||||
|
aria-label="Previous page"
|
||||||
|
icon={<Icon as={FiChevronLeft} boxSize={5} strokeWidth={1.5} />}
|
||||||
|
bgColor="white"
|
||||||
|
/>
|
||||||
|
{condense && (
|
||||||
|
<Text>
|
||||||
|
Page {page} of {lastPage}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
<IconButton
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={nextPage}
|
||||||
|
isDisabled={page === lastPage}
|
||||||
|
aria-label="Next page"
|
||||||
|
icon={<Icon as={FiChevronRight} boxSize={5} strokeWidth={1.5} />}
|
||||||
|
bgColor="white"
|
||||||
|
/>
|
||||||
|
<IconButton
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={goToLastPage}
|
||||||
|
isDisabled={page === lastPage}
|
||||||
|
aria-label="Go to last page"
|
||||||
|
icon={<Icon as={FiChevronsRight} boxSize={5} strokeWidth={1.5} />}
|
||||||
|
bgColor="white"
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
</HStack>
|
</HStack>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { useState, useMemo, useCallback } from "react";
|
||||||
import {
|
import {
|
||||||
Button,
|
Button,
|
||||||
Modal,
|
Modal,
|
||||||
@@ -9,22 +10,23 @@ import {
|
|||||||
ModalOverlay,
|
ModalOverlay,
|
||||||
VStack,
|
VStack,
|
||||||
Text,
|
Text,
|
||||||
Spinner,
|
|
||||||
HStack,
|
HStack,
|
||||||
Icon,
|
Icon,
|
||||||
SimpleGrid,
|
SimpleGrid,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { BsStars } from "react-icons/bs";
|
import { BsStars } from "react-icons/bs";
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import { useHandledAsyncCallback, useVisibleScenarioIds } from "~/utils/hooks";
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
import { type PromptVariant } from "@prisma/client";
|
import { type PromptVariant } from "@prisma/client";
|
||||||
import { useState } from "react";
|
|
||||||
import CompareFunctions from "./CompareFunctions";
|
import CompareFunctions from "./CompareFunctions";
|
||||||
import { CustomInstructionsInput } from "../CustomInstructionsInput";
|
import { CustomInstructionsInput } from "../CustomInstructionsInput";
|
||||||
import { RefineAction } from "./RefineAction";
|
import { RefineAction } from "./RefineAction";
|
||||||
import { isObject, isString } from "lodash-es";
|
import { isString } from "lodash-es";
|
||||||
import { type RefinementAction, type SupportedProvider } from "~/modelProviders/types";
|
import { type RefinementAction, type SupportedProvider } from "~/modelProviders/types";
|
||||||
import frontendModelProviders from "~/modelProviders/frontendModelProviders";
|
import frontendModelProviders from "~/modelProviders/frontendModelProviders";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
|
|
||||||
export const RefinePromptModal = ({
|
export const RefinePromptModal = ({
|
||||||
variant,
|
variant,
|
||||||
@@ -33,19 +35,23 @@ export const RefinePromptModal = ({
|
|||||||
variant: PromptVariant;
|
variant: PromptVariant;
|
||||||
onClose: () => void;
|
onClose: () => void;
|
||||||
}) => {
|
}) => {
|
||||||
const utils = api.useContext();
|
const editorOptionsMap = useAppStore((s) => s.sharedVariantEditor.editorOptionsMap);
|
||||||
const visibleScenarios = useVisibleScenarioIds();
|
const originalPromptFn = useMemo(
|
||||||
|
() => editorOptionsMap[variant.uiId]?.getContent() || "",
|
||||||
|
[editorOptionsMap, variant.uiId],
|
||||||
|
);
|
||||||
|
|
||||||
const refinementActions =
|
const refinementActions =
|
||||||
frontendModelProviders[variant.modelProvider as SupportedProvider].refinementActions || {};
|
frontendModelProviders[variant.modelProvider as SupportedProvider].refinementActions || {};
|
||||||
|
|
||||||
const { mutateAsync: getModifiedPromptMutateAsync, data: refinedPromptFn } =
|
const { mutateAsync: getModifiedPromptMutateAsync } =
|
||||||
api.promptVariants.getModifiedPromptFn.useMutation();
|
api.promptVariants.getModifiedPromptFn.useMutation();
|
||||||
const [instructions, setInstructions] = useState<string>("");
|
const [instructions, setInstructions] = useState<string>("");
|
||||||
|
|
||||||
const [activeRefineActionLabel, setActiveRefineActionLabel] = useState<string | undefined>(
|
const [activeRefineActionLabel, setActiveRefineActionLabel] = useState<string | undefined>(
|
||||||
undefined,
|
undefined,
|
||||||
);
|
);
|
||||||
|
const [refinedPromptFn, setRefinedPromptFn] = useState<string>();
|
||||||
|
|
||||||
const [getModifiedPromptFn, modificationInProgress] = useHandledAsyncCallback(
|
const [getModifiedPromptFn, modificationInProgress] = useHandledAsyncCallback(
|
||||||
async (label?: string) => {
|
async (label?: string) => {
|
||||||
@@ -54,31 +60,22 @@ export const RefinePromptModal = ({
|
|||||||
? (refinementActions[label] as RefinementAction).instructions
|
? (refinementActions[label] as RefinementAction).instructions
|
||||||
: instructions;
|
: instructions;
|
||||||
setActiveRefineActionLabel(label);
|
setActiveRefineActionLabel(label);
|
||||||
await getModifiedPromptMutateAsync({
|
const resp = await getModifiedPromptMutateAsync({
|
||||||
id: variant.id,
|
id: variant.id,
|
||||||
|
originalPromptFn,
|
||||||
instructions: updatedInstructions,
|
instructions: updatedInstructions,
|
||||||
});
|
});
|
||||||
|
if (maybeReportError(resp)) return;
|
||||||
|
setRefinedPromptFn(resp.payload);
|
||||||
},
|
},
|
||||||
[getModifiedPromptMutateAsync, onClose, variant, instructions, setActiveRefineActionLabel],
|
[getModifiedPromptMutateAsync, onClose, variant, instructions, setActiveRefineActionLabel],
|
||||||
);
|
);
|
||||||
|
|
||||||
const replaceVariantMutation = api.promptVariants.replaceVariant.useMutation();
|
const replaceVariant = useCallback(() => {
|
||||||
|
if (!refinedPromptFn) return;
|
||||||
const [replaceVariant, replacementInProgress] = useHandledAsyncCallback(async () => {
|
editorOptionsMap[variant.uiId]?.setContent(refinedPromptFn);
|
||||||
if (
|
|
||||||
!variant.experimentId ||
|
|
||||||
!refinedPromptFn ||
|
|
||||||
(isObject(refinedPromptFn) && "status" in refinedPromptFn)
|
|
||||||
)
|
|
||||||
return;
|
|
||||||
await replaceVariantMutation.mutateAsync({
|
|
||||||
id: variant.id,
|
|
||||||
promptConstructor: refinedPromptFn,
|
|
||||||
streamScenarios: visibleScenarios,
|
|
||||||
});
|
|
||||||
await utils.promptVariants.list.invalidate();
|
|
||||||
onClose();
|
onClose();
|
||||||
}, [replaceVariantMutation, variant, onClose, refinedPromptFn]);
|
}, [variant.uiId, editorOptionsMap, onClose, refinedPromptFn]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
@@ -126,7 +123,7 @@ export const RefinePromptModal = ({
|
|||||||
/>
|
/>
|
||||||
</VStack>
|
</VStack>
|
||||||
<CompareFunctions
|
<CompareFunctions
|
||||||
originalFunction={variant.promptConstructor}
|
originalFunction={originalPromptFn}
|
||||||
newFunction={isString(refinedPromptFn) ? refinedPromptFn : undefined}
|
newFunction={isString(refinedPromptFn) ? refinedPromptFn : undefined}
|
||||||
maxH="40vh"
|
maxH="40vh"
|
||||||
/>
|
/>
|
||||||
@@ -139,9 +136,9 @@ export const RefinePromptModal = ({
|
|||||||
colorScheme="blue"
|
colorScheme="blue"
|
||||||
onClick={replaceVariant}
|
onClick={replaceVariant}
|
||||||
minW={24}
|
minW={24}
|
||||||
isDisabled={replacementInProgress || !refinedPromptFn}
|
isDisabled={!refinedPromptFn}
|
||||||
>
|
>
|
||||||
{replacementInProgress ? <Spinner boxSize={4} /> : <Text>Accept</Text>}
|
Accept
|
||||||
</Button>
|
</Button>
|
||||||
</HStack>
|
</HStack>
|
||||||
</ModalFooter>
|
</ModalFooter>
|
||||||
|
|||||||
@@ -1,26 +0,0 @@
|
|||||||
import { VStack, HStack, type StackProps, Text, Divider } from "@chakra-ui/react";
|
|
||||||
import Link, { type LinkProps } from "next/link";
|
|
||||||
|
|
||||||
const StatsCard = ({
|
|
||||||
title,
|
|
||||||
href,
|
|
||||||
children,
|
|
||||||
...rest
|
|
||||||
}: { title: string; href: string } & StackProps & LinkProps) => {
|
|
||||||
return (
|
|
||||||
<VStack flex={1} borderWidth={1} padding={4} borderRadius={4} borderColor="gray.300" {...rest}>
|
|
||||||
<HStack w="full" justifyContent="space-between">
|
|
||||||
<Text fontSize="md" fontWeight="bold">
|
|
||||||
{title}
|
|
||||||
</Text>
|
|
||||||
<Link href={href}>
|
|
||||||
<Text color="blue">View all</Text>
|
|
||||||
</Link>
|
|
||||||
</HStack>
|
|
||||||
<Divider />
|
|
||||||
{children}
|
|
||||||
</VStack>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default StatsCard;
|
|
||||||
@@ -1,209 +0,0 @@
|
|||||||
import {
|
|
||||||
Box,
|
|
||||||
Card,
|
|
||||||
CardHeader,
|
|
||||||
Heading,
|
|
||||||
Table,
|
|
||||||
Tbody,
|
|
||||||
Td,
|
|
||||||
Th,
|
|
||||||
Thead,
|
|
||||||
Tr,
|
|
||||||
Tooltip,
|
|
||||||
Collapse,
|
|
||||||
HStack,
|
|
||||||
VStack,
|
|
||||||
IconButton,
|
|
||||||
useToast,
|
|
||||||
Icon,
|
|
||||||
Button,
|
|
||||||
ButtonGroup,
|
|
||||||
} from "@chakra-ui/react";
|
|
||||||
import dayjs from "dayjs";
|
|
||||||
import relativeTime from "dayjs/plugin/relativeTime";
|
|
||||||
import { ChevronUpIcon, ChevronDownIcon, CopyIcon } from "lucide-react";
|
|
||||||
import { useMemo, useState } from "react";
|
|
||||||
import { type RouterOutputs, api } from "~/utils/api";
|
|
||||||
import SyntaxHighlighter from "react-syntax-highlighter";
|
|
||||||
import { atelierCaveLight } from "react-syntax-highlighter/dist/cjs/styles/hljs";
|
|
||||||
import stringify from "json-stringify-pretty-compact";
|
|
||||||
import Link from "next/link";
|
|
||||||
|
|
||||||
dayjs.extend(relativeTime);
|
|
||||||
|
|
||||||
type LoggedCall = RouterOutputs["dashboard"]["loggedCalls"][0];
|
|
||||||
|
|
||||||
const FormattedJson = ({ json }: { json: any }) => {
|
|
||||||
const jsonString = stringify(json, { maxLength: 40 });
|
|
||||||
const toast = useToast();
|
|
||||||
|
|
||||||
const copyToClipboard = async (text: string) => {
|
|
||||||
try {
|
|
||||||
await navigator.clipboard.writeText(text);
|
|
||||||
toast({
|
|
||||||
title: "Copied to clipboard",
|
|
||||||
status: "success",
|
|
||||||
duration: 2000,
|
|
||||||
});
|
|
||||||
} catch (err) {
|
|
||||||
toast({
|
|
||||||
title: "Failed to copy to clipboard",
|
|
||||||
status: "error",
|
|
||||||
duration: 2000,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Box position="relative" fontSize="sm" borderRadius="md" overflow="hidden">
|
|
||||||
<SyntaxHighlighter
|
|
||||||
customStyle={{ overflowX: "unset" }}
|
|
||||||
language="json"
|
|
||||||
style={atelierCaveLight}
|
|
||||||
lineProps={{
|
|
||||||
style: { wordBreak: "break-all", whiteSpace: "pre-wrap" },
|
|
||||||
}}
|
|
||||||
wrapLines
|
|
||||||
>
|
|
||||||
{jsonString}
|
|
||||||
</SyntaxHighlighter>
|
|
||||||
<IconButton
|
|
||||||
aria-label="Copy"
|
|
||||||
icon={<CopyIcon />}
|
|
||||||
position="absolute"
|
|
||||||
top={1}
|
|
||||||
right={1}
|
|
||||||
size="xs"
|
|
||||||
variant="ghost"
|
|
||||||
onClick={() => void copyToClipboard(jsonString)}
|
|
||||||
/>
|
|
||||||
</Box>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
function TableRow({
|
|
||||||
loggedCall,
|
|
||||||
isExpanded,
|
|
||||||
onToggle,
|
|
||||||
}: {
|
|
||||||
loggedCall: LoggedCall;
|
|
||||||
isExpanded: boolean;
|
|
||||||
onToggle: () => void;
|
|
||||||
}) {
|
|
||||||
const isError = loggedCall.modelResponse?.statusCode !== 200;
|
|
||||||
const timeAgo = dayjs(loggedCall.requestedAt).fromNow();
|
|
||||||
const fullTime = dayjs(loggedCall.requestedAt).toString();
|
|
||||||
|
|
||||||
const model = useMemo(
|
|
||||||
() => loggedCall.tags.find((tag) => tag.name.startsWith("$model"))?.value,
|
|
||||||
[loggedCall.tags],
|
|
||||||
);
|
|
||||||
|
|
||||||
const durationCell = (
|
|
||||||
<Td isNumeric>
|
|
||||||
{loggedCall.cacheHit
|
|
||||||
? "Cache hit"
|
|
||||||
: ((loggedCall.modelResponse?.durationMs ?? 0) / 1000).toFixed(2) + "s"}
|
|
||||||
</Td>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Tr
|
|
||||||
onClick={onToggle}
|
|
||||||
key={loggedCall.id}
|
|
||||||
_hover={{ bgColor: "gray.100", cursor: "pointer" }}
|
|
||||||
sx={{
|
|
||||||
"> td": { borderBottom: "none" },
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Td>
|
|
||||||
<Icon boxSize={6} as={isExpanded ? ChevronUpIcon : ChevronDownIcon} />
|
|
||||||
</Td>
|
|
||||||
<Td>
|
|
||||||
<Tooltip label={fullTime} placement="top">
|
|
||||||
<Box whiteSpace="nowrap" minW="120px">
|
|
||||||
{timeAgo}
|
|
||||||
</Box>
|
|
||||||
</Tooltip>
|
|
||||||
</Td>
|
|
||||||
<Td width="100%">{model}</Td>
|
|
||||||
{durationCell}
|
|
||||||
<Td isNumeric>{loggedCall.modelResponse?.inputTokens}</Td>
|
|
||||||
<Td isNumeric>{loggedCall.modelResponse?.outputTokens}</Td>
|
|
||||||
<Td sx={{ color: isError ? "red.500" : "green.500", fontWeight: "semibold" }} isNumeric>
|
|
||||||
{loggedCall.modelResponse?.statusCode ?? "No response"}
|
|
||||||
</Td>
|
|
||||||
</Tr>
|
|
||||||
<Tr>
|
|
||||||
<Td colSpan={8} p={0}>
|
|
||||||
<Collapse in={isExpanded} unmountOnExit={true}>
|
|
||||||
<VStack p={4} align="stretch">
|
|
||||||
<HStack align="stretch">
|
|
||||||
<VStack flex={1} align="stretch">
|
|
||||||
<Heading size="sm">Input</Heading>
|
|
||||||
<FormattedJson json={loggedCall.modelResponse?.reqPayload} />
|
|
||||||
</VStack>
|
|
||||||
<VStack flex={1} align="stretch">
|
|
||||||
<Heading size="sm">Output</Heading>
|
|
||||||
<FormattedJson json={loggedCall.modelResponse?.respPayload} />
|
|
||||||
</VStack>
|
|
||||||
</HStack>
|
|
||||||
<ButtonGroup alignSelf="flex-end">
|
|
||||||
<Button as={Link} colorScheme="blue" href={{ pathname: "/experiments" }}>
|
|
||||||
Experiments
|
|
||||||
</Button>
|
|
||||||
</ButtonGroup>
|
|
||||||
</VStack>
|
|
||||||
</Collapse>
|
|
||||||
</Td>
|
|
||||||
</Tr>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default function LoggedCallTable() {
|
|
||||||
const [expandedRow, setExpandedRow] = useState<string | null>(null);
|
|
||||||
const loggedCalls = api.dashboard.loggedCalls.useQuery({});
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Card variant="outline" width="100%" overflow="hidden">
|
|
||||||
<CardHeader>
|
|
||||||
<Heading as="h3" size="sm">
|
|
||||||
Logged Calls
|
|
||||||
</Heading>
|
|
||||||
</CardHeader>
|
|
||||||
<Table>
|
|
||||||
<Thead>
|
|
||||||
<Tr>
|
|
||||||
<Th />
|
|
||||||
<Th>Time</Th>
|
|
||||||
<Th>Model</Th>
|
|
||||||
<Th isNumeric>Duration</Th>
|
|
||||||
<Th isNumeric>Input tokens</Th>
|
|
||||||
<Th isNumeric>Output tokens</Th>
|
|
||||||
<Th isNumeric>Status</Th>
|
|
||||||
</Tr>
|
|
||||||
</Thead>
|
|
||||||
<Tbody>
|
|
||||||
{loggedCalls.data?.map((loggedCall) => {
|
|
||||||
return (
|
|
||||||
<TableRow
|
|
||||||
key={loggedCall.id}
|
|
||||||
loggedCall={loggedCall}
|
|
||||||
isExpanded={loggedCall.id === expandedRow}
|
|
||||||
onToggle={() => {
|
|
||||||
if (loggedCall.id === expandedRow) {
|
|
||||||
setExpandedRow(null);
|
|
||||||
} else {
|
|
||||||
setExpandedRow(loggedCall.id);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</Tbody>
|
|
||||||
</Table>
|
|
||||||
</Card>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
51
app/src/components/dashboard/LoggedCallsTable.tsx
Normal file
51
app/src/components/dashboard/LoggedCallsTable.tsx
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import { Card, CardHeader, Heading, Table, Tbody, HStack, Button, Text } from "@chakra-ui/react";
|
||||||
|
import { useState } from "react";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useLoggedCalls } from "~/utils/hooks";
|
||||||
|
import { EmptyTableRow, TableHeader, TableRow } from "../requestLogs/TableRow";
|
||||||
|
|
||||||
|
export default function LoggedCallsTable() {
|
||||||
|
const { data: loggedCalls } = useLoggedCalls(false);
|
||||||
|
|
||||||
|
const [expandedRow, setExpandedRow] = useState<string | null>(null);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card width="100%" overflow="hidden">
|
||||||
|
<CardHeader>
|
||||||
|
<HStack justifyContent="space-between">
|
||||||
|
<Heading as="h3" size="sm">
|
||||||
|
Request Logs
|
||||||
|
</Heading>
|
||||||
|
<Button as={Link} href="/request-logs" variant="ghost" colorScheme="blue">
|
||||||
|
<Text>View All</Text>
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</CardHeader>
|
||||||
|
<Table>
|
||||||
|
<TableHeader />
|
||||||
|
<Tbody>
|
||||||
|
{loggedCalls?.calls.length ? (
|
||||||
|
loggedCalls?.calls.map((loggedCall) => {
|
||||||
|
return (
|
||||||
|
<TableRow
|
||||||
|
key={loggedCall.id}
|
||||||
|
loggedCall={loggedCall}
|
||||||
|
isExpanded={loggedCall.id === expandedRow}
|
||||||
|
onToggle={() => {
|
||||||
|
if (loggedCall.id === expandedRow) {
|
||||||
|
setExpandedRow(null);
|
||||||
|
} else {
|
||||||
|
setExpandedRow(loggedCall.id);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
) : (
|
||||||
|
<EmptyTableRow filtersApplied={false} />
|
||||||
|
)}
|
||||||
|
</Tbody>
|
||||||
|
</Table>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -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>
|
|
||||||
);
|
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
import {
|
||||||
|
Drawer,
|
||||||
|
DrawerBody,
|
||||||
|
DrawerCloseButton,
|
||||||
|
DrawerContent,
|
||||||
|
DrawerHeader,
|
||||||
|
DrawerOverlay,
|
||||||
|
Heading,
|
||||||
|
VStack,
|
||||||
|
type UseDisclosureReturn,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
|
||||||
|
import { DeleteButton } from "./DeleteButton";
|
||||||
|
|
||||||
|
export default function DatasetConfigurationDrawer({
|
||||||
|
disclosure,
|
||||||
|
}: {
|
||||||
|
disclosure: UseDisclosureReturn;
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<Drawer placement="right" size="md" {...disclosure}>
|
||||||
|
<DrawerOverlay />
|
||||||
|
<DrawerContent>
|
||||||
|
<DrawerCloseButton />
|
||||||
|
<DrawerHeader>
|
||||||
|
<Heading size="md">Dataset Configuration</Heading>
|
||||||
|
</DrawerHeader>
|
||||||
|
<DrawerBody h="full" pb={4}>
|
||||||
|
<VStack h="full" justifyContent="space-between">
|
||||||
|
<VStack spacing={6}></VStack>
|
||||||
|
<DeleteButton closeDrawer={disclosure.onClose} />
|
||||||
|
</VStack>
|
||||||
|
</DrawerBody>
|
||||||
|
</DrawerContent>
|
||||||
|
</Drawer>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
import { Button, Icon, useDisclosure, Text } from "@chakra-ui/react";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback, useDataset } from "~/utils/hooks";
|
||||||
|
import DeleteDatasetDialog from "./DeleteDatasetDialog";
|
||||||
|
|
||||||
|
export const DeleteButton = ({ closeDrawer }: { closeDrawer: () => void }) => {
|
||||||
|
const dataset = useDataset();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
const [onDelete] = useHandledAsyncCallback(async () => {
|
||||||
|
await router.push({ pathname: "/datasets" });
|
||||||
|
closeDrawer();
|
||||||
|
}, [router, closeDrawer]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
colorScheme="red"
|
||||||
|
fontWeight="normal"
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
>
|
||||||
|
<Icon as={BsTrash} boxSize={4} />
|
||||||
|
<Text ml={2}>Delete Dataset</Text>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<DeleteDatasetDialog
|
||||||
|
datasetId={dataset.data?.id}
|
||||||
|
onDelete={onDelete}
|
||||||
|
disclosure={disclosure}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
import { useRef } from "react";
|
||||||
|
import {
|
||||||
|
type UseDisclosureReturn,
|
||||||
|
AlertDialog,
|
||||||
|
AlertDialogOverlay,
|
||||||
|
AlertDialogContent,
|
||||||
|
AlertDialogHeader,
|
||||||
|
AlertDialogBody,
|
||||||
|
AlertDialogFooter,
|
||||||
|
Button,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
|
||||||
|
const DeleteDatasetDialog = ({
|
||||||
|
datasetId,
|
||||||
|
onDelete,
|
||||||
|
disclosure,
|
||||||
|
}: {
|
||||||
|
datasetId?: string;
|
||||||
|
onDelete?: () => void;
|
||||||
|
disclosure: UseDisclosureReturn;
|
||||||
|
}) => {
|
||||||
|
const cancelRef = useRef<HTMLButtonElement>(null);
|
||||||
|
|
||||||
|
const mutation = api.datasets.delete.useMutation();
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const [onDeleteConfirm, deletionInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!datasetId) return;
|
||||||
|
await mutation.mutateAsync({ id: datasetId });
|
||||||
|
await utils.datasets.list.invalidate();
|
||||||
|
onDelete?.();
|
||||||
|
|
||||||
|
disclosure.onClose();
|
||||||
|
}, [mutation, datasetId, disclosure.onClose]);
|
||||||
|
|
||||||
|
console.log("dataset id", datasetId);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<AlertDialog leastDestructiveRef={cancelRef} {...disclosure}>
|
||||||
|
<AlertDialogOverlay>
|
||||||
|
<AlertDialogContent>
|
||||||
|
<AlertDialogHeader fontSize="lg" fontWeight="bold">
|
||||||
|
Delete Dataset
|
||||||
|
</AlertDialogHeader>
|
||||||
|
|
||||||
|
<AlertDialogBody>
|
||||||
|
If you delete this dataset all the associated dataset entries will be deleted as well.
|
||||||
|
Are you sure?
|
||||||
|
</AlertDialogBody>
|
||||||
|
|
||||||
|
<AlertDialogFooter>
|
||||||
|
<Button ref={cancelRef} onClick={disclosure.onClose}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
colorScheme="red"
|
||||||
|
isLoading={deletionInProgress}
|
||||||
|
onClick={onDeleteConfirm}
|
||||||
|
ml={3}
|
||||||
|
>
|
||||||
|
Delete
|
||||||
|
</Button>
|
||||||
|
</AlertDialogFooter>
|
||||||
|
</AlertDialogContent>
|
||||||
|
</AlertDialogOverlay>
|
||||||
|
</AlertDialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DeleteDatasetDialog;
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
import { useDatasetEntries } from "~/utils/hooks";
|
|
||||||
import Paginator from "../Paginator";
|
|
||||||
|
|
||||||
const DatasetEntriesPaginator = () => {
|
|
||||||
const { data } = useDatasetEntries();
|
|
||||||
|
|
||||||
if (!data) return null;
|
|
||||||
|
|
||||||
const { entries, startIndex, lastPage, count } = data;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Paginator
|
|
||||||
numItemsLoaded={entries.length}
|
|
||||||
startIndex={startIndex}
|
|
||||||
lastPage={lastPage}
|
|
||||||
count={count}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
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;
|
|
||||||
@@ -0,0 +1,46 @@
|
|||||||
|
import { Card, Table, Tbody } from "@chakra-ui/react";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { useDatasetEntries } from "~/utils/hooks";
|
||||||
|
import { TableHeader, TableRow, EmptyTableRow } from "./TableRow";
|
||||||
|
import DatasetEntryEditorDrawer from "./DatasetEntryEditorDrawer";
|
||||||
|
|
||||||
|
export default function DatasetEntriesTable() {
|
||||||
|
const [expandedDatasetEntryId, setExpandedDatasetEntryId] = useState<string | null>(null);
|
||||||
|
const datasetEntries = useDatasetEntries().data?.entries;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Card width="100%" overflowX="auto">
|
||||||
|
<Table>
|
||||||
|
<TableHeader />
|
||||||
|
<Tbody>
|
||||||
|
{datasetEntries?.length ? (
|
||||||
|
datasetEntries?.map((entry) => {
|
||||||
|
return (
|
||||||
|
<TableRow
|
||||||
|
key={entry.id}
|
||||||
|
datasetEntry={entry}
|
||||||
|
onToggle={() => {
|
||||||
|
if (entry.id === expandedDatasetEntryId) {
|
||||||
|
setExpandedDatasetEntryId(null);
|
||||||
|
} else {
|
||||||
|
setExpandedDatasetEntryId(entry.id);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
showOptions
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
) : (
|
||||||
|
<EmptyTableRow />
|
||||||
|
)}
|
||||||
|
</Tbody>
|
||||||
|
</Table>
|
||||||
|
</Card>
|
||||||
|
<DatasetEntryEditorDrawer
|
||||||
|
datasetEntryId={expandedDatasetEntryId}
|
||||||
|
clearDatasetEntryId={() => setExpandedDatasetEntryId(null)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,174 @@
|
|||||||
|
import { useState, useEffect, useMemo } from "react";
|
||||||
|
import {
|
||||||
|
Drawer,
|
||||||
|
DrawerBody,
|
||||||
|
DrawerCloseButton,
|
||||||
|
DrawerContent,
|
||||||
|
DrawerHeader,
|
||||||
|
DrawerOverlay,
|
||||||
|
DrawerFooter,
|
||||||
|
Heading,
|
||||||
|
VStack,
|
||||||
|
HStack,
|
||||||
|
Button,
|
||||||
|
Text,
|
||||||
|
Divider,
|
||||||
|
Icon,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { type CreateChatCompletionRequestMessage } from "openai/resources/chat";
|
||||||
|
import { BsPlus } from "react-icons/bs";
|
||||||
|
import { type DatasetEntryType } from "@prisma/client";
|
||||||
|
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useDatasetEntry, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import EditableMessage from "./EditableMessage";
|
||||||
|
import EntryTypeDropdown from "./EntryTypeDropdown";
|
||||||
|
|
||||||
|
export default function DatasetDentryEditorDrawer({
|
||||||
|
datasetEntryId,
|
||||||
|
clearDatasetEntryId,
|
||||||
|
}: {
|
||||||
|
datasetEntryId: string | null;
|
||||||
|
clearDatasetEntryId: () => void;
|
||||||
|
}) {
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const datasetEntry = useDatasetEntry(datasetEntryId).data;
|
||||||
|
|
||||||
|
const savedInputMessages = useMemo(
|
||||||
|
() => datasetEntry?.input as unknown as CreateChatCompletionRequestMessage[],
|
||||||
|
[datasetEntry],
|
||||||
|
);
|
||||||
|
const savedOutputMessage = useMemo(
|
||||||
|
() => datasetEntry?.output as unknown as CreateChatCompletionRequestMessage,
|
||||||
|
[datasetEntry],
|
||||||
|
);
|
||||||
|
|
||||||
|
const [inputMessagesToSave, setInputMessagesToSave] = useState<
|
||||||
|
CreateChatCompletionRequestMessage[]
|
||||||
|
>([]);
|
||||||
|
const [outputMessageToSave, setOutputMessageToSave] =
|
||||||
|
useState<CreateChatCompletionRequestMessage | null>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (savedInputMessages) {
|
||||||
|
setInputMessagesToSave(savedInputMessages);
|
||||||
|
setOutputMessageToSave(savedOutputMessage);
|
||||||
|
}
|
||||||
|
}, [savedInputMessages, savedOutputMessage]);
|
||||||
|
|
||||||
|
const updateMutation = api.datasetEntries.update.useMutation();
|
||||||
|
const [onSave, savingInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!datasetEntryId || !inputMessagesToSave) return;
|
||||||
|
await updateMutation.mutateAsync({
|
||||||
|
id: datasetEntryId,
|
||||||
|
updates: {
|
||||||
|
input: JSON.stringify(inputMessagesToSave),
|
||||||
|
output: JSON.stringify(outputMessageToSave),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await utils.datasetEntries.list.invalidate();
|
||||||
|
await utils.datasetEntries.get.invalidate({ id: datasetEntryId });
|
||||||
|
}, [updateMutation, datasetEntryId, inputMessagesToSave, outputMessageToSave, utils]);
|
||||||
|
|
||||||
|
const [onUpdateType] = useHandledAsyncCallback(
|
||||||
|
async (type: DatasetEntryType) => {
|
||||||
|
if (!datasetEntryId) return;
|
||||||
|
await updateMutation.mutateAsync({
|
||||||
|
id: datasetEntryId,
|
||||||
|
updates: {
|
||||||
|
type,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await utils.datasetEntries.list.invalidate();
|
||||||
|
await utils.datasetEntries.get.invalidate({ id: datasetEntryId });
|
||||||
|
},
|
||||||
|
[updateMutation, datasetEntryId, utils],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Drawer isOpen={!!datasetEntryId} onClose={clearDatasetEntryId} placement="right" size="md">
|
||||||
|
<DrawerOverlay />
|
||||||
|
<DrawerContent>
|
||||||
|
<DrawerCloseButton pt={6} />
|
||||||
|
<DrawerHeader bgColor="orange.50">
|
||||||
|
<HStack w="full" justifyContent="space-between" pr={8}>
|
||||||
|
<Heading size="md">Dataset Entry</Heading>
|
||||||
|
{datasetEntry && (
|
||||||
|
<EntryTypeDropdown type={datasetEntry.type} onTypeChange={onUpdateType} />
|
||||||
|
)}
|
||||||
|
</HStack>
|
||||||
|
</DrawerHeader>
|
||||||
|
<DrawerBody h="full" pb={4} bgColor="orange.50">
|
||||||
|
<VStack h="full" justifyContent="space-between">
|
||||||
|
<VStack w="full" spacing={12} py={4}>
|
||||||
|
<VStack w="full" alignItems="flex-start">
|
||||||
|
<Text fontWeight="bold">Input</Text>
|
||||||
|
{inputMessagesToSave.map((message, i) => {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Divider key={`divider-${i}`} my={4} />
|
||||||
|
<EditableMessage
|
||||||
|
key={i}
|
||||||
|
message={message}
|
||||||
|
onEdit={(message) => {
|
||||||
|
const newInputMessages = [...inputMessagesToSave];
|
||||||
|
newInputMessages[i] = message;
|
||||||
|
setInputMessagesToSave(newInputMessages);
|
||||||
|
}}
|
||||||
|
onDelete={() => {
|
||||||
|
const newInputMessages = [...inputMessagesToSave];
|
||||||
|
newInputMessages.splice(i, 1);
|
||||||
|
setInputMessagesToSave(newInputMessages);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
<Divider my={4} />
|
||||||
|
<Button
|
||||||
|
w="full"
|
||||||
|
onClick={() =>
|
||||||
|
setInputMessagesToSave([...inputMessagesToSave, { role: "user", content: "" }])
|
||||||
|
}
|
||||||
|
variant="outline"
|
||||||
|
color="gray.500"
|
||||||
|
_hover={{ bgColor: "orange.100" }}
|
||||||
|
>
|
||||||
|
<HStack spacing={0}>
|
||||||
|
<Text>Add Message</Text>
|
||||||
|
<Icon as={BsPlus} boxSize={6} />
|
||||||
|
</HStack>
|
||||||
|
</Button>
|
||||||
|
</VStack>
|
||||||
|
<VStack w="full" alignItems="flex-start">
|
||||||
|
<Text fontWeight="bold">Output</Text>
|
||||||
|
<Divider my={4} />
|
||||||
|
<EditableMessage
|
||||||
|
message={outputMessageToSave}
|
||||||
|
onEdit={(message) => setOutputMessageToSave(message)}
|
||||||
|
isOutput
|
||||||
|
/>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
</DrawerBody>
|
||||||
|
<DrawerFooter bgColor="orange.50">
|
||||||
|
<HStack>
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
setInputMessagesToSave(savedInputMessages);
|
||||||
|
setOutputMessageToSave(savedOutputMessage);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Reset
|
||||||
|
</Button>
|
||||||
|
<Button isLoading={savingInProgress} onClick={onSave} colorScheme="orange">
|
||||||
|
Save
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</DrawerFooter>
|
||||||
|
</DrawerContent>
|
||||||
|
</Drawer>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,105 @@
|
|||||||
|
import { VStack, HStack, Tooltip, IconButton, Icon } from "@chakra-ui/react";
|
||||||
|
import { type CreateChatCompletionRequestMessage } from "openai/resources/chat";
|
||||||
|
import { BsX } from "react-icons/bs";
|
||||||
|
|
||||||
|
import AutoResizeTextArea from "~/components/AutoResizeTextArea";
|
||||||
|
import InputDropdown from "~/components/InputDropdown";
|
||||||
|
import { parseableToFunctionCall } from "~/utils/utils";
|
||||||
|
import FunctionCallEditor from "./FunctionCallEditor";
|
||||||
|
|
||||||
|
const MESSAGE_ROLE_OPTIONS = ["system", "user", "assistant", "function"] as const;
|
||||||
|
const OUTPUT_OPTIONS = ["plaintext", "func_call"] as const;
|
||||||
|
|
||||||
|
const EditableMessage = ({
|
||||||
|
message,
|
||||||
|
onEdit,
|
||||||
|
onDelete,
|
||||||
|
isOutput,
|
||||||
|
}: {
|
||||||
|
message: CreateChatCompletionRequestMessage | null;
|
||||||
|
onEdit: (message: CreateChatCompletionRequestMessage) => void;
|
||||||
|
onDelete?: () => void;
|
||||||
|
isOutput?: boolean;
|
||||||
|
}) => {
|
||||||
|
const { role = "assistant", content = "", function_call } = message || {};
|
||||||
|
|
||||||
|
const currentOutputOption: (typeof OUTPUT_OPTIONS)[number] = function_call
|
||||||
|
? "func_call"
|
||||||
|
: "plaintext";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack w="full">
|
||||||
|
<HStack w="full" justifyContent="space-between">
|
||||||
|
<HStack>
|
||||||
|
{!isOutput && (
|
||||||
|
<InputDropdown
|
||||||
|
options={MESSAGE_ROLE_OPTIONS}
|
||||||
|
selectedOption={role}
|
||||||
|
onSelect={(option) => {
|
||||||
|
const updatedMessage = { role: option, content };
|
||||||
|
if (role === "assistant" && currentOutputOption === "func_call") {
|
||||||
|
updatedMessage.content = JSON.stringify(function_call, null, 2);
|
||||||
|
}
|
||||||
|
onEdit(updatedMessage);
|
||||||
|
}}
|
||||||
|
inputGroupProps={{ w: "32", bgColor: "white" }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{role === "assistant" && (
|
||||||
|
<InputDropdown
|
||||||
|
options={OUTPUT_OPTIONS}
|
||||||
|
selectedOption={currentOutputOption}
|
||||||
|
onSelect={(option) => {
|
||||||
|
const updatedMessage: CreateChatCompletionRequestMessage = {
|
||||||
|
role,
|
||||||
|
content: null,
|
||||||
|
function_call: undefined,
|
||||||
|
};
|
||||||
|
if (option === "plaintext") {
|
||||||
|
updatedMessage.content = JSON.stringify(function_call, null, 2);
|
||||||
|
} else if (option === "func_call") {
|
||||||
|
updatedMessage.function_call =
|
||||||
|
content && parseableToFunctionCall(content)
|
||||||
|
? JSON.parse(content)
|
||||||
|
: { name: "", arguments: "{}" };
|
||||||
|
}
|
||||||
|
onEdit(updatedMessage);
|
||||||
|
}}
|
||||||
|
inputGroupProps={{ w: "32", bgColor: "white" }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</HStack>
|
||||||
|
{!isOutput && (
|
||||||
|
<HStack>
|
||||||
|
<Tooltip label="Delete" hasArrow>
|
||||||
|
<IconButton
|
||||||
|
aria-label="Delete"
|
||||||
|
icon={<Icon as={BsX} boxSize={6} />}
|
||||||
|
onClick={onDelete}
|
||||||
|
size="xs"
|
||||||
|
display="flex"
|
||||||
|
colorScheme="gray"
|
||||||
|
color="gray.500"
|
||||||
|
variant="ghost"
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</HStack>
|
||||||
|
)}
|
||||||
|
</HStack>
|
||||||
|
{function_call ? (
|
||||||
|
<FunctionCallEditor
|
||||||
|
function_call={function_call}
|
||||||
|
onEdit={(function_call) => onEdit({ role, function_call, content: null })}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<AutoResizeTextArea
|
||||||
|
value={content || JSON.stringify(function_call, null, 2)}
|
||||||
|
onChange={(e) => onEdit({ role, content: e.target.value })}
|
||||||
|
bgColor="white"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</VStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default EditableMessage;
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
import { type DatasetEntryType } from "@prisma/client";
|
||||||
|
|
||||||
|
import InputDropdown from "~/components/InputDropdown";
|
||||||
|
|
||||||
|
const ENTRY_TYPE_OPTIONS: DatasetEntryType[] = ["TRAIN", "TEST"];
|
||||||
|
|
||||||
|
const EntryTypeDropdown = ({
|
||||||
|
type,
|
||||||
|
onTypeChange,
|
||||||
|
}: {
|
||||||
|
type: DatasetEntryType;
|
||||||
|
onTypeChange: (type: DatasetEntryType) => void;
|
||||||
|
}) => {
|
||||||
|
return (
|
||||||
|
<InputDropdown
|
||||||
|
options={ENTRY_TYPE_OPTIONS}
|
||||||
|
selectedOption={type}
|
||||||
|
onSelect={onTypeChange}
|
||||||
|
inputGroupProps={{ w: "32", bgColor: "white" }}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default EntryTypeDropdown;
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
import { useRef, useMemo, useEffect } from "react";
|
||||||
|
import { VStack, HStack, Text, Input, Box } from "@chakra-ui/react";
|
||||||
|
import { type CreateChatCompletionRequestMessage } from "openai/resources/chat";
|
||||||
|
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { type CreatedEditor } from "~/state/sharedVariantEditor.slice";
|
||||||
|
|
||||||
|
const FunctionCallEditor = ({
|
||||||
|
function_call,
|
||||||
|
onEdit,
|
||||||
|
}: {
|
||||||
|
function_call: CreateChatCompletionRequestMessage.FunctionCall;
|
||||||
|
onEdit: (function_call: CreateChatCompletionRequestMessage.FunctionCall) => void;
|
||||||
|
}) => {
|
||||||
|
const monaco = useAppStore.use.sharedArgumentsEditor.monaco();
|
||||||
|
const editorRef = useRef<CreatedEditor | null>(null);
|
||||||
|
const editorId = useMemo(() => `editor_${Math.random().toString(36).substring(7)}`, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (monaco) {
|
||||||
|
const container = document.getElementById(editorId) as HTMLElement;
|
||||||
|
|
||||||
|
const editor = monaco.editor.create(container, {
|
||||||
|
value: function_call.arguments,
|
||||||
|
language: "json",
|
||||||
|
theme: "customTheme",
|
||||||
|
lineNumbers: "off",
|
||||||
|
minimap: { enabled: false },
|
||||||
|
wrappingIndent: "indent",
|
||||||
|
wrappingStrategy: "advanced",
|
||||||
|
wordWrap: "on",
|
||||||
|
folding: false,
|
||||||
|
scrollbar: {
|
||||||
|
alwaysConsumeMouseWheel: false,
|
||||||
|
verticalScrollbarSize: 0,
|
||||||
|
},
|
||||||
|
wordWrapBreakAfterCharacters: "",
|
||||||
|
wordWrapBreakBeforeCharacters: "",
|
||||||
|
quickSuggestions: true,
|
||||||
|
renderLineHighlight: "none",
|
||||||
|
fontSize: 14,
|
||||||
|
scrollBeyondLastLine: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
editorRef.current = editor;
|
||||||
|
|
||||||
|
const updateHeight = () => {
|
||||||
|
const contentHeight = editor.getContentHeight();
|
||||||
|
container.style.height = `${contentHeight}px`;
|
||||||
|
editor.layout();
|
||||||
|
};
|
||||||
|
|
||||||
|
const attemptDocumentFormat = () => {
|
||||||
|
const action = editor.getAction("editor.action.formatDocument");
|
||||||
|
if (action) {
|
||||||
|
action
|
||||||
|
.run()
|
||||||
|
.then(updateHeight)
|
||||||
|
.catch((error) => {
|
||||||
|
console.error("Error running formatDocument:", error);
|
||||||
|
});
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
editor.onDidBlurEditorText(() => {
|
||||||
|
attemptDocumentFormat();
|
||||||
|
onEdit({ name: function_call.name, arguments: editor.getValue() });
|
||||||
|
});
|
||||||
|
|
||||||
|
// Interval function to check for action availability
|
||||||
|
const checkForActionInterval = setInterval(() => {
|
||||||
|
const formatted = attemptDocumentFormat();
|
||||||
|
if (formatted) {
|
||||||
|
clearInterval(checkForActionInterval); // Clear the interval once the action is found and run
|
||||||
|
}
|
||||||
|
}, 100); // Check every 100ms
|
||||||
|
|
||||||
|
// Add content change listener
|
||||||
|
const contentChangeListener = editor.onDidChangeModelContent(updateHeight);
|
||||||
|
|
||||||
|
const resizeObserver = new ResizeObserver(() => {
|
||||||
|
editor.layout();
|
||||||
|
});
|
||||||
|
resizeObserver.observe(container);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
contentChangeListener.dispose();
|
||||||
|
resizeObserver.disconnect();
|
||||||
|
editor?.dispose();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}, [monaco, editorId, function_call.name, function_call.arguments, onEdit]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack w="full" alignItems="flex-start">
|
||||||
|
<HStack w="full">
|
||||||
|
<Text fontWeight="bold" w={192}>
|
||||||
|
Name:
|
||||||
|
</Text>
|
||||||
|
<Input
|
||||||
|
value={function_call.name}
|
||||||
|
onChange={(e) => onEdit({ name: e.target.value, arguments: function_call.arguments })}
|
||||||
|
bgColor="white"
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
|
<Text fontWeight="bold" w={32}>
|
||||||
|
Arguments
|
||||||
|
</Text>
|
||||||
|
<VStack
|
||||||
|
borderRadius={4}
|
||||||
|
border="1px solid"
|
||||||
|
borderColor="gray.200"
|
||||||
|
w="full"
|
||||||
|
py={1}
|
||||||
|
bgColor="white"
|
||||||
|
>
|
||||||
|
<Box id={editorId} w="full" />
|
||||||
|
</VStack>
|
||||||
|
</VStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FunctionCallEditor;
|
||||||
128
app/src/components/datasets/DatasetEntriesTable/TableRow.tsx
Normal file
128
app/src/components/datasets/DatasetEntriesTable/TableRow.tsx
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
import { Box, Td, Tr, Thead, Th, Tooltip, HStack, Text, Checkbox } from "@chakra-ui/react";
|
||||||
|
import Link from "next/link";
|
||||||
|
|
||||||
|
import dayjs from "~/utils/dayjs";
|
||||||
|
import { type RouterOutputs } from "~/utils/api";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import { useIsClientRehydrated, useDatasetEntries } from "~/utils/hooks";
|
||||||
|
import { useMemo } from "react";
|
||||||
|
|
||||||
|
type DatasetEntry = RouterOutputs["datasetEntries"]["list"]["entries"][0];
|
||||||
|
|
||||||
|
export const TableHeader = () => {
|
||||||
|
const matchingDatasetEntryIds = useDatasetEntries().data?.matchingEntryIds;
|
||||||
|
const selectedDatasetEntryIds = useAppStore((s) => s.selectedDatasetEntries.selectedIds);
|
||||||
|
const addSelectedIds = useAppStore((s) => s.selectedDatasetEntries.addSelectedIds);
|
||||||
|
const clearSelectedIds = useAppStore((s) => s.selectedDatasetEntries.clearSelectedIds);
|
||||||
|
const allSelected = useMemo(() => {
|
||||||
|
if (!matchingDatasetEntryIds || !matchingDatasetEntryIds.length) return false;
|
||||||
|
return matchingDatasetEntryIds.every((id) => selectedDatasetEntryIds.has(id));
|
||||||
|
}, [matchingDatasetEntryIds, selectedDatasetEntryIds]);
|
||||||
|
const isClientRehydrated = useIsClientRehydrated();
|
||||||
|
if (!isClientRehydrated) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Thead>
|
||||||
|
<Tr>
|
||||||
|
<Th pr={0}>
|
||||||
|
<HStack minW={16}>
|
||||||
|
<Checkbox
|
||||||
|
isChecked={allSelected}
|
||||||
|
onChange={() => {
|
||||||
|
allSelected ? clearSelectedIds() : addSelectedIds(matchingDatasetEntryIds || []);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Text>
|
||||||
|
({selectedDatasetEntryIds.size ? `${selectedDatasetEntryIds.size}/` : ""}
|
||||||
|
{matchingDatasetEntryIds?.length || 0})
|
||||||
|
</Text>
|
||||||
|
</HStack>
|
||||||
|
</Th>
|
||||||
|
<Th>Created At</Th>
|
||||||
|
<Th isNumeric>Input tokens</Th>
|
||||||
|
<Th isNumeric>Output tokens</Th>
|
||||||
|
<Th isNumeric>Type</Th>
|
||||||
|
</Tr>
|
||||||
|
</Thead>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const TableRow = ({
|
||||||
|
datasetEntry,
|
||||||
|
onToggle,
|
||||||
|
showOptions,
|
||||||
|
}: {
|
||||||
|
datasetEntry: DatasetEntry;
|
||||||
|
onToggle: () => void;
|
||||||
|
showOptions?: boolean;
|
||||||
|
}) => {
|
||||||
|
const createdAt = dayjs(datasetEntry.createdAt).format("MMMM D h:mm A");
|
||||||
|
const fullTime = dayjs(datasetEntry.createdAt).toString();
|
||||||
|
|
||||||
|
const isChecked = useAppStore((s) => s.selectedDatasetEntries.selectedIds.has(datasetEntry.id));
|
||||||
|
const toggleChecked = useAppStore((s) => s.selectedDatasetEntries.toggleSelectedId);
|
||||||
|
|
||||||
|
const isClientRehydrated = useIsClientRehydrated();
|
||||||
|
if (!isClientRehydrated) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tr
|
||||||
|
onClick={onToggle}
|
||||||
|
key={datasetEntry.id}
|
||||||
|
_hover={{ bgColor: "gray.50", cursor: "pointer" }}
|
||||||
|
fontSize="sm"
|
||||||
|
>
|
||||||
|
{showOptions && (
|
||||||
|
<Td>
|
||||||
|
<Checkbox isChecked={isChecked} onChange={() => toggleChecked(datasetEntry.id)} />
|
||||||
|
</Td>
|
||||||
|
)}
|
||||||
|
<Td>
|
||||||
|
<Tooltip label={fullTime} placement="top">
|
||||||
|
<Box whiteSpace="nowrap" minW="120px">
|
||||||
|
{createdAt}
|
||||||
|
</Box>
|
||||||
|
</Tooltip>
|
||||||
|
</Td>
|
||||||
|
<Td isNumeric>{datasetEntry.inputTokens}</Td>
|
||||||
|
<Td isNumeric>{datasetEntry.outputTokens}</Td>
|
||||||
|
<Td isNumeric>{datasetEntry.type}</Td>
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const EmptyTableRow = ({ filtersApplied = true }: { filtersApplied?: boolean }) => {
|
||||||
|
const visibleColumns = useAppStore((s) => s.columnVisibility.visibleColumns);
|
||||||
|
const filters = useAppStore((state) => state.logFilters.filters);
|
||||||
|
const { isLoading } = useDatasetEntries();
|
||||||
|
|
||||||
|
if (isLoading) return null;
|
||||||
|
|
||||||
|
if (filters.length && filtersApplied) {
|
||||||
|
return (
|
||||||
|
<Tr>
|
||||||
|
<Td w="full" colSpan={visibleColumns.size + 1}>
|
||||||
|
<Text color="gray.500" textAlign="center" w="full" p={4}>
|
||||||
|
No matching entries found. Try removing some filters.
|
||||||
|
</Text>
|
||||||
|
</Td>
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tr>
|
||||||
|
<Td w="full" colSpan={visibleColumns.size + 1}>
|
||||||
|
<Text color="gray.500" textAlign="center" w="full" p={4}>
|
||||||
|
This dataset has no entries. Add some logs in the{" "}
|
||||||
|
<Link href="/request-logs">
|
||||||
|
<Text as="span" color="blue.600">
|
||||||
|
Request Logs
|
||||||
|
</Text>
|
||||||
|
</Link>{" "}
|
||||||
|
tab.
|
||||||
|
</Text>
|
||||||
|
</Td>
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
};
|
||||||
16
app/src/components/datasets/DatasetEntryPaginator.tsx
Normal file
16
app/src/components/datasets/DatasetEntryPaginator.tsx
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import { type StackProps } from "@chakra-ui/react";
|
||||||
|
|
||||||
|
import { useDatasetEntries } from "~/utils/hooks";
|
||||||
|
import Paginator from "../Paginator";
|
||||||
|
|
||||||
|
const DatasetEntryPaginator = (props: StackProps) => {
|
||||||
|
const { data } = useDatasetEntries();
|
||||||
|
|
||||||
|
if (!data) return null;
|
||||||
|
|
||||||
|
const { matchingEntryIds } = data;
|
||||||
|
|
||||||
|
return <Paginator count={matchingEntryIds.length} {...props} />;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DatasetEntryPaginator;
|
||||||
20
app/src/components/datasets/DatasetHeaderButtons.tsx
Normal file
20
app/src/components/datasets/DatasetHeaderButtons.tsx
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { Button, HStack, Icon, Text } from "@chakra-ui/react";
|
||||||
|
import { useDataset } from "~/utils/hooks";
|
||||||
|
import { BsGearFill } from "react-icons/bs";
|
||||||
|
|
||||||
|
export const DatasetHeaderButtons = ({ openDrawer }: { openDrawer: () => void }) => {
|
||||||
|
const dataset = useDataset();
|
||||||
|
|
||||||
|
if (dataset.isLoading) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<HStack spacing={0} mt={{ base: 2, md: 0 }}>
|
||||||
|
<Button variant={{ base: "solid", md: "ghost" }} onClick={openDrawer}>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={BsGearFill} />
|
||||||
|
<Text>Configure</Text>
|
||||||
|
</HStack>
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -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>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
52
app/src/components/datasets/DatasetsTable.tsx
Normal file
52
app/src/components/datasets/DatasetsTable.tsx
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import { Card, Table, Thead, Tr, Th, Tbody, Td, VStack, Icon, Text } from "@chakra-ui/react";
|
||||||
|
import { FaTable } from "react-icons/fa";
|
||||||
|
import Link from "next/link";
|
||||||
|
|
||||||
|
import dayjs from "~/utils/dayjs";
|
||||||
|
import { useDatasets } from "~/utils/hooks";
|
||||||
|
|
||||||
|
const DatasetsTable = ({}) => {
|
||||||
|
const { data } = useDatasets();
|
||||||
|
|
||||||
|
const datasets = data || [];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card width="100%" overflowX="auto">
|
||||||
|
{datasets.length ? (
|
||||||
|
<Table>
|
||||||
|
<Thead>
|
||||||
|
<Tr>
|
||||||
|
<Th>Name</Th>
|
||||||
|
<Th>Created At</Th>
|
||||||
|
<Th>Size</Th>
|
||||||
|
</Tr>
|
||||||
|
</Thead>
|
||||||
|
<Tbody>
|
||||||
|
{datasets.map((dataset) => {
|
||||||
|
return (
|
||||||
|
<Tr key={dataset.id}>
|
||||||
|
<Td>
|
||||||
|
<Link href={{ pathname: "/datasets/[id]", query: { id: dataset.id } }}>
|
||||||
|
<Text color="blue.600">{dataset.name}</Text>
|
||||||
|
</Link>
|
||||||
|
</Td>
|
||||||
|
<Td>{dayjs(dataset.createdAt).format("MMMM D h:mm A")}</Td>
|
||||||
|
<Td>{dataset._count.datasetEntries}</Td>
|
||||||
|
</Tr>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</Tbody>
|
||||||
|
</Table>
|
||||||
|
) : (
|
||||||
|
<VStack py={8}>
|
||||||
|
<Icon as={FaTable} boxSize={16} color="gray.300" />
|
||||||
|
<Text color="gray.400" fontSize="lg" fontWeight="bold">
|
||||||
|
No Datasets Found. Create your first dataset.
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DatasetsTable;
|
||||||
107
app/src/components/datasets/DeleteButton.tsx
Normal file
107
app/src/components/datasets/DeleteButton.tsx
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
import {
|
||||||
|
Modal,
|
||||||
|
ModalOverlay,
|
||||||
|
ModalContent,
|
||||||
|
ModalHeader,
|
||||||
|
ModalCloseButton,
|
||||||
|
ModalBody,
|
||||||
|
ModalFooter,
|
||||||
|
HStack,
|
||||||
|
VStack,
|
||||||
|
Icon,
|
||||||
|
Text,
|
||||||
|
Button,
|
||||||
|
useDisclosure,
|
||||||
|
type UseDisclosureReturn,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback, useDataset } from "~/utils/hooks";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import ActionButton from "../ActionButton";
|
||||||
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
|
import pluralize from "pluralize";
|
||||||
|
|
||||||
|
const DeleteButton = () => {
|
||||||
|
const selectedIds = useAppStore((s) => s.selectedDatasetEntries.selectedIds);
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ActionButton
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
label="Delete"
|
||||||
|
icon={BsTrash}
|
||||||
|
isDisabled={selectedIds.size === 0}
|
||||||
|
requireBeta
|
||||||
|
/>
|
||||||
|
<DeleteDatasetEntriesModal disclosure={disclosure} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default DeleteButton;
|
||||||
|
|
||||||
|
const DeleteDatasetEntriesModal = ({ disclosure }: { disclosure: UseDisclosureReturn }) => {
|
||||||
|
const dataset = useDataset().data;
|
||||||
|
const selectedIds = useAppStore((s) => s.selectedDatasetEntries.selectedIds);
|
||||||
|
const clearSelectedIds = useAppStore((s) => s.selectedDatasetEntries.clearSelectedIds);
|
||||||
|
|
||||||
|
const deleteRowsMutation = api.datasetEntries.delete.useMutation();
|
||||||
|
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const [deleteRows, deletionInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!dataset?.id || !selectedIds.size) return;
|
||||||
|
|
||||||
|
// divide selectedIds into chunks of 15000 to reduce request size
|
||||||
|
const chunkSize = 15000;
|
||||||
|
const idsArray = Array.from(selectedIds);
|
||||||
|
for (let i = 0; i < idsArray.length; i += chunkSize) {
|
||||||
|
const response = await deleteRowsMutation.mutateAsync({
|
||||||
|
ids: idsArray.slice(i, i + chunkSize),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (maybeReportError(response)) return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await utils.datasetEntries.list.invalidate();
|
||||||
|
disclosure.onClose();
|
||||||
|
clearSelectedIds();
|
||||||
|
}, [deleteRowsMutation, dataset, selectedIds, utils]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal size={{ base: "xl", md: "2xl" }} {...disclosure}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={BsTrash} />
|
||||||
|
<Text>Delete Logs</Text>
|
||||||
|
</HStack>
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalCloseButton />
|
||||||
|
<ModalBody maxW="unset">
|
||||||
|
<VStack w="full" spacing={8} pt={4} alignItems="flex-start">
|
||||||
|
<Text>
|
||||||
|
Are you sure you want to delete the <b>{selectedIds.size}</b>{" "}
|
||||||
|
{pluralize("row", selectedIds.size)} rows you've selected?
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter>
|
||||||
|
<HStack>
|
||||||
|
<Button colorScheme="gray" onClick={disclosure.onClose} minW={24}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button colorScheme="red" onClick={deleteRows} isLoading={deletionInProgress} minW={24}>
|
||||||
|
Delete
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
182
app/src/components/datasets/DownloadButton.tsx
Normal file
182
app/src/components/datasets/DownloadButton.tsx
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
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 { AiOutlineDownload } from "react-icons/ai";
|
||||||
|
|
||||||
|
import { useHandledAsyncCallback, useDataset } from "~/utils/hooks";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import ActionButton from "../ActionButton";
|
||||||
|
import { FiChevronUp, FiChevronDown } from "react-icons/fi";
|
||||||
|
import InfoCircle from "../InfoCircle";
|
||||||
|
|
||||||
|
const ExportButton = () => {
|
||||||
|
const selectedIds = useAppStore((s) => s.selectedDatasetEntries.selectedIds);
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ActionButton
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
label="Download"
|
||||||
|
icon={AiOutlineDownload}
|
||||||
|
isDisabled={selectedIds.size === 0}
|
||||||
|
requireBeta
|
||||||
|
/>
|
||||||
|
<ExportDatasetEntriesModal disclosure={disclosure} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ExportButton;
|
||||||
|
|
||||||
|
const ExportDatasetEntriesModal = ({ disclosure }: { disclosure: UseDisclosureReturn }) => {
|
||||||
|
const dataset = useDataset().data;
|
||||||
|
const selectedIds = useAppStore((s) => s.selectedDatasetEntries.selectedIds);
|
||||||
|
const clearSelectedIds = useAppStore((s) => s.selectedDatasetEntries.clearSelectedIds);
|
||||||
|
|
||||||
|
const [testingSplit, setTestingSplit] = useState(10);
|
||||||
|
const [removeDuplicates, setRemoveDuplicates] = useState(false);
|
||||||
|
const [showAdvancedOptions, setShowAdvancedOptions] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (disclosure.isOpen) {
|
||||||
|
setTestingSplit(10);
|
||||||
|
setRemoveDuplicates(false);
|
||||||
|
}
|
||||||
|
}, [disclosure.isOpen]);
|
||||||
|
|
||||||
|
const exportDataMutation = api.datasetEntries.export.useMutation();
|
||||||
|
|
||||||
|
const [exportData, exportInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!dataset?.id || !selectedIds.size || !testingSplit) return;
|
||||||
|
const response = await exportDataMutation.mutateAsync({
|
||||||
|
datasetId: dataset.id,
|
||||||
|
datasetEntryIds: Array.from(selectedIds),
|
||||||
|
testingSplit,
|
||||||
|
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();
|
||||||
|
clearSelectedIds();
|
||||||
|
}, [exportDataMutation, dataset, selectedIds, testingSplit, removeDuplicates]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal size={{ base: "xl", md: "2xl" }} {...disclosure}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={AiOutlineDownload} />
|
||||||
|
<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>{selectedIds.size}</b> rows you have selected in the OpenAI
|
||||||
|
training format.
|
||||||
|
</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">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={exportData} isLoading={exportInProgress} minW={24}>
|
||||||
|
Download
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
21
app/src/components/datasets/ExperimentButton.tsx
Normal file
21
app/src/components/datasets/ExperimentButton.tsx
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import { RiFlaskLine } from "react-icons/ri";
|
||||||
|
|
||||||
|
import { useAppStore } from "~/state/store";
|
||||||
|
import ActionButton from "../ActionButton";
|
||||||
|
|
||||||
|
const ExperimentButton = () => {
|
||||||
|
const selectedIds = useAppStore((s) => s.selectedDatasetEntries.selectedIds);
|
||||||
|
return (
|
||||||
|
<ActionButton
|
||||||
|
onClick={() => {
|
||||||
|
console.log("experimenting with these ids", selectedIds);
|
||||||
|
}}
|
||||||
|
label="Experiment"
|
||||||
|
icon={RiFlaskLine}
|
||||||
|
isDisabled={selectedIds.size === 0}
|
||||||
|
requireBeta
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ExperimentButton;
|
||||||
148
app/src/components/datasets/FileUploadsCard.tsx
Normal file
148
app/src/components/datasets/FileUploadsCard.tsx
Normal file
@@ -0,0 +1,148 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
|
import { VStack, HStack, Button, Text, Progress, IconButton, Portal } from "@chakra-ui/react";
|
||||||
|
import { BsX } from "react-icons/bs";
|
||||||
|
|
||||||
|
import { type RouterOutputs, api } from "~/utils/api";
|
||||||
|
import { useDataset, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import { formatFileSize } from "~/utils/utils";
|
||||||
|
|
||||||
|
type FileUpload = RouterOutputs["datasets"]["listFileUploads"][0];
|
||||||
|
|
||||||
|
const FileUploadsCard = () => {
|
||||||
|
const dataset = useDataset();
|
||||||
|
const [fileUploadsRefetchInterval, setFileUploadsRefetchInterval] = useState<number>(500);
|
||||||
|
const fileUploads = api.datasets.listFileUploads.useQuery(
|
||||||
|
{ datasetId: dataset.data?.id as string },
|
||||||
|
{ enabled: !!dataset.data?.id, refetchInterval: fileUploadsRefetchInterval },
|
||||||
|
);
|
||||||
|
useEffect(() => {
|
||||||
|
if (fileUploads?.data?.some((fu) => fu.status !== "COMPLETE" && fu.status !== "ERROR")) {
|
||||||
|
setFileUploadsRefetchInterval(500);
|
||||||
|
} else {
|
||||||
|
setFileUploadsRefetchInterval(15000);
|
||||||
|
}
|
||||||
|
}, [fileUploads]);
|
||||||
|
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const hideFileUploadsMutation = api.datasets.hideFileUploads.useMutation();
|
||||||
|
const [hideAllFileUploads] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!fileUploads.data?.length) return;
|
||||||
|
await hideFileUploadsMutation.mutateAsync({
|
||||||
|
fileUploadIds: fileUploads.data.map((upload) => upload.id),
|
||||||
|
});
|
||||||
|
await utils.datasets.listFileUploads.invalidate();
|
||||||
|
}, [hideFileUploadsMutation, fileUploads.data, utils]);
|
||||||
|
|
||||||
|
if (!fileUploads.data?.length) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Portal>
|
||||||
|
<VStack
|
||||||
|
w={72}
|
||||||
|
borderRadius={8}
|
||||||
|
position="fixed"
|
||||||
|
bottom={8}
|
||||||
|
right={8}
|
||||||
|
overflow="hidden"
|
||||||
|
borderWidth={1}
|
||||||
|
boxShadow="0 0 40px 4px rgba(0, 0, 0, 0.1);"
|
||||||
|
minW={0}
|
||||||
|
bgColor="white"
|
||||||
|
>
|
||||||
|
<HStack p={4} w="full" bgColor="gray.200" justifyContent="space-between">
|
||||||
|
<Text fontWeight="bold">Uploads</Text>
|
||||||
|
<IconButton
|
||||||
|
aria-label="Close uploads"
|
||||||
|
as={BsX}
|
||||||
|
boxSize={6}
|
||||||
|
minW={0}
|
||||||
|
variant="ghost"
|
||||||
|
onClick={hideAllFileUploads}
|
||||||
|
cursor="pointer"
|
||||||
|
/>
|
||||||
|
</HStack>
|
||||||
|
{fileUploads?.data?.map((upload) => <FileUploadRow key={upload.id} fileUpload={upload} />)}
|
||||||
|
</VStack>
|
||||||
|
</Portal>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FileUploadsCard;
|
||||||
|
|
||||||
|
const FileUploadRow = ({ fileUpload }: { fileUpload: FileUpload }) => {
|
||||||
|
const { id, fileName, fileSize, progress, status, errorMessage } = fileUpload;
|
||||||
|
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const hideFileUploadsMutation = api.datasets.hideFileUploads.useMutation();
|
||||||
|
const [hideFileUpload, hidingInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
await hideFileUploadsMutation.mutateAsync({ fileUploadIds: [id] });
|
||||||
|
}, [id, hideFileUploadsMutation, utils]);
|
||||||
|
|
||||||
|
const [refreshDatasetEntries] = useHandledAsyncCallback(async () => {
|
||||||
|
await hideFileUploadsMutation.mutateAsync({ fileUploadIds: [id] });
|
||||||
|
await utils.datasets.listFileUploads.invalidate();
|
||||||
|
await utils.datasetEntries.list.invalidate();
|
||||||
|
}, [id, hideFileUploadsMutation, utils]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<VStack w="full" alignItems="flex-start" p={4} borderBottomWidth={1}>
|
||||||
|
<HStack w="full" justifyContent="space-between" alignItems="flex-start">
|
||||||
|
<VStack alignItems="flex-start" spacing={0}>
|
||||||
|
<Text fontWeight="bold">{fileName}</Text>
|
||||||
|
<Text fontSize="xs">({formatFileSize(fileSize, 2)})</Text>
|
||||||
|
</VStack>
|
||||||
|
|
||||||
|
<HStack spacing={0}>
|
||||||
|
{status === "COMPLETE" ? (
|
||||||
|
<Button variant="ghost" onClick={refreshDatasetEntries} color="orange.400" size="xs">
|
||||||
|
Refresh Table
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
<IconButton
|
||||||
|
aria-label="Hide file upload"
|
||||||
|
as={BsX}
|
||||||
|
boxSize={6}
|
||||||
|
minW={0}
|
||||||
|
variant="ghost"
|
||||||
|
isLoading={hidingInProgress}
|
||||||
|
onClick={hideFileUpload}
|
||||||
|
cursor="pointer"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</HStack>
|
||||||
|
</HStack>
|
||||||
|
|
||||||
|
{errorMessage ? (
|
||||||
|
<Text alignSelf="center" pt={2}>
|
||||||
|
{errorMessage}
|
||||||
|
</Text>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<Text alignSelf="center" fontSize="xs">
|
||||||
|
{getStatusText(status)}
|
||||||
|
</Text>
|
||||||
|
<Progress w="full" value={progress} borderRadius={2} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</VStack>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getStatusText = (status: FileUpload["status"]) => {
|
||||||
|
switch (status) {
|
||||||
|
case "PENDING":
|
||||||
|
return "Pending";
|
||||||
|
case "DOWNLOADING":
|
||||||
|
return "Downloading to Server";
|
||||||
|
case "PROCESSING":
|
||||||
|
return "Processing";
|
||||||
|
case "SAVING":
|
||||||
|
return "Saving";
|
||||||
|
case "COMPLETE":
|
||||||
|
return "Complete";
|
||||||
|
case "ERROR":
|
||||||
|
return "Error";
|
||||||
|
}
|
||||||
|
};
|
||||||
161
app/src/components/datasets/FineTuneButton.tsx
Normal file
161
app/src/components/datasets/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 { AiTwotoneThunderbolt } from "react-icons/ai";
|
||||||
|
import humanId from "human-id";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
|
||||||
|
import { useDataset, useDatasetEntries, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
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 datasetEntries = useDatasetEntries().data;
|
||||||
|
|
||||||
|
const numEntries = datasetEntries?.matchingEntryIds.length || 0;
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ActionButton
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
label="Fine Tune"
|
||||||
|
icon={AiTwotoneThunderbolt}
|
||||||
|
isDisabled={numEntries === 0}
|
||||||
|
requireBeta
|
||||||
|
/>
|
||||||
|
<FineTuneModal disclosure={disclosure} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default FineTuneButton;
|
||||||
|
|
||||||
|
const FineTuneModal = ({ disclosure }: { disclosure: UseDisclosureReturn }) => {
|
||||||
|
const dataset = useDataset().data;
|
||||||
|
const datasetEntries = useDatasetEntries().data;
|
||||||
|
|
||||||
|
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 (!modelSlug || !selectedBaseModel || !dataset) return;
|
||||||
|
await createFineTuneMutation.mutateAsync({
|
||||||
|
slug: modelSlug,
|
||||||
|
baseModel: selectedBaseModel,
|
||||||
|
datasetId: dataset.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
await utils.fineTunes.list.invalidate();
|
||||||
|
await router.push({ pathname: "/fine-tunes" });
|
||||||
|
disclosure.onClose();
|
||||||
|
}, [createFineTuneMutation, modelSlug, selectedBaseModel]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal size={{ base: "xl", md: "2xl" }} {...disclosure}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Icon as={AiTwotoneThunderbolt} />
|
||||||
|
<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 <b>{datasetEntries?.trainingCount}</b> and test on{" "}
|
||||||
|
<b>{datasetEntries?.testingCount}</b> entries in this dataset.
|
||||||
|
</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>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -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;
|
|
||||||
276
app/src/components/datasets/UploadDataButton.tsx
Normal file
276
app/src/components/datasets/UploadDataButton.tsx
Normal file
@@ -0,0 +1,276 @@
|
|||||||
|
import { useState, useEffect, useRef, useCallback } from "react";
|
||||||
|
import {
|
||||||
|
Modal,
|
||||||
|
ModalOverlay,
|
||||||
|
ModalContent,
|
||||||
|
ModalHeader,
|
||||||
|
ModalCloseButton,
|
||||||
|
ModalBody,
|
||||||
|
ModalFooter,
|
||||||
|
HStack,
|
||||||
|
VStack,
|
||||||
|
Icon,
|
||||||
|
Text,
|
||||||
|
Button,
|
||||||
|
Box,
|
||||||
|
useDisclosure,
|
||||||
|
type UseDisclosureReturn,
|
||||||
|
} from "@chakra-ui/react";
|
||||||
|
import pluralize from "pluralize";
|
||||||
|
import { AiOutlineCloudUpload, AiOutlineFile } from "react-icons/ai";
|
||||||
|
|
||||||
|
import { useDataset, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import { api } from "~/utils/api";
|
||||||
|
import ActionButton from "../ActionButton";
|
||||||
|
import { validateTrainingRows, type TrainingRow, parseJSONL } from "./validateTrainingRows";
|
||||||
|
import { uploadDatasetEntryFile } from "~/utils/azure/website";
|
||||||
|
import { formatFileSize } from "~/utils/utils";
|
||||||
|
|
||||||
|
const UploadDataButton = () => {
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ActionButton
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
label="Upload Data"
|
||||||
|
icon={AiOutlineCloudUpload}
|
||||||
|
iconBoxSize={4}
|
||||||
|
requireBeta
|
||||||
|
/>
|
||||||
|
<UploadDataModal disclosure={disclosure} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default UploadDataButton;
|
||||||
|
|
||||||
|
const UploadDataModal = ({ disclosure }: { disclosure: UseDisclosureReturn }) => {
|
||||||
|
const dataset = useDataset().data;
|
||||||
|
|
||||||
|
const [validationError, setValidationError] = useState<string | null>(null);
|
||||||
|
const [trainingRows, setTrainingRows] = useState<TrainingRow[] | null>(null);
|
||||||
|
const [file, setFile] = useState<File | null>(null);
|
||||||
|
|
||||||
|
const fileInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
|
const handleFileDrop = (e: React.DragEvent<HTMLDivElement>) => {
|
||||||
|
e.preventDefault();
|
||||||
|
const files = e.dataTransfer.files;
|
||||||
|
if (files.length > 0) {
|
||||||
|
processFile(files[0] as File);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleFileChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const files = e.target.files;
|
||||||
|
if (files && files.length > 0) {
|
||||||
|
processFile(files[0] as File);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const processFile = (file: File) => {
|
||||||
|
setFile(file);
|
||||||
|
|
||||||
|
// skip reading if file is larger than 10MB
|
||||||
|
if (file.size > 10000000) {
|
||||||
|
setTrainingRows(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = (e: ProgressEvent<FileReader>) => {
|
||||||
|
const content = e.target?.result as string;
|
||||||
|
// Process the content, e.g., set to state
|
||||||
|
let parsedJSONL;
|
||||||
|
try {
|
||||||
|
parsedJSONL = parseJSONL(content) as TrainingRow[];
|
||||||
|
const validationError = validateTrainingRows(parsedJSONL);
|
||||||
|
if (validationError) {
|
||||||
|
setValidationError(validationError);
|
||||||
|
setTrainingRows(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setTrainingRows(parsedJSONL);
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
} catch (e: any) {
|
||||||
|
setValidationError("Unable to parse JSONL file: " + (e.message as string));
|
||||||
|
setTrainingRows(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
reader.readAsText(file);
|
||||||
|
};
|
||||||
|
|
||||||
|
const resetState = useCallback(() => {
|
||||||
|
setValidationError(null);
|
||||||
|
setTrainingRows(null);
|
||||||
|
setFile(null);
|
||||||
|
}, [setValidationError, setTrainingRows, setFile]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (disclosure.isOpen) {
|
||||||
|
resetState();
|
||||||
|
}
|
||||||
|
}, [disclosure.isOpen, resetState]);
|
||||||
|
|
||||||
|
const triggerFileDownloadMutation = api.datasets.triggerFileDownload.useMutation();
|
||||||
|
|
||||||
|
const utils = api.useContext();
|
||||||
|
|
||||||
|
const [sendJSONL, sendingInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!dataset || !file) return;
|
||||||
|
|
||||||
|
const blobName = await uploadDatasetEntryFile(file);
|
||||||
|
|
||||||
|
await triggerFileDownloadMutation.mutateAsync({
|
||||||
|
datasetId: dataset.id,
|
||||||
|
blobName,
|
||||||
|
fileName: file.name,
|
||||||
|
fileSize: file.size,
|
||||||
|
});
|
||||||
|
|
||||||
|
await utils.datasets.listFileUploads.invalidate();
|
||||||
|
|
||||||
|
disclosure.onClose();
|
||||||
|
}, [dataset, trainingRows, triggerFileDownloadMutation, file, utils]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Modal size={{ base: "xl", md: "2xl" }} {...disclosure}>
|
||||||
|
<ModalOverlay />
|
||||||
|
<ModalContent w={1200}>
|
||||||
|
<ModalHeader>
|
||||||
|
<HStack>
|
||||||
|
<Text>Upload Training Logs</Text>
|
||||||
|
</HStack>
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalCloseButton />
|
||||||
|
<ModalBody maxW="unset" p={8}>
|
||||||
|
<Box w="full" aspectRatio={1.5}>
|
||||||
|
{validationError && (
|
||||||
|
<VStack w="full" h="full" justifyContent="center" spacing={8}>
|
||||||
|
<Icon as={AiOutlineFile} boxSize={24} color="gray.300" />
|
||||||
|
<VStack w="full">
|
||||||
|
<Text fontSize={32} color="gray.500" fontWeight="bold">
|
||||||
|
Error
|
||||||
|
</Text>
|
||||||
|
<Text color="gray.500">{validationError}</Text>
|
||||||
|
</VStack>
|
||||||
|
<Text
|
||||||
|
as="span"
|
||||||
|
textDecor="underline"
|
||||||
|
color="gray.500"
|
||||||
|
_hover={{ color: "orange.400" }}
|
||||||
|
cursor="pointer"
|
||||||
|
onClick={resetState}
|
||||||
|
>
|
||||||
|
Try again
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
)}
|
||||||
|
{!validationError && !file && (
|
||||||
|
<VStack
|
||||||
|
w="full"
|
||||||
|
h="full"
|
||||||
|
stroke="gray.300"
|
||||||
|
justifyContent="center"
|
||||||
|
borderRadius={8}
|
||||||
|
sx={{
|
||||||
|
"background-image": `url("data:image/svg+xml,%3csvg width='100%25' height='100%25' xmlns='http://www.w3.org/2000/svg'%3e%3crect x='2%25' y='2%25' width='96%25' height='96%25' fill='none' stroke='%23eee' stroke-width='4' stroke-dasharray='6%2c 14' stroke-dashoffset='0' stroke-linecap='square' rx='8' ry='8'/%3e%3c/svg%3e")`,
|
||||||
|
}}
|
||||||
|
onDragOver={(e) => e.preventDefault()}
|
||||||
|
onDrop={handleFileDrop}
|
||||||
|
>
|
||||||
|
<JsonFileIcon />
|
||||||
|
<Icon as={AiOutlineCloudUpload} boxSize={24} color="gray.300" />
|
||||||
|
|
||||||
|
<Text fontSize={32} color="gray.500" fontWeight="bold">
|
||||||
|
Drag & Drop
|
||||||
|
</Text>
|
||||||
|
<Text color="gray.500">
|
||||||
|
your .jsonl file here, or{" "}
|
||||||
|
<input
|
||||||
|
type="file"
|
||||||
|
ref={fileInputRef}
|
||||||
|
onChange={handleFileChange}
|
||||||
|
style={{ display: "none" }}
|
||||||
|
accept=".jsonl"
|
||||||
|
/>
|
||||||
|
<Text
|
||||||
|
as="span"
|
||||||
|
textDecor="underline"
|
||||||
|
_hover={{ color: "orange.400" }}
|
||||||
|
cursor="pointer"
|
||||||
|
onClick={() => fileInputRef.current?.click()}
|
||||||
|
>
|
||||||
|
browse
|
||||||
|
</Text>
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
)}
|
||||||
|
{!validationError && file && (
|
||||||
|
<VStack w="full" h="full" justifyContent="center" spacing={8}>
|
||||||
|
<JsonFileIcon />
|
||||||
|
<VStack w="full">
|
||||||
|
{trainingRows ? (
|
||||||
|
<>
|
||||||
|
<Text fontSize={32} color="gray.500" fontWeight="bold">
|
||||||
|
Success
|
||||||
|
</Text>
|
||||||
|
<Text color="gray.500">
|
||||||
|
We'll upload <b>{trainingRows.length}</b>{" "}
|
||||||
|
{pluralize("row", trainingRows.length)} into <b>{dataset?.name}</b>.{" "}
|
||||||
|
</Text>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<Text fontSize={32} color="gray.500" fontWeight="bold">
|
||||||
|
{file.name}
|
||||||
|
</Text>
|
||||||
|
<Text color="gray.500">{formatFileSize(file.size)}</Text>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</VStack>
|
||||||
|
<Text
|
||||||
|
as="span"
|
||||||
|
textDecor="underline"
|
||||||
|
color="gray.500"
|
||||||
|
_hover={{ color: "orange.400" }}
|
||||||
|
cursor="pointer"
|
||||||
|
onClick={resetState}
|
||||||
|
>
|
||||||
|
Change file
|
||||||
|
</Text>
|
||||||
|
</VStack>
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
</ModalBody>
|
||||||
|
<ModalFooter>
|
||||||
|
<HStack>
|
||||||
|
<Button colorScheme="gray" onClick={disclosure.onClose} minW={24}>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
colorScheme="orange"
|
||||||
|
onClick={sendJSONL}
|
||||||
|
isLoading={sendingInProgress}
|
||||||
|
minW={24}
|
||||||
|
isDisabled={!file || !!validationError}
|
||||||
|
>
|
||||||
|
Upload
|
||||||
|
</Button>
|
||||||
|
</HStack>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalContent>
|
||||||
|
</Modal>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const JsonFileIcon = () => (
|
||||||
|
<Box position="relative" display="flex" alignItems="center" justifyContent="center">
|
||||||
|
<Icon as={AiOutlineFile} boxSize={24} color="gray.300" />
|
||||||
|
<Text position="absolute" color="orange.400" fontWeight="bold" fontSize={12} pt={4}>
|
||||||
|
JSONL
|
||||||
|
</Text>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
71
app/src/components/datasets/validateTrainingRows.ts
Normal file
71
app/src/components/datasets/validateTrainingRows.ts
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
import { type CreateChatCompletionRequestMessage } from "openai/resources/chat";
|
||||||
|
|
||||||
|
export type TrainingRow = {
|
||||||
|
input: CreateChatCompletionRequestMessage[];
|
||||||
|
output?: CreateChatCompletionRequestMessage;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const parseJSONL = (jsonlString: string): unknown[] => {
|
||||||
|
const lines = jsonlString.trim().split("\n");
|
||||||
|
|
||||||
|
let lineNumber = 0;
|
||||||
|
const parsedLines = [];
|
||||||
|
|
||||||
|
try {
|
||||||
|
for (const line of lines) {
|
||||||
|
lineNumber++;
|
||||||
|
parsedLines.push(JSON.parse(line));
|
||||||
|
}
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
} catch (e: any) {
|
||||||
|
throw new Error(`Error parsing line ${lineNumber}: ${e.message as string}`);
|
||||||
|
}
|
||||||
|
return parsedLines;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const validateTrainingRows = (rows: unknown): string | null => {
|
||||||
|
if (!Array.isArray(rows)) return "training data is not an array";
|
||||||
|
for (let i = 0; i < rows.length; i++) {
|
||||||
|
const row = rows[i] as TrainingRow;
|
||||||
|
let errorMessage: string | null = null;
|
||||||
|
try {
|
||||||
|
errorMessage = validateTrainingRow(row);
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
} catch (error: any) {
|
||||||
|
errorMessage = error.message;
|
||||||
|
}
|
||||||
|
if (errorMessage) return `row ${i + 1}: ${errorMessage}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const validateTrainingRow = (row: TrainingRow): string | null => {
|
||||||
|
if (!row) return "empty row";
|
||||||
|
if (!row.input) return "missing input";
|
||||||
|
|
||||||
|
// Validate input
|
||||||
|
if (!Array.isArray(row.input)) return "input is not an array";
|
||||||
|
if ((row.input as unknown[]).some((x) => typeof x !== "object"))
|
||||||
|
return "input contains invalid item";
|
||||||
|
if (row.input.some((x) => !x)) return "input contains empty item";
|
||||||
|
if (row.input.some((x) => !x.content && !x.function_call))
|
||||||
|
return "input contains item with no content or function_call";
|
||||||
|
if (row.input.some((x) => x.function_call && !x.function_call.arguments))
|
||||||
|
return "input contains item with function_call but no arguments";
|
||||||
|
if (row.input.some((x) => x.function_call && !x.function_call.name))
|
||||||
|
return "input contains item with function_call but no name";
|
||||||
|
|
||||||
|
// Validate output
|
||||||
|
if (row.output) {
|
||||||
|
if (typeof row.output !== "object") return "output is not an object";
|
||||||
|
if (!row.output.content && !row.output.function_call)
|
||||||
|
return "output contains no content or function_call";
|
||||||
|
if (row.output.function_call && !row.output.function_call.arguments)
|
||||||
|
return "output contains function_call but no arguments";
|
||||||
|
if (row.output.function_call && !row.output.function_call.name)
|
||||||
|
return "output contains function_call but no name";
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
@@ -1,36 +1,43 @@
|
|||||||
|
import { useRef } from "react";
|
||||||
import {
|
import {
|
||||||
Button,
|
type UseDisclosureReturn,
|
||||||
AlertDialog,
|
AlertDialog,
|
||||||
|
AlertDialogOverlay,
|
||||||
|
AlertDialogContent,
|
||||||
|
AlertDialogHeader,
|
||||||
AlertDialogBody,
|
AlertDialogBody,
|
||||||
AlertDialogFooter,
|
AlertDialogFooter,
|
||||||
AlertDialogHeader,
|
Button,
|
||||||
AlertDialogContent,
|
|
||||||
AlertDialogOverlay,
|
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
|
|
||||||
import { useRouter } from "next/router";
|
|
||||||
import { useRef } from "react";
|
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import { useExperiment, useHandledAsyncCallback } from "~/utils/hooks";
|
|
||||||
|
|
||||||
export const DeleteDialog = ({ onClose }: { onClose: () => void }) => {
|
import { useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
const experiment = useExperiment();
|
|
||||||
const deleteMutation = api.experiments.delete.useMutation();
|
|
||||||
const utils = api.useContext();
|
|
||||||
const router = useRouter();
|
|
||||||
|
|
||||||
|
const DeleteExperimentDialog = ({
|
||||||
|
experimentId,
|
||||||
|
onDelete,
|
||||||
|
disclosure,
|
||||||
|
}: {
|
||||||
|
experimentId?: string;
|
||||||
|
onDelete?: () => void;
|
||||||
|
disclosure: UseDisclosureReturn;
|
||||||
|
}) => {
|
||||||
const cancelRef = useRef<HTMLButtonElement>(null);
|
const cancelRef = useRef<HTMLButtonElement>(null);
|
||||||
|
|
||||||
const [onDeleteConfirm] = useHandledAsyncCallback(async () => {
|
const mutation = api.experiments.delete.useMutation();
|
||||||
if (!experiment.data?.id) return;
|
const utils = api.useContext();
|
||||||
await deleteMutation.mutateAsync({ id: experiment.data.id });
|
|
||||||
|
const [onDeleteConfirm, deletionInProgress] = useHandledAsyncCallback(async () => {
|
||||||
|
if (!experimentId) return;
|
||||||
|
await mutation.mutateAsync({ id: experimentId });
|
||||||
await utils.experiments.list.invalidate();
|
await utils.experiments.list.invalidate();
|
||||||
await router.push({ pathname: "/experiments" });
|
onDelete?.();
|
||||||
onClose();
|
|
||||||
}, [deleteMutation, experiment.data?.id, router]);
|
disclosure.onClose();
|
||||||
|
}, [mutation, experimentId, disclosure.onClose]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AlertDialog isOpen leastDestructiveRef={cancelRef} onClose={onClose}>
|
<AlertDialog leastDestructiveRef={cancelRef} {...disclosure}>
|
||||||
<AlertDialogOverlay>
|
<AlertDialogOverlay>
|
||||||
<AlertDialogContent>
|
<AlertDialogContent>
|
||||||
<AlertDialogHeader fontSize="lg" fontWeight="bold">
|
<AlertDialogHeader fontSize="lg" fontWeight="bold">
|
||||||
@@ -43,10 +50,15 @@ export const DeleteDialog = ({ onClose }: { onClose: () => void }) => {
|
|||||||
</AlertDialogBody>
|
</AlertDialogBody>
|
||||||
|
|
||||||
<AlertDialogFooter>
|
<AlertDialogFooter>
|
||||||
<Button ref={cancelRef} onClick={onClose}>
|
<Button ref={cancelRef} onClick={disclosure.onClose}>
|
||||||
Cancel
|
Cancel
|
||||||
</Button>
|
</Button>
|
||||||
<Button colorScheme="red" onClick={onDeleteConfirm} ml={3}>
|
<Button
|
||||||
|
colorScheme="red"
|
||||||
|
isLoading={deletionInProgress}
|
||||||
|
onClick={onDeleteConfirm}
|
||||||
|
ml={3}
|
||||||
|
>
|
||||||
Delete
|
Delete
|
||||||
</Button>
|
</Button>
|
||||||
</AlertDialogFooter>
|
</AlertDialogFooter>
|
||||||
@@ -55,3 +67,5 @@ export const DeleteDialog = ({ onClose }: { onClose: () => void }) => {
|
|||||||
</AlertDialog>
|
</AlertDialog>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export default DeleteExperimentDialog;
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { type MouseEvent, useState } from "react";
|
||||||
import {
|
import {
|
||||||
HStack,
|
HStack,
|
||||||
Icon,
|
Icon,
|
||||||
@@ -7,44 +8,60 @@ import {
|
|||||||
Spinner,
|
Spinner,
|
||||||
AspectRatio,
|
AspectRatio,
|
||||||
SkeletonText,
|
SkeletonText,
|
||||||
|
Card,
|
||||||
|
useDisclosure,
|
||||||
|
Box,
|
||||||
|
Menu,
|
||||||
|
MenuButton,
|
||||||
|
MenuList,
|
||||||
|
MenuItem,
|
||||||
|
IconButton,
|
||||||
|
useToast,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import { RiFlaskLine } from "react-icons/ri";
|
import { RiFlaskLine } from "react-icons/ri";
|
||||||
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, BsThreeDotsVertical, BsLink45Deg, BsTrash } from "react-icons/bs";
|
||||||
import { api } from "~/utils/api";
|
|
||||||
|
import { formatTimePast } from "~/utils/dayjs";
|
||||||
|
import { type 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";
|
||||||
|
import DeleteExperimentDialog from "./DeleteExperimentDialog";
|
||||||
|
|
||||||
type ExperimentData = {
|
export const ExperimentCard = ({ exp }: { exp: RouterOutputs["experiments"]["list"][0] }) => {
|
||||||
testScenarioCount: number;
|
const [isMenuHovered, setIsMenuHovered] = useState(false);
|
||||||
promptVariantCount: number;
|
|
||||||
id: string;
|
|
||||||
label: string;
|
|
||||||
sortIndex: number;
|
|
||||||
createdAt: Date;
|
|
||||||
updatedAt: Date;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const ExperimentCard = ({ exp }: { exp: ExperimentData }) => {
|
|
||||||
return (
|
return (
|
||||||
<AspectRatio ratio={1.2} w="full">
|
<Card
|
||||||
|
w="full"
|
||||||
|
h="full"
|
||||||
|
cursor="pointer"
|
||||||
|
p={4}
|
||||||
|
bg="white"
|
||||||
|
borderRadius={4}
|
||||||
|
_hover={{ bg: isMenuHovered ? undefined : "gray.100" }}
|
||||||
|
transition="background 0.2s"
|
||||||
|
aspectRatio={1.2}
|
||||||
|
>
|
||||||
<VStack
|
<VStack
|
||||||
as={Link}
|
as={Link}
|
||||||
href={{ pathname: "/experiments/[id]", query: { id: exp.id } }}
|
w="full"
|
||||||
bg="gray.50"
|
h="full"
|
||||||
_hover={{ bg: "gray.100" }}
|
href={{ pathname: "/experiments/[experimentSlug]", query: { experimentSlug: exp.slug } }}
|
||||||
transition="background 0.2s"
|
|
||||||
cursor="pointer"
|
|
||||||
borderColor="gray.200"
|
|
||||||
borderWidth={1}
|
|
||||||
p={4}
|
|
||||||
justify="space-between"
|
justify="space-between"
|
||||||
>
|
>
|
||||||
<HStack w="full" color="gray.700" justify="center">
|
<HStack w="full" justify="space-between" spacing={0}>
|
||||||
<Icon as={RiFlaskLine} boxSize={4} />
|
<Box w={6} />
|
||||||
<Text fontWeight="bold">{exp.label}</Text>
|
<HStack color="gray.700" justify="center">
|
||||||
|
<Icon as={RiFlaskLine} boxSize={4} />
|
||||||
|
<Text fontWeight="bold">{exp.label}</Text>
|
||||||
|
</HStack>
|
||||||
|
<CardMenu
|
||||||
|
experimentId={exp.id}
|
||||||
|
experimentSlug={exp.slug}
|
||||||
|
setIsMenuHovered={setIsMenuHovered}
|
||||||
|
/>
|
||||||
</HStack>
|
</HStack>
|
||||||
<HStack h="full" spacing={4} flex={1} align="center">
|
<HStack h="full" spacing={4} flex={1} align="center">
|
||||||
<CountLabel label="Variants" count={exp.promptVariantCount} />
|
<CountLabel label="Variants" count={exp.promptVariantCount} />
|
||||||
@@ -57,7 +74,76 @@ export const ExperimentCard = ({ exp }: { exp: ExperimentData }) => {
|
|||||||
<Text flex={1}>Updated {formatTimePast(exp.updatedAt)}</Text>
|
<Text flex={1}>Updated {formatTimePast(exp.updatedAt)}</Text>
|
||||||
</HStack>
|
</HStack>
|
||||||
</VStack>
|
</VStack>
|
||||||
</AspectRatio>
|
</Card>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const CardMenu = ({
|
||||||
|
experimentId,
|
||||||
|
experimentSlug,
|
||||||
|
setIsMenuHovered,
|
||||||
|
}: {
|
||||||
|
experimentId: string;
|
||||||
|
experimentSlug: string;
|
||||||
|
setIsMenuHovered: (isHovered: boolean) => void;
|
||||||
|
}) => {
|
||||||
|
const deleteDisclosure = useDisclosure();
|
||||||
|
const menuDisclosure = useDisclosure();
|
||||||
|
const toast = useToast();
|
||||||
|
const [copyShareLink] = useHandledAsyncCallback(
|
||||||
|
async (e: MouseEvent<HTMLButtonElement>) => {
|
||||||
|
if (typeof window === "undefined") return;
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
const shareLink = `${window.location.origin}/experiments/${experimentSlug}`;
|
||||||
|
await navigator.clipboard.writeText(shareLink);
|
||||||
|
toast({
|
||||||
|
title: "Share link copied to clipboard",
|
||||||
|
status: "success",
|
||||||
|
duration: 2000,
|
||||||
|
isClosable: true,
|
||||||
|
});
|
||||||
|
menuDisclosure.onClose();
|
||||||
|
},
|
||||||
|
[toast, menuDisclosure.onClose, experimentSlug],
|
||||||
|
);
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Menu isLazy {...menuDisclosure}>
|
||||||
|
<MenuButton
|
||||||
|
as={IconButton}
|
||||||
|
aria-label="Options"
|
||||||
|
icon={<BsThreeDotsVertical />}
|
||||||
|
variant="ghost"
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
menuDisclosure.onOpen();
|
||||||
|
}}
|
||||||
|
onMouseEnter={() => setIsMenuHovered(true)}
|
||||||
|
onMouseLeave={() => setIsMenuHovered(false)}
|
||||||
|
boxSize={6}
|
||||||
|
minW={0}
|
||||||
|
/>
|
||||||
|
<MenuList>
|
||||||
|
<MenuItem icon={<Icon as={BsLink45Deg} boxSize={5} />} onClick={copyShareLink}>
|
||||||
|
Copy Link
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem
|
||||||
|
icon={<Icon as={BsTrash} boxSize={5} />}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
deleteDisclosure.onOpen();
|
||||||
|
}}
|
||||||
|
color="red.500"
|
||||||
|
>
|
||||||
|
Delete
|
||||||
|
</MenuItem>
|
||||||
|
</MenuList>
|
||||||
|
</Menu>
|
||||||
|
<DeleteExperimentDialog experimentId={experimentId} disclosure={deleteDisclosure} />
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -83,36 +169,34 @@ 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]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AspectRatio ratio={1.2} w="full">
|
<Card
|
||||||
<VStack
|
w="full"
|
||||||
align="center"
|
h="full"
|
||||||
justify="center"
|
cursor="pointer"
|
||||||
_hover={{ cursor: "pointer", bg: "gray.50" }}
|
p={4}
|
||||||
transition="background 0.2s"
|
bg="white"
|
||||||
cursor="pointer"
|
borderRadius={4}
|
||||||
borderColor="gray.200"
|
_hover={{ bg: "gray.100" }}
|
||||||
borderWidth={1}
|
transition="background 0.2s"
|
||||||
p={4}
|
aspectRatio={1.2}
|
||||||
onClick={createExperiment}
|
>
|
||||||
>
|
<VStack align="center" justify="center" w="full" h="full" p={4} onClick={createExperiment}>
|
||||||
<Icon as={isLoading ? Spinner : BsPlusSquare} boxSize={8} />
|
<Icon as={isLoading ? Spinner : BsPlusSquare} boxSize={8} />
|
||||||
<Text display={{ base: "none", md: "block" }} ml={2}>
|
<Text ml={2}>New Experiment</Text>
|
||||||
New Experiment
|
|
||||||
</Text>
|
|
||||||
</VStack>
|
</VStack>
|
||||||
</AspectRatio>
|
</Card>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const ExperimentCardSkeleton = () => (
|
export const ExperimentCardSkeleton = () => (
|
||||||
<AspectRatio ratio={1.2} w="full">
|
<AspectRatio ratio={1.2} w="full">
|
||||||
<VStack align="center" borderColor="gray.200" borderWidth={1} p={4} bg="gray.50">
|
<VStack align="center" borderColor="gray.200" borderWidth={1} p={4} bg="white">
|
||||||
<SkeletonText noOfLines={1} w="80%" />
|
<SkeletonText noOfLines={1} w="80%" />
|
||||||
<SkeletonText noOfLines={2} w="60%" />
|
<SkeletonText noOfLines={2} w="60%" />
|
||||||
<SkeletonText noOfLines={1} w="80%" />
|
<SkeletonText noOfLines={1} w="80%" />
|
||||||
|
|||||||
@@ -3,17 +3,14 @@ import { useOnForkButtonPressed } from "./useOnForkButtonPressed";
|
|||||||
import { useExperiment } from "~/utils/hooks";
|
import { useExperiment } from "~/utils/hooks";
|
||||||
import { BsGearFill } from "react-icons/bs";
|
import { BsGearFill } from "react-icons/bs";
|
||||||
import { TbGitFork } from "react-icons/tb";
|
import { TbGitFork } from "react-icons/tb";
|
||||||
import { useAppStore } from "~/state/store";
|
|
||||||
|
|
||||||
export const ExperimentHeaderButtons = () => {
|
export const ExperimentHeaderButtons = ({ openDrawer }: { openDrawer: () => void }) => {
|
||||||
const experiment = useExperiment();
|
const experiment = useExperiment();
|
||||||
|
|
||||||
const canModify = experiment.data?.access.canModify ?? false;
|
const canModify = experiment.data?.access.canModify ?? false;
|
||||||
|
|
||||||
const { onForkButtonPressed, isForking } = useOnForkButtonPressed();
|
const { onForkButtonPressed, isForking } = useOnForkButtonPressed();
|
||||||
|
|
||||||
const openDrawer = useAppStore((s) => s.openDrawer);
|
|
||||||
|
|
||||||
if (experiment.isLoading) return null;
|
if (experiment.isLoading) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -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(() => {
|
||||||
|
|||||||
@@ -0,0 +1,39 @@
|
|||||||
|
import { Button, Icon, useDisclosure, Text } from "@chakra-ui/react";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
|
||||||
|
import { useExperiment, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
|
import DeleteExperimentDialog from "../DeleteExperimentDialog";
|
||||||
|
|
||||||
|
export const DeleteButton = ({ closeDrawer }: { closeDrawer: () => void }) => {
|
||||||
|
const experiment = useExperiment();
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const disclosure = useDisclosure();
|
||||||
|
|
||||||
|
const [onDelete] = useHandledAsyncCallback(async () => {
|
||||||
|
await router.push({ pathname: "/experiments" });
|
||||||
|
closeDrawer();
|
||||||
|
}, [router, closeDrawer]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
colorScheme="red"
|
||||||
|
fontWeight="normal"
|
||||||
|
onClick={disclosure.onOpen}
|
||||||
|
>
|
||||||
|
<Icon as={BsTrash} boxSize={4} />
|
||||||
|
<Text ml={2}>Delete Experiment</Text>
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<DeleteExperimentDialog
|
||||||
|
experimentId={experiment.data?.id}
|
||||||
|
onDelete={onDelete}
|
||||||
|
disclosure={disclosure}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -19,7 +19,7 @@ import { useCallback, useState } from "react";
|
|||||||
import { BsPencil, BsX } from "react-icons/bs";
|
import { BsPencil, BsX } from "react-icons/bs";
|
||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import { useExperiment, useHandledAsyncCallback } from "~/utils/hooks";
|
import { useExperiment, useHandledAsyncCallback } from "~/utils/hooks";
|
||||||
import AutoResizeTextArea from "../AutoResizeTextArea";
|
import AutoResizeTextArea from "~/components/AutoResizeTextArea";
|
||||||
|
|
||||||
type EvalValues = Pick<Evaluation, "label" | "value" | "evalType">;
|
type EvalValues = Pick<Evaluation, "label" | "value" | "evalType">;
|
||||||
|
|
||||||
@@ -5,7 +5,7 @@ import { BsPencil, BsX } from "react-icons/bs";
|
|||||||
import { api } from "~/utils/api";
|
import { api } from "~/utils/api";
|
||||||
import { useExperiment, useHandledAsyncCallback, useScenarioVars } from "~/utils/hooks";
|
import { useExperiment, useHandledAsyncCallback, useScenarioVars } from "~/utils/hooks";
|
||||||
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
import { maybeReportError } from "~/utils/errorHandling/maybeReportError";
|
||||||
import { FloatingLabelInput } from "./FloatingLabelInput";
|
import { FloatingLabelInput } from "~/components/OutputsTable/FloatingLabelInput";
|
||||||
|
|
||||||
export const ScenarioVar = ({
|
export const ScenarioVar = ({
|
||||||
variable,
|
variable,
|
||||||
@@ -7,18 +7,19 @@ import {
|
|||||||
DrawerOverlay,
|
DrawerOverlay,
|
||||||
Heading,
|
Heading,
|
||||||
VStack,
|
VStack,
|
||||||
|
type UseDisclosureReturn,
|
||||||
} from "@chakra-ui/react";
|
} from "@chakra-ui/react";
|
||||||
import EditScenarioVars from "../OutputsTable/EditScenarioVars";
|
import EditScenarioVars from "./EditScenarioVars";
|
||||||
import EditEvaluations from "../OutputsTable/EditEvaluations";
|
import EditEvaluations from "./EditEvaluations";
|
||||||
import { useAppStore } from "~/state/store";
|
|
||||||
import { DeleteButton } from "./DeleteButton";
|
import { DeleteButton } from "./DeleteButton";
|
||||||
|
|
||||||
export default function ExperimentSettingsDrawer() {
|
export default function ExperimentSettingsDrawer({
|
||||||
const isOpen = useAppStore((state) => state.drawerOpen);
|
disclosure,
|
||||||
const closeDrawer = useAppStore((state) => state.closeDrawer);
|
}: {
|
||||||
|
disclosure: UseDisclosureReturn;
|
||||||
|
}) {
|
||||||
return (
|
return (
|
||||||
<Drawer isOpen={isOpen} placement="right" onClose={closeDrawer} size="md">
|
<Drawer placement="right" size="md" {...disclosure}>
|
||||||
<DrawerOverlay />
|
<DrawerOverlay />
|
||||||
<DrawerContent>
|
<DrawerContent>
|
||||||
<DrawerCloseButton />
|
<DrawerCloseButton />
|
||||||
@@ -31,7 +32,7 @@ export default function ExperimentSettingsDrawer() {
|
|||||||
<EditScenarioVars />
|
<EditScenarioVars />
|
||||||
<EditEvaluations />
|
<EditEvaluations />
|
||||||
</VStack>
|
</VStack>
|
||||||
<DeleteButton />
|
<DeleteButton closeDrawer={disclosure.onClose} />
|
||||||
</VStack>
|
</VStack>
|
||||||
</DrawerBody>
|
</DrawerBody>
|
||||||
</DrawerContent>
|
</DrawerContent>
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user