Compare commits
61 Commits
feature/ll
...
dev/ui
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7faf20e0c8 | ||
|
|
ad17b27c3d | ||
|
|
112d7ef8f9 | ||
|
|
31db041084 | ||
|
|
882efb5284 | ||
|
|
604cc8e1dc | ||
|
|
a778755492 | ||
|
|
3bbfebb5e3 | ||
|
|
6883fff352 | ||
|
|
179bab605e | ||
|
|
202402a123 | ||
|
|
c6969fdd90 | ||
|
|
b9d556ff1b | ||
|
|
194a664f20 | ||
|
|
09c3f0ccc6 | ||
|
|
5e14b9b0e1 | ||
|
|
7165953b50 | ||
|
|
e362feaa82 | ||
|
|
66054dccb0 | ||
|
|
7efd7183d8 | ||
|
|
5b27e797b3 | ||
|
|
269a87da74 | ||
|
|
88fbf5e400 | ||
|
|
f45316904b | ||
|
|
d528a8df4c | ||
|
|
d0de78eaf0 | ||
|
|
2fc79dcf37 | ||
|
|
174c9a740f | ||
|
|
445908f8ae | ||
|
|
18803469de | ||
|
|
49502e1534 | ||
|
|
f7f6943d31 | ||
|
|
df21270a7e | ||
|
|
0a6d06b7a6 | ||
|
|
84ac5b62cb | ||
|
|
5174ddacfc | ||
|
|
5d7681cb62 | ||
|
|
82d6d420b3 | ||
|
|
4d81734ceb | ||
|
|
fb65bb8a95 | ||
|
|
68e06b0d53 | ||
|
|
3235ff4ed0 | ||
|
|
ea6438311f | ||
|
|
37ae7a99d8 | ||
|
|
20e8a8f197 | ||
|
|
8f8eeacc3d | ||
|
|
50a679278d | ||
|
|
d16fc16de6 | ||
|
|
3260d1b6b8 | ||
|
|
c5ddf2ddae | ||
|
|
9b5fa3c7e1 | ||
|
|
426262e62a | ||
|
|
3b39677d46 | ||
|
|
3ef82991fb | ||
|
|
f9ba3805a6 | ||
|
|
936f697110 | ||
|
|
b07bbd7d8c | ||
|
|
42f7d2da60 | ||
|
|
802bde2d76 | ||
|
|
b2db0307eb | ||
|
|
391cbd8334 |
@@ -9,3 +9,6 @@ screenshoots
|
|||||||
.env
|
.env
|
||||||
.blog
|
.blog
|
||||||
docs
|
docs
|
||||||
|
.log
|
||||||
|
blog
|
||||||
|
config.json
|
||||||
|
|||||||
45
CLAUDE.md
@@ -1,12 +1,43 @@
|
|||||||
# CLAUDE.md
|
# CLAUDE.md
|
||||||
|
|
||||||
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.You need use English to write text.
|
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
||||||
|
|
||||||
## Key Development Commands
|
## Commands
|
||||||
- Build: `npm run build`
|
|
||||||
- Start: `npm start`
|
- **Build the project**:
|
||||||
|
```bash
|
||||||
|
npm run build
|
||||||
|
```
|
||||||
|
- **Start the router server**:
|
||||||
|
```bash
|
||||||
|
ccr start
|
||||||
|
```
|
||||||
|
- **Stop the router server**:
|
||||||
|
```bash
|
||||||
|
ccr stop
|
||||||
|
```
|
||||||
|
- **Check the server status**:
|
||||||
|
```bash
|
||||||
|
ccr status
|
||||||
|
```
|
||||||
|
- **Run Claude Code through the router**:
|
||||||
|
```bash
|
||||||
|
ccr code "<your prompt>"
|
||||||
|
```
|
||||||
|
- **Release a new version**:
|
||||||
|
```bash
|
||||||
|
npm run release
|
||||||
|
```
|
||||||
|
|
||||||
## Architecture
|
## Architecture
|
||||||
- Uses `express` for routing (see `src/server.ts`)
|
|
||||||
- Bundles with `esbuild` for CLI distribution
|
This project is a TypeScript-based router for Claude Code requests. It allows routing requests to different large language models (LLMs) from various providers based on custom rules.
|
||||||
- Plugins are loaded from `$HOME/.claude-code-router/plugins`
|
|
||||||
|
- **Entry Point**: The main command-line interface logic is in `src/cli.ts`. It handles parsing commands like `start`, `stop`, and `code`.
|
||||||
|
- **Server**: The `ccr start` command launches a server that listens for requests from Claude Code. The server logic is initiated from `src/index.ts`.
|
||||||
|
- **Configuration**: The router is configured via a JSON file located at `~/.claude-code-router/config.json`. This file defines API providers, routing rules, and custom transformers. An example can be found in `config.example.json`.
|
||||||
|
- **Routing**: The core routing logic determines which LLM provider and model to use for a given request. It supports default routes for different scenarios (`default`, `background`, `think`, `longContext`, `webSearch`) and can be extended with a custom JavaScript router file. The router logic is likely in `src/utils/router.ts`.
|
||||||
|
- **Providers and Transformers**: The application supports multiple LLM providers. Transformers adapt the request and response formats for different provider APIs.
|
||||||
|
- **Claude Code Integration**: When a user runs `ccr code`, the command is forwarded to the running router service. The service then processes the request, applies routing rules, and sends it to the configured LLM. If the service isn't running, `ccr code` will attempt to start it automatically.
|
||||||
|
- **Dependencies**: The project is built with `esbuild`. It has a key local dependency `@musistudio/llms`, which probably contains the core logic for interacting with different LLM APIs.
|
||||||
|
- `@musistudio/llms` is implemented based on `fastify` and exposes `fastify`'s hook and middleware interfaces, allowing direct use of `server.addHook`.
|
||||||
378
README.md
@@ -1,38 +1,60 @@
|
|||||||
# Claude Code Router
|
# Claude Code Router
|
||||||
|
|
||||||
> This is a tool for routing Claude Code requests to different models, and you can customize any request.
|
[中文版](README_zh.md)
|
||||||
|
|
||||||

|
> A powerful tool to route Claude Code requests to different models and customize any request.
|
||||||
|
|
||||||
## Usage
|

|
||||||
|
|
||||||
1. Install Claude Code
|
## ✨ Features
|
||||||
|
|
||||||
|
- **Model Routing**: Route requests to different models based on your needs (e.g., background tasks, thinking, long context).
|
||||||
|
- **Multi-Provider Support**: Supports various model providers like OpenRouter, DeepSeek, Ollama, Gemini, Volcengine, and SiliconFlow.
|
||||||
|
- **Request/Response Transformation**: Customize requests and responses for different providers using transformers.
|
||||||
|
- **Dynamic Model Switching**: Switch models on-the-fly within Claude Code using the `/model` command.
|
||||||
|
- **GitHub Actions Integration**: Trigger Claude Code tasks in your GitHub workflows.
|
||||||
|
- **Plugin System**: Extend functionality with custom transformers.
|
||||||
|
|
||||||
|
## 🚀 Getting Started
|
||||||
|
|
||||||
|
### 1. Installation
|
||||||
|
|
||||||
|
First, ensure you have [Claude Code](https://docs.anthropic.com/en/docs/claude-code/quickstart) installed:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
npm install -g @anthropic-ai/claude-code
|
npm install -g @anthropic-ai/claude-code
|
||||||
```
|
```
|
||||||
|
|
||||||
2. Install Claude Code Router
|
Then, install Claude Code Router:
|
||||||
|
|
||||||
```shell
|
```shell
|
||||||
npm install -g @musistudio/claude-code-router
|
npm install -g @musistudio/claude-code-router
|
||||||
```
|
```
|
||||||
|
|
||||||
3. Start Claude Code by claude-code-router
|
### 2. Configuration
|
||||||
|
|
||||||
```shell
|
Create and configure your `~/.claude-code-router/config.json` file. For more details, you can refer to `config.example.json`.
|
||||||
ccr code
|
|
||||||
```
|
|
||||||
|
|
||||||
4. Configure routing[optional]
|
The `config.json` file has several key sections:
|
||||||
Set up your `~/.claude-code-router/config.json` file like this:
|
|
||||||
|
- **`PROXY_URL`** (optional): You can set a proxy for API requests, for example: `"PROXY_URL": "http://127.0.0.1:7890"`.
|
||||||
|
- **`LOG`** (optional): You can enable logging by setting it to `true`. The log file will be located at `$HOME/.claude-code-router.log`.
|
||||||
|
- **`APIKEY`** (optional): You can set a secret key to authenticate requests. When set, clients must provide this key in the `Authorization` header (e.g., `Bearer your-secret-key`) or the `x-api-key` header. Example: `"APIKEY": "your-secret-key"`.
|
||||||
|
- **`HOST`** (optional): You can set the host address for the server. If `APIKEY` is not set, the host will be forced to `127.0.0.1` for security reasons to prevent unauthorized access. Example: `"HOST": "0.0.0.0"`.
|
||||||
|
|
||||||
|
- **`Providers`**: Used to configure different model providers.
|
||||||
|
- **`Router`**: Used to set up routing rules. `default` specifies the default model, which will be used for all requests if no other route is configured.
|
||||||
|
|
||||||
|
Here is a comprehensive example:
|
||||||
|
|
||||||
```json
|
```json
|
||||||
{
|
{
|
||||||
|
"APIKEY": "your-secret-key",
|
||||||
|
"PROXY_URL": "http://127.0.0.1:7890",
|
||||||
|
"LOG": true,
|
||||||
"Providers": [
|
"Providers": [
|
||||||
{
|
{
|
||||||
"name": "openrouter",
|
"name": "openrouter",
|
||||||
// IMPORTANT: api_base_url must be a complete (full) URL.
|
|
||||||
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
||||||
"api_key": "sk-xxx",
|
"api_key": "sk-xxx",
|
||||||
"models": [
|
"models": [
|
||||||
@@ -40,32 +62,31 @@ ccr code
|
|||||||
"anthropic/claude-sonnet-4",
|
"anthropic/claude-sonnet-4",
|
||||||
"anthropic/claude-3.5-sonnet",
|
"anthropic/claude-3.5-sonnet",
|
||||||
"anthropic/claude-3.7-sonnet:thinking"
|
"anthropic/claude-3.7-sonnet:thinking"
|
||||||
]
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["openrouter"]
|
||||||
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "deepseek",
|
"name": "deepseek",
|
||||||
// IMPORTANT: api_base_url must be a complete (full) URL.
|
|
||||||
"api_base_url": "https://api.deepseek.com/chat/completions",
|
"api_base_url": "https://api.deepseek.com/chat/completions",
|
||||||
"api_key": "sk-xxx",
|
"api_key": "sk-xxx",
|
||||||
"models": ["deepseek-chat", "deepseek-reasoner"],
|
"models": ["deepseek-chat", "deepseek-reasoner"],
|
||||||
"transformer": {
|
"transformer": {
|
||||||
"use": ["deepseek"],
|
"use": ["deepseek"],
|
||||||
"deepseek-chat": {
|
"deepseek-chat": {
|
||||||
// Enhance tool usage for the deepseek-chat model using the ToolUse transformer.
|
|
||||||
"use": ["tooluse"]
|
"use": ["tooluse"]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "ollama",
|
"name": "ollama",
|
||||||
// IMPORTANT: api_base_url must be a complete (full) URL.
|
|
||||||
"api_base_url": "http://localhost:11434/v1/chat/completions",
|
"api_base_url": "http://localhost:11434/v1/chat/completions",
|
||||||
"api_key": "ollama",
|
"api_key": "ollama",
|
||||||
"models": ["qwen2.5-coder:latest"]
|
"models": ["qwen2.5-coder:latest"]
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "gemini",
|
"name": "gemini",
|
||||||
// IMPORTANT: api_base_url must be a complete (full) URL.
|
|
||||||
"api_base_url": "https://generativelanguage.googleapis.com/v1beta/models/",
|
"api_base_url": "https://generativelanguage.googleapis.com/v1beta/models/",
|
||||||
"api_key": "sk-xxx",
|
"api_key": "sk-xxx",
|
||||||
"models": ["gemini-2.5-flash", "gemini-2.5-pro"],
|
"models": ["gemini-2.5-flash", "gemini-2.5-pro"],
|
||||||
@@ -75,50 +96,223 @@ ccr code
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "volcengine",
|
"name": "volcengine",
|
||||||
// IMPORTANT: api_base_url must be a complete (full) URL.
|
|
||||||
"api_base_url": "https://ark.cn-beijing.volces.com/api/v3/chat/completions",
|
"api_base_url": "https://ark.cn-beijing.volces.com/api/v3/chat/completions",
|
||||||
"api_key": "sk-xxx",
|
"api_key": "sk-xxx",
|
||||||
"models": ["deepseek-v3-250324", "deepseek-r1-250528"],
|
"models": ["deepseek-v3-250324", "deepseek-r1-250528"],
|
||||||
"transformer": {
|
"transformer": {
|
||||||
"use": ["deepseek"]
|
"use": ["deepseek"]
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "modelscope",
|
||||||
|
"api_base_url": "https://api-inference.modelscope.cn/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": ["Qwen/Qwen3-Coder-480B-A35B-Instruct", "Qwen/Qwen3-235B-A22B-Thinking-2507"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 65536
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"enhancetool"
|
||||||
|
],
|
||||||
|
"Qwen/Qwen3-235B-A22B-Thinking-2507": {
|
||||||
|
"use": ["reasoning"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "dashscope",
|
||||||
|
"api_base_url": "https://dashscope.aliyuncs.com/compatible-mode/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": ["qwen3-coder-plus"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 65536
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"enhancetool"
|
||||||
|
]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"Router": {
|
"Router": {
|
||||||
"default": "deepseek,deepseek-chat", // IMPORTANT OPENAI_MODEL has been deprecated
|
"default": "deepseek,deepseek-chat",
|
||||||
"background": "ollama,qwen2.5-coder:latest",
|
"background": "ollama,qwen2.5-coder:latest",
|
||||||
"think": "deepseek,deepseek-reasoner",
|
"think": "deepseek,deepseek-reasoner",
|
||||||
"longContext": "openrouter,google/gemini-2.5-pro-preview"
|
"longContext": "openrouter,google/gemini-2.5-pro-preview",
|
||||||
|
"longContextThreshold": 60000,
|
||||||
|
"webSearch": "gemini,gemini-2.5-flash"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
- `background`
|
### 3. Running Claude Code with the Router
|
||||||
This model will be used to handle some background tasks([background-token-usage](https://docs.anthropic.com/en/docs/claude-code/costs#background-token-usage)). Based on my tests, it doesn’t require high intelligence. I’m using the qwen-coder-2.5:7b model running locally on my MacBook Pro M1 (32GB) via Ollama.
|
|
||||||
If your computer can’t run Ollama, you can also use some free models, such as qwen-coder-2.5:3b.
|
|
||||||
|
|
||||||
- `think`
|
Start Claude Code using the router:
|
||||||
This model will be used when enabling Claude Code to perform reasoning. However, reasoning budget control has not yet been implemented (since the DeepSeek-R1 model does not support it), so there is currently no difference between using UltraThink and Think modes.
|
|
||||||
It is worth noting that Plan Mode also use this model to achieve better planning results.
|
|
||||||
Note: The reasoning process via the official DeepSeek API may be very slow, so you may need to wait for an extended period of time.
|
|
||||||
|
|
||||||
- `longContext`
|
```shell
|
||||||
This model will be used when the context length exceeds 32K (this value may be modified in the future). You can route the request to a model that performs well with long contexts (I’ve chosen google/gemini-2.5-pro-preview). This scenario has not been thoroughly tested yet, so if you encounter any issues, please submit an issue.
|
ccr code
|
||||||
|
```
|
||||||
|
|
||||||
- model command
|
> **Note**: After modifying the configuration file, you need to restart the service for the changes to take effect:
|
||||||
You can also switch models within Claude Code by using the `/model` command. The format is: `provider,model`, like this:
|
>
|
||||||
`/model openrouter,anthropic/claude-3.5-sonnet`
|
> ```shell
|
||||||
This will use the anthropic/claude-3.5-sonnet model provided by OpenRouter to handle all subsequent tasks.
|
> ccr restart
|
||||||
|
> ```
|
||||||
|
|
||||||
## Features
|
#### Providers
|
||||||
|
|
||||||
- [x] Support change models
|
The `Providers` array is where you define the different model providers you want to use. Each provider object requires:
|
||||||
- [x] Github Actions
|
|
||||||
- [ ] More detailed logs
|
|
||||||
|
|
||||||
## Github Actions
|
- `name`: A unique name for the provider.
|
||||||
|
- `api_base_url`: The full API endpoint for chat completions.
|
||||||
|
- `api_key`: Your API key for the provider.
|
||||||
|
- `models`: A list of model names available from this provider.
|
||||||
|
- `transformer` (optional): Specifies transformers to process requests and responses.
|
||||||
|
|
||||||
You just need to install `Claude Code Actions` in your repository according to the [official documentation](https://docs.anthropic.com/en/docs/claude-code/github-actions). For `ANTHROPIC_API_KEY`, you can use any string. Then, modify your `.github/workflows/claude.yaml` file to include claude-code-router, like this:
|
#### Transformers
|
||||||
|
|
||||||
|
Transformers allow you to modify the request and response payloads to ensure compatibility with different provider APIs.
|
||||||
|
|
||||||
|
- **Global Transformer**: Apply a transformer to all models from a provider. In this example, the `openrouter` transformer is applied to all models under the `openrouter` provider.
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "openrouter",
|
||||||
|
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": [
|
||||||
|
"google/gemini-2.5-pro-preview",
|
||||||
|
"anthropic/claude-sonnet-4",
|
||||||
|
"anthropic/claude-3.5-sonnet"
|
||||||
|
],
|
||||||
|
"transformer": { "use": ["openrouter"] }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
- **Model-Specific Transformer**: Apply a transformer to a specific model. In this example, the `deepseek` transformer is applied to all models, and an additional `tooluse` transformer is applied only to the `deepseek-chat` model.
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "deepseek",
|
||||||
|
"api_base_url": "https://api.deepseek.com/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["deepseek-chat", "deepseek-reasoner"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["deepseek"],
|
||||||
|
"deepseek-chat": { "use": ["tooluse"] }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
- **Passing Options to a Transformer**: Some transformers, like `maxtoken`, accept options. To pass options, use a nested array where the first element is the transformer name and the second is an options object.
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "siliconflow",
|
||||||
|
"api_base_url": "https://api.siliconflow.cn/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["moonshotai/Kimi-K2-Instruct"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 16384
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**Available Built-in Transformers:**
|
||||||
|
|
||||||
|
- `deepseek`: Adapts requests/responses for DeepSeek API.
|
||||||
|
- `gemini`: Adapts requests/responses for Gemini API.
|
||||||
|
- `openrouter`: Adapts requests/responses for OpenRouter API.
|
||||||
|
- `groq`: Adapts requests/responses for groq API.
|
||||||
|
- `maxtoken`: Sets a specific `max_tokens` value.
|
||||||
|
- `tooluse`: Optimizes tool usage for certain models via `tool_choice`.
|
||||||
|
- `gemini-cli` (experimental): Unofficial support for Gemini via Gemini CLI [gemini-cli.js](https://gist.github.com/musistudio/1c13a65f35916a7ab690649d3df8d1cd).
|
||||||
|
|
||||||
|
**Custom Transformers:**
|
||||||
|
|
||||||
|
You can also create your own transformers and load them via the `transformers` field in `config.json`.
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"transformers": [
|
||||||
|
{
|
||||||
|
"path": "$HOME/.claude-code-router/plugins/gemini-cli.js",
|
||||||
|
"options": {
|
||||||
|
"project": "xxx"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Router
|
||||||
|
|
||||||
|
The `Router` object defines which model to use for different scenarios:
|
||||||
|
|
||||||
|
- `default`: The default model for general tasks.
|
||||||
|
- `background`: A model for background tasks. This can be a smaller, local model to save costs.
|
||||||
|
- `think`: A model for reasoning-heavy tasks, like Plan Mode.
|
||||||
|
- `longContext`: A model for handling long contexts (e.g., > 60K tokens).
|
||||||
|
- `longContextThreshold` (optional): The token count threshold for triggering the long context model. Defaults to 60000 if not specified.
|
||||||
|
- `webSearch`: Used for handling web search tasks and this requires the model itself to support the feature. If you're using openrouter, you need to add the `:online` suffix after the model name.
|
||||||
|
|
||||||
|
You can also switch models dynamically in Claude Code with the `/model` command:
|
||||||
|
`/model provider_name,model_name`
|
||||||
|
Example: `/model openrouter,anthropic/claude-3.5-sonnet`
|
||||||
|
|
||||||
|
#### Custom Router
|
||||||
|
|
||||||
|
For more advanced routing logic, you can specify a custom router script via the `CUSTOM_ROUTER_PATH` in your `config.json`. This allows you to implement complex routing rules beyond the default scenarios.
|
||||||
|
|
||||||
|
In your `config.json`:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"CUSTOM_ROUTER_PATH": "$HOME/.claude-code-router/custom-router.js"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
The custom router file must be a JavaScript module that exports an `async` function. This function receives the request object and the config object as arguments and should return the provider and model name as a string (e.g., `"provider_name,model_name"`), or `null` to fall back to the default router.
|
||||||
|
|
||||||
|
Here is an example of a `custom-router.js` based on `custom-router.example.js`:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// $HOME/.claude-code-router/custom-router.js
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A custom router function to determine which model to use based on the request.
|
||||||
|
*
|
||||||
|
* @param {object} req - The request object from Claude Code, containing the request body.
|
||||||
|
* @param {object} config - The application's config object.
|
||||||
|
* @returns {Promise<string|null>} - A promise that resolves to the "provider,model_name" string, or null to use the default router.
|
||||||
|
*/
|
||||||
|
module.exports = async function router(req, config) {
|
||||||
|
const userMessage = req.body.messages.find((m) => m.role === "user")?.content;
|
||||||
|
|
||||||
|
if (userMessage && userMessage.includes("explain this code")) {
|
||||||
|
// Use a powerful model for code explanation
|
||||||
|
return "openrouter,anthropic/claude-3.5-sonnet";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback to the default router configuration
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
## 🤖 GitHub Actions
|
||||||
|
|
||||||
|
Integrate Claude Code Router into your CI/CD pipeline. After setting up [Claude Code Actions](https://docs.anthropic.com/en/docs/claude-code/github-actions), modify your `.github/workflows/claude.yaml` to use the router:
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
name: Claude Code
|
name: Claude Code
|
||||||
@@ -126,20 +320,13 @@ name: Claude Code
|
|||||||
on:
|
on:
|
||||||
issue_comment:
|
issue_comment:
|
||||||
types: [created]
|
types: [created]
|
||||||
pull_request_review_comment:
|
# ... other triggers
|
||||||
types: [created]
|
|
||||||
issues:
|
|
||||||
types: [opened, assigned]
|
|
||||||
pull_request_review:
|
|
||||||
types: [submitted]
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
claude:
|
claude:
|
||||||
if: |
|
if: |
|
||||||
(github.event_name == 'issue_comment' && contains(github.event.comment.body, '@claude')) ||
|
(github.event_name == 'issue_comment' && contains(github.event.comment.body, '@claude')) ||
|
||||||
(github.event_name == 'pull_request_review_comment' && contains(github.event.comment.body, '@claude')) ||
|
# ... other conditions
|
||||||
(github.event_name == 'pull_request_review' && contains(github.event.review.body, '@claude')) ||
|
|
||||||
(github.event_name == 'issues' && (contains(github.event.issue.body, '@claude') || contains(github.event.issue.title, '@claude')))
|
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
@@ -177,64 +364,69 @@ jobs:
|
|||||||
env:
|
env:
|
||||||
ANTHROPIC_BASE_URL: http://localhost:3456
|
ANTHROPIC_BASE_URL: http://localhost:3456
|
||||||
with:
|
with:
|
||||||
anthropic_api_key: "test"
|
anthropic_api_key: "any-string-is-ok"
|
||||||
```
|
```
|
||||||
|
|
||||||
You can modify the contents of `$HOME/.claude-code-router/config.json` as needed.
|
This setup allows for interesting automations, like running tasks during off-peak hours to reduce API costs.
|
||||||
GitHub Actions support allows you to trigger Claude Code at specific times, which opens up some interesting possibilities.
|
|
||||||
|
|
||||||
For example, between 00:30 and 08:30 Beijing Time, using the official DeepSeek API:
|
## 📝 Further Reading
|
||||||
|
|
||||||
- The cost of the `deepseek-v3` model is only 50% of the normal time.
|
- [Project Motivation and How It Works](blog/en/project-motivation-and-how-it-works.md)
|
||||||
|
- [Maybe We Can Do More with the Router](blog/en/maybe-we-can-do-more-with-the-route.md)
|
||||||
|
|
||||||
- The `deepseek-r1` model is just 25% of the normal time.
|
## ❤️ Support & Sponsoring
|
||||||
|
|
||||||
So maybe in the future, I’ll describe detailed tasks for Claude Code ahead of time and let it run during these discounted hours to reduce costs?
|
If you find this project helpful, please consider sponsoring its development. Your support is greatly appreciated!
|
||||||
|
|
||||||
## Some tips:
|
|
||||||
|
|
||||||
Now you can use deepseek-v3 models directly without using any plugins.
|
|
||||||
|
|
||||||
If you’re using the DeepSeek API provided by the official website, you might encounter an “exceeding context” error after several rounds of conversation (since the official API only supports a 64K context window). In this case, you’ll need to discard the previous context and start fresh. Alternatively, you can use ByteDance’s DeepSeek API, which offers a 128K context window and supports KV cache.
|
|
||||||
|
|
||||||

|
|
||||||
|
|
||||||
Note: claude code consumes a huge amount of tokens, but thanks to DeepSeek’s low cost, you can use claude code at a fraction of Claude’s price, and you don’t need to subscribe to the Claude Max plan.
|
|
||||||
|
|
||||||
Some interesting points: Based on my testing, including a lot of context information can help narrow the performance gap between these LLM models. For instance, when I used Claude-4 in VSCode Copilot to handle a Flutter issue, it messed up the files in three rounds of conversation, and I had to roll everything back. However, when I used claude code with DeepSeek, after three or four rounds of conversation, I finally managed to complete my task—and the cost was less than 1 RMB!
|
|
||||||
|
|
||||||
## Some articles:
|
|
||||||
|
|
||||||
1. [Project Motivation and Principles](blog/en/project-motivation-and-how-it-works.md) ([项目初衷及原理](blog/zh/项目初衷及原理.md))
|
|
||||||
2. [Maybe We Can Do More with the Router](blog/en/maybe-we-can-do-more-with-the-route.md) ([或许我们能在 Router 中做更多事情](blog/zh/或许我们能在Router中做更多事情.md))
|
|
||||||
|
|
||||||
## Buy me a coffee
|
|
||||||
|
|
||||||
If you find this project helpful, you can choose to sponsor the author with a cup of coffee. Please provide your GitHub information so I can add you to the sponsor list below.
|
|
||||||
|
|
||||||
[](https://ko-fi.com/F1F31GN2GM)
|
[](https://ko-fi.com/F1F31GN2GM)
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
<tr>
|
<tr>
|
||||||
<td><img src="/blog/images/alipay.jpg" width="200" /></td>
|
<td><img src="/blog/images/alipay.jpg" width="200" alt="Alipay" /></td>
|
||||||
<td><img src="/blog/images/wechat.jpg" width="200" /></td>
|
<td><img src="/blog/images/wechat.jpg" width="200" alt="WeChat Pay" /></td>
|
||||||
</tr>
|
</tr>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
## Sponsors
|
### Our Sponsors
|
||||||
|
|
||||||
Thanks to the following sponsors:
|
A huge thank you to all our sponsors for their generous support!
|
||||||
|
|
||||||
@Simon Leischnig (If you see this, feel free to contact me and I can update it with your GitHub information)
|
- @Simon Leischnig
|
||||||
[@duanshuaimin](https://github.com/duanshuaimin)
|
- [@duanshuaimin](https://github.com/duanshuaimin)
|
||||||
[@vrgitadmin](https://github.com/vrgitadmin)
|
- [@vrgitadmin](https://github.com/vrgitadmin)
|
||||||
@\*o (可通过主页邮箱联系我修改 github 用户名)
|
- @\*o
|
||||||
@\*\*聪 (可通过主页邮箱联系我修改 github 用户名)
|
- [@ceilwoo](https://github.com/ceilwoo)
|
||||||
@\*说 (可通过主页邮箱联系我修改 github 用户名)
|
- @\*说
|
||||||
@\*更 (可通过主页邮箱联系我修改 github 用户名)
|
- @\*更
|
||||||
@\*更 (可通过主页邮箱联系我修改 github 用户名)
|
- @K\*g
|
||||||
@K\*g (可通过主页邮箱联系我修改 github 用户名)
|
- @R\*R
|
||||||
@R\*R (可通过主页邮箱联系我修改 github 用户名)
|
- [@bobleer](https://github.com/bobleer)
|
||||||
@[@bobleer](https://github.com/bobleer) (可通过主页邮箱联系我修改 github 用户名)
|
- @\*苗
|
||||||
@\*苗 (可通过主页邮箱联系我修改 github 用户名)
|
- @\*划
|
||||||
@\*划 (可通过主页邮箱联系我修改 github 用户名)
|
- [@Clarence-pan](https://github.com/Clarence-pan)
|
||||||
|
- [@carter003](https://github.com/carter003)
|
||||||
|
- @S\*r
|
||||||
|
- @\*晖
|
||||||
|
- @\*敏
|
||||||
|
- @Z\*z
|
||||||
|
- @\*然
|
||||||
|
- [@cluic](https://github.com/cluic)
|
||||||
|
- @\*苗
|
||||||
|
- [@PromptExpert](https://github.com/PromptExpert)
|
||||||
|
- @\*应
|
||||||
|
- [@yusnake](https://github.com/yusnake)
|
||||||
|
- @\*飞
|
||||||
|
- @董\*
|
||||||
|
- @\*汀
|
||||||
|
- @\*涯
|
||||||
|
- @\*:-)
|
||||||
|
- @\*\*磊
|
||||||
|
- @\*琢
|
||||||
|
- @\*成
|
||||||
|
- @Z\*o
|
||||||
|
- @\*琨
|
||||||
|
- [@congzhangzh](https://github.com/congzhangzh)
|
||||||
|
- @\*\_
|
||||||
|
- @Z\*m
|
||||||
|
|
||||||
|
(If your name is masked, please contact me via my homepage email to update it with your GitHub username.)
|
||||||
|
|||||||
431
README_zh.md
Normal file
@@ -0,0 +1,431 @@
|
|||||||
|
# Claude Code Router
|
||||||
|
|
||||||
|
> 一款强大的工具,可将 Claude Code 请求路由到不同的模型,并自定义任何请求。
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
## ✨ 功能
|
||||||
|
|
||||||
|
- **模型路由**: 根据您的需求将请求路由到不同的模型(例如,后台任务、思考、长上下文)。
|
||||||
|
- **多提供商支持**: 支持 OpenRouter、DeepSeek、Ollama、Gemini、Volcengine 和 SiliconFlow 等各种模型提供商。
|
||||||
|
- **请求/响应转换**: 使用转换器为不同的提供商自定义请求和响应。
|
||||||
|
- **动态模型切换**: 在 Claude Code 中使用 `/model` 命令动态切换模型。
|
||||||
|
- **GitHub Actions 集成**: 在您的 GitHub 工作流程中触发 Claude Code 任务。
|
||||||
|
- **插件系统**: 使用自定义转换器扩展功能。
|
||||||
|
|
||||||
|
## 🚀 快速入门
|
||||||
|
|
||||||
|
### 1. 安装
|
||||||
|
|
||||||
|
首先,请确保您已安装 [Claude Code](https://docs.anthropic.com/en/docs/claude-code/quickstart):
|
||||||
|
|
||||||
|
```shell
|
||||||
|
npm install -g @anthropic-ai/claude-code
|
||||||
|
```
|
||||||
|
|
||||||
|
然后,安装 Claude Code Router:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
npm install -g @musistudio/claude-code-router
|
||||||
|
```
|
||||||
|
|
||||||
|
### 2. 配置
|
||||||
|
|
||||||
|
创建并配置您的 `~/.claude-code-router/config.json` 文件。有关更多详细信息,您可以参考 `config.example.json`。
|
||||||
|
|
||||||
|
`config.json` 文件有几个关键部分:
|
||||||
|
- **`PROXY_URL`** (可选): 您可以为 API 请求设置代理,例如:`"PROXY_URL": "http://127.0.0.1:7890"`。
|
||||||
|
- **`LOG`** (可选): 您可以通过将其设置为 `true` 来启用日志记录。日志文件将位于 `$HOME/.claude-code-router.log`。
|
||||||
|
- **`APIKEY`** (可选): 您可以设置一个密钥来进行身份验证。设置后,客户端请求必须在 `Authorization` 请求头 (例如, `Bearer your-secret-key`) 或 `x-api-key` 请求头中提供此密钥。例如:`"APIKEY": "your-secret-key"`。
|
||||||
|
- **`HOST`** (可选): 您可以设置服务的主机地址。如果未设置 `APIKEY`,出于安全考虑,主机地址将强制设置为 `127.0.0.1`,以防止未经授权的访问。例如:`"HOST": "0.0.0.0"`。
|
||||||
|
- **`Providers`**: 用于配置不同的模型提供商。
|
||||||
|
- **`Router`**: 用于设置路由规则。`default` 指定默认模型,如果未配置其他路由,则该模型将用于所有请求。
|
||||||
|
|
||||||
|
这是一个综合示例:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"APIKEY": "your-secret-key",
|
||||||
|
"PROXY_URL": "http://127.0.0.1:7890",
|
||||||
|
"LOG": true,
|
||||||
|
"Providers": [
|
||||||
|
{
|
||||||
|
"name": "openrouter",
|
||||||
|
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": [
|
||||||
|
"google/gemini-2.5-pro-preview",
|
||||||
|
"anthropic/claude-sonnet-4",
|
||||||
|
"anthropic/claude-3.5-sonnet",
|
||||||
|
"anthropic/claude-3.7-sonnet:thinking"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["openrouter"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "deepseek",
|
||||||
|
"api_base_url": "https://api.deepseek.com/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["deepseek-chat", "deepseek-reasoner"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["deepseek"],
|
||||||
|
"deepseek-chat": {
|
||||||
|
"use": ["tooluse"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "ollama",
|
||||||
|
"api_base_url": "http://localhost:11434/v1/chat/completions",
|
||||||
|
"api_key": "ollama",
|
||||||
|
"models": ["qwen2.5-coder:latest"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "gemini",
|
||||||
|
"api_base_url": "https://generativelanguage.googleapis.com/v1beta/models/",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["gemini-2.5-flash", "gemini-2.5-pro"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["gemini"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "volcengine",
|
||||||
|
"api_base_url": "https://ark.cn-beijing.volces.com/api/v3/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["deepseek-v3-250324", "deepseek-r1-250528"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["deepseek"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "modelscope",
|
||||||
|
"api_base_url": "https://api-inference.modelscope.cn/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": ["Qwen/Qwen3-Coder-480B-A35B-Instruct", "Qwen/Qwen3-235B-A22B-Thinking-2507"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 65536
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"enhancetool"
|
||||||
|
],
|
||||||
|
"Qwen/Qwen3-235B-A22B-Thinking-2507": {
|
||||||
|
"use": ["reasoning"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "dashscope",
|
||||||
|
"api_base_url": "https://dashscope.aliyuncs.com/compatible-mode/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": ["qwen3-coder-plus"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 65536
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"enhancetool"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Router": {
|
||||||
|
"default": "deepseek,deepseek-chat",
|
||||||
|
"background": "ollama,qwen2.5-coder:latest",
|
||||||
|
"think": "deepseek,deepseek-reasoner",
|
||||||
|
"longContext": "openrouter,google/gemini-2.5-pro-preview",
|
||||||
|
"longContextThreshold": 60000,
|
||||||
|
"webSearch": "gemini,gemini-2.5-flash"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### 3. 使用 Router 运行 Claude Code
|
||||||
|
|
||||||
|
使用 router 启动 Claude Code:
|
||||||
|
|
||||||
|
```shell
|
||||||
|
ccr code
|
||||||
|
```
|
||||||
|
|
||||||
|
> **注意**: 修改配置文件后,需要重启服务使配置生效:
|
||||||
|
> ```shell
|
||||||
|
> ccr restart
|
||||||
|
> ```
|
||||||
|
|
||||||
|
#### Providers
|
||||||
|
|
||||||
|
`Providers` 数组是您定义要使用的不同模型提供商的地方。每个提供商对象都需要:
|
||||||
|
|
||||||
|
- `name`: 提供商的唯一名称。
|
||||||
|
- `api_base_url`: 聊天补全的完整 API 端点。
|
||||||
|
- `api_key`: 您提供商的 API 密钥。
|
||||||
|
- `models`: 此提供商可用的模型名称列表。
|
||||||
|
- `transformer` (可选): 指定用于处理请求和响应的转换器。
|
||||||
|
|
||||||
|
#### Transformers
|
||||||
|
|
||||||
|
Transformers 允许您修改请求和响应负载,以确保与不同提供商 API 的兼容性。
|
||||||
|
|
||||||
|
- **全局 Transformer**: 将转换器应用于提供商的所有模型。在此示例中,`openrouter` 转换器将应用于 `openrouter` 提供商下的所有模型。
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "openrouter",
|
||||||
|
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": [
|
||||||
|
"google/gemini-2.5-pro-preview",
|
||||||
|
"anthropic/claude-sonnet-4",
|
||||||
|
"anthropic/claude-3.5-sonnet"
|
||||||
|
],
|
||||||
|
"transformer": { "use": ["openrouter"] }
|
||||||
|
}
|
||||||
|
```
|
||||||
|
- **特定于模型的 Transformer**: 将转换器应用于特定模型。在此示例中,`deepseek` 转换器应用于所有模型,而额外的 `tooluse` 转换器仅应用于 `deepseek-chat` 模型。
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "deepseek",
|
||||||
|
"api_base_url": "https://api.deepseek.com/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["deepseek-chat", "deepseek-reasoner"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["deepseek"],
|
||||||
|
"deepseek-chat": { "use": ["tooluse"] }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
- **向 Transformer 传递选项**: 某些转换器(如 `maxtoken`)接受选项。要传递选项,请使用嵌套数组,其中第一个元素是转换器名称,第二个元素是选项对象。
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"name": "siliconflow",
|
||||||
|
"api_base_url": "https://api.siliconflow.cn/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["moonshotai/Kimi-K2-Instruct"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 16384
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
**可用的内置 Transformer:**
|
||||||
|
|
||||||
|
- `deepseek`: 适配 DeepSeek API 的请求/响应。
|
||||||
|
- `gemini`: 适配 Gemini API 的请求/响应。
|
||||||
|
- `openrouter`: 适配 OpenRouter API 的请求/响应。
|
||||||
|
- `groq`: 适配 groq API 的请求/响应
|
||||||
|
- `maxtoken`: 设置特定的 `max_tokens` 值。
|
||||||
|
- `tooluse`: 优化某些模型的工具使用(通过`tool_choice`参数)。
|
||||||
|
- `gemini-cli` (实验性): 通过 Gemini CLI [gemini-cli.js](https://gist.github.com/musistudio/1c13a65f35916a7ab690649d3df8d1cd) 对 Gemini 的非官方支持。
|
||||||
|
|
||||||
|
**自定义 Transformer:**
|
||||||
|
|
||||||
|
您还可以创建自己的转换器,并通过 `config.json` 中的 `transformers` 字段加载它们。
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"transformers": [
|
||||||
|
{
|
||||||
|
"path": "$HOME/.claude-code-router/plugins/gemini-cli.js",
|
||||||
|
"options": {
|
||||||
|
"project": "xxx"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Router
|
||||||
|
|
||||||
|
`Router` 对象定义了在不同场景下使用哪个模型:
|
||||||
|
|
||||||
|
- `default`: 用于常规任务的默认模型。
|
||||||
|
- `background`: 用于后台任务的模型。这可以是一个较小的本地模型以节省成本。
|
||||||
|
- `think`: 用于推理密集型任务(如计划模式)的模型。
|
||||||
|
- `longContext`: 用于处理长上下文(例如,> 60K 令牌)的模型。
|
||||||
|
- `longContextThreshold` (可选): 触发长上下文模型的令牌数阈值。如果未指定,默认为 60000。
|
||||||
|
- `webSearch`: 用于处理网络搜索任务,需要模型本身支持。如果使用`openrouter`需要在模型后面加上`:online`后缀。
|
||||||
|
|
||||||
|
您还可以使用 `/model` 命令在 Claude Code 中动态切换模型:
|
||||||
|
`/model provider_name,model_name`
|
||||||
|
示例: `/model openrouter,anthropic/claude-3.5-sonnet`
|
||||||
|
|
||||||
|
#### 自定义路由器
|
||||||
|
|
||||||
|
对于更高级的路由逻辑,您可以在 `config.json` 中通过 `CUSTOM_ROUTER_PATH` 字段指定一个自定义路由器脚本。这允许您实现超出默认场景的复杂路由规则。
|
||||||
|
|
||||||
|
在您的 `config.json` 中配置:
|
||||||
|
|
||||||
|
```json
|
||||||
|
{
|
||||||
|
"CUSTOM_ROUTER_PATH": "$HOME/.claude-code-router/custom-router.js"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
自定义路由器文件必须是一个导出 `async` 函数的 JavaScript 模块。该函数接收请求对象和配置对象作为参数,并应返回提供商和模型名称的字符串(例如 `"provider_name,model_name"`),如果返回 `null` 则回退到默认路由。
|
||||||
|
|
||||||
|
这是一个基于 `custom-router.example.js` 的 `custom-router.js` 示例:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// $HOME/.claude-code-router/custom-router.js
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 一个自定义路由函数,用于根据请求确定使用哪个模型。
|
||||||
|
*
|
||||||
|
* @param {object} req - 来自 Claude Code 的请求对象,包含请求体。
|
||||||
|
* @param {object} config - 应用程序的配置对象。
|
||||||
|
* @returns {Promise<string|null>} - 一个解析为 "provider,model_name" 字符串的 Promise,如果返回 null,则使用默认路由。
|
||||||
|
*/
|
||||||
|
module.exports = async function router(req, config) {
|
||||||
|
const userMessage = req.body.messages.find(m => m.role === 'user')?.content;
|
||||||
|
|
||||||
|
if (userMessage && userMessage.includes('解释这段代码')) {
|
||||||
|
// 为代码解释任务使用更强大的模型
|
||||||
|
return 'openrouter,anthropic/claude-3.5-sonnet';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 回退到默认的路由配置
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
## 🤖 GitHub Actions
|
||||||
|
|
||||||
|
将 Claude Code Router 集成到您的 CI/CD 管道中。在设置 [Claude Code Actions](https://docs.anthropic.com/en/docs/claude-code/github-actions) 后,修改您的 `.github/workflows/claude.yaml` 以使用路由器:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
name: Claude Code
|
||||||
|
|
||||||
|
on:
|
||||||
|
issue_comment:
|
||||||
|
types: [created]
|
||||||
|
# ... other triggers
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
claude:
|
||||||
|
if: |
|
||||||
|
(github.event_name == 'issue_comment' && contains(github.event.comment.body, '@claude')) ||
|
||||||
|
# ... other conditions
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
pull-requests: read
|
||||||
|
issues: read
|
||||||
|
id-token: write
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 1
|
||||||
|
|
||||||
|
- name: Prepare Environment
|
||||||
|
run: |
|
||||||
|
curl -fsSL https://bun.sh/install | bash
|
||||||
|
mkdir -p $HOME/.claude-code-router
|
||||||
|
cat << 'EOF' > $HOME/.claude-code-router/config.json
|
||||||
|
{
|
||||||
|
"log": true,
|
||||||
|
"OPENAI_API_KEY": "${{ secrets.OPENAI_API_KEY }}",
|
||||||
|
"OPENAI_BASE_URL": "https://api.deepseek.com",
|
||||||
|
"OPENAI_MODEL": "deepseek-chat"
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Start Claude Code Router
|
||||||
|
run: |
|
||||||
|
nohup ~/.bun/bin/bunx @musistudio/claude-code-router@1.0.8 start &
|
||||||
|
shell: bash
|
||||||
|
|
||||||
|
- name: Run Claude Code
|
||||||
|
id: claude
|
||||||
|
uses: anthropics/claude-code-action@beta
|
||||||
|
env:
|
||||||
|
ANTHROPIC_BASE_URL: http://localhost:3456
|
||||||
|
with:
|
||||||
|
anthropic_api_key: "any-string-is-ok"
|
||||||
|
```
|
||||||
|
|
||||||
|
这种设置可以实现有趣的自动化,例如在非高峰时段运行任务以降低 API 成本。
|
||||||
|
|
||||||
|
## 📝 深入阅读
|
||||||
|
|
||||||
|
- [项目动机和工作原理](blog/zh/项目初衷及原理.md)
|
||||||
|
- [也许我们可以用路由器做更多事情](blog/zh/或许我们能在Router中做更多事情.md)
|
||||||
|
|
||||||
|
## ❤️ 支持与赞助
|
||||||
|
|
||||||
|
如果您觉得这个项目有帮助,请考虑赞助它的开发。非常感谢您的支持!
|
||||||
|
|
||||||
|
[](https://ko-fi.com/F1F31GN2GM)
|
||||||
|
|
||||||
|
<table>
|
||||||
|
<tr>
|
||||||
|
<td><img src="/blog/images/alipay.jpg" width="200" alt="Alipay" /></td>
|
||||||
|
<td><img src="/blog/images/wechat.jpg" width="200" alt="WeChat Pay" /></td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
### 我们的赞助商
|
||||||
|
|
||||||
|
非常感谢所有赞助商的慷慨支持!
|
||||||
|
|
||||||
|
- @Simon Leischnig
|
||||||
|
- [@duanshuaimin](https://github.com/duanshuaimin)
|
||||||
|
- [@vrgitadmin](https://github.com/vrgitadmin)
|
||||||
|
- @*o
|
||||||
|
- [@ceilwoo](https://github.com/ceilwoo)
|
||||||
|
- @*说
|
||||||
|
- @*更
|
||||||
|
- @K*g
|
||||||
|
- @R*R
|
||||||
|
- [@bobleer](https://github.com/bobleer)
|
||||||
|
- @*苗
|
||||||
|
- @*划
|
||||||
|
- [@Clarence-pan](https://github.com/Clarence-pan)
|
||||||
|
- [@carter003](https://github.com/carter003)
|
||||||
|
- @S*r
|
||||||
|
- @*晖
|
||||||
|
- @*敏
|
||||||
|
- @Z*z
|
||||||
|
- @*然
|
||||||
|
- [@cluic](https://github.com/cluic)
|
||||||
|
- @*苗
|
||||||
|
- [@PromptExpert](https://github.com/PromptExpert)
|
||||||
|
- @*应
|
||||||
|
- [@yusnake](https://github.com/yusnake)
|
||||||
|
- @*飞
|
||||||
|
- @董*
|
||||||
|
- @*汀
|
||||||
|
- @*涯
|
||||||
|
- @*:-)
|
||||||
|
- @**磊
|
||||||
|
- @*琢
|
||||||
|
- @*成
|
||||||
|
- @Z*o
|
||||||
|
- [@congzhangzh](https://github.com/congzhangzh)
|
||||||
|
- @*_
|
||||||
|
- @Z\*m
|
||||||
|
|
||||||
|
(如果您的名字被屏蔽,请通过我的主页电子邮件与我联系,以便使用您的 GitHub 用户名进行更新。)
|
||||||
|
|
||||||
|
|
||||||
|
## 交流群
|
||||||
|
<img src="/blog/images/wechat_group.jpg" width="200" alt="wechat_group" />
|
||||||
|
Before Width: | Height: | Size: 353 KiB After Width: | Height: | Size: 353 KiB |
BIN
blog/images/wechat_group.jpg
Normal file
|
After Width: | Height: | Size: 243 KiB |
117
config.example.json
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
{
|
||||||
|
"Providers": [
|
||||||
|
{
|
||||||
|
"name": "openrouter",
|
||||||
|
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": [
|
||||||
|
"google/gemini-2.5-pro-preview",
|
||||||
|
"anthropic/claude-sonnet-4",
|
||||||
|
"anthropic/claude-3.5-sonnet",
|
||||||
|
"anthropic/claude-3.7-sonnet:thinking"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["openrouter"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "deepseek",
|
||||||
|
"api_base_url": "https://api.deepseek.com/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["deepseek-chat", "deepseek-reasoner"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["deepseek"],
|
||||||
|
"deepseek-chat": {
|
||||||
|
"use": ["tooluse"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "ollama",
|
||||||
|
"api_base_url": "http://localhost:11434/v1/chat/completions",
|
||||||
|
"api_key": "ollama",
|
||||||
|
"models": ["qwen2.5-coder:latest"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "gemini",
|
||||||
|
"api_base_url": "https://generativelanguage.googleapis.com/v1beta/models/",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["gemini-2.5-flash", "gemini-2.5-pro"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["gemini"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "volcengine",
|
||||||
|
"api_base_url": "https://ark.cn-beijing.volces.com/api/v3/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["deepseek-v3-250324", "deepseek-r1-250528"],
|
||||||
|
"transformer": {
|
||||||
|
"use": ["deepseek"]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "siliconflow",
|
||||||
|
"api_base_url": "https://api.siliconflow.cn/v1/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": ["moonshotai/Kimi-K2-Instruct"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 16384
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "modelscope",
|
||||||
|
"api_base_url": "https://api-inference.modelscope.cn/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": ["Qwen/Qwen3-Coder-480B-A35B-Instruct", "Qwen/Qwen3-235B-A22B-Thinking-2507"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 65536
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"enhancetool"
|
||||||
|
],
|
||||||
|
"Qwen/Qwen3-235B-A22B-Thinking-2507": {
|
||||||
|
"use": ["reasoning"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "dashscope",
|
||||||
|
"api_base_url": "https://dashscope.aliyuncs.com/compatible-mode/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": ["qwen3-coder-plus"],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 65536
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"enhancetool"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Router": {
|
||||||
|
"default": "deepseek,deepseek-chat",
|
||||||
|
"background": "ollama,qwen2.5-coder:latest",
|
||||||
|
"think": "deepseek,deepseek-reasoner",
|
||||||
|
"longContext": "openrouter,google/gemini-2.5-pro-preview",
|
||||||
|
"longContextThreshold": 60000,
|
||||||
|
"webSearch": "gemini,gemini-2.5-flash"
|
||||||
|
},
|
||||||
|
"APIKEY": "your-secret-key",
|
||||||
|
"HOST": "0.0.0.0"
|
||||||
|
}
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
{
|
|
||||||
"usePlugin": "",
|
|
||||||
"LOG": true,
|
|
||||||
"OPENAI_API_KEY": "",
|
|
||||||
"OPENAI_BASE_URL": "",
|
|
||||||
"OPENAI_MODEL": "",
|
|
||||||
"modelProviders": {}
|
|
||||||
}
|
|
||||||
3
custom-router.example.js
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
module.exports = async function router(req, config) {
|
||||||
|
return "deepseek,deepseek-chat";
|
||||||
|
};
|
||||||
1927
package-lock.json
generated
13
package.json
@@ -1,12 +1,13 @@
|
|||||||
{
|
{
|
||||||
"name": "@musistudio/claude-code-router",
|
"name": "@musistudio/claude-code-router",
|
||||||
"version": "1.0.9",
|
"version": "1.0.28",
|
||||||
"description": "Use Claude Code without an Anthropics account and route it to another LLM provider",
|
"description": "Use Claude Code without an Anthropics account and route it to another LLM provider",
|
||||||
"bin": {
|
"bin": {
|
||||||
"ccr": "./dist/cli.js"
|
"ccr": "./dist/cli.js"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "esbuild src/cli.ts --bundle --platform=node --outfile=dist/cli.js && cp node_modules/tiktoken/tiktoken_bg.wasm dist/tiktoken_bg.wasm"
|
"build": "node scripts/build.js",
|
||||||
|
"release": "npm run build && npm publish"
|
||||||
},
|
},
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"claude",
|
"claude",
|
||||||
@@ -18,13 +19,19 @@
|
|||||||
"author": "musistudio",
|
"author": "musistudio",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@musistudio/llms": "^1.0.0",
|
"@fastify/static": "^8.2.0",
|
||||||
|
"@musistudio/llms": "file:../llms",
|
||||||
"dotenv": "^16.4.7",
|
"dotenv": "^16.4.7",
|
||||||
|
"json5": "^2.2.3",
|
||||||
|
"openurl": "^1.1.1",
|
||||||
"tiktoken": "^1.0.21",
|
"tiktoken": "^1.0.21",
|
||||||
"uuid": "^11.1.0"
|
"uuid": "^11.1.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@types/node": "^24.0.15",
|
||||||
"esbuild": "^0.25.1",
|
"esbuild": "^0.25.1",
|
||||||
|
"fastify": "^5.4.0",
|
||||||
|
"shx": "^0.4.0",
|
||||||
"typescript": "^5.8.2"
|
"typescript": "^5.8.2"
|
||||||
},
|
},
|
||||||
"publishConfig": {
|
"publishConfig": {
|
||||||
|
|||||||
927
pnpm-lock.yaml
generated
|
Before Width: | Height: | Size: 48 KiB |
|
Before Width: | Height: | Size: 52 KiB |
|
Before Width: | Height: | Size: 36 KiB |
|
Before Width: | Height: | Size: 83 KiB |
35
scripts/build.js
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
const { execSync } = require('child_process');
|
||||||
|
const path = require('path');
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
console.log('Building Claude Code Router...');
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Build the main CLI application
|
||||||
|
console.log('Building CLI application...');
|
||||||
|
execSync('esbuild src/cli.ts --bundle --platform=node --outfile=dist/cli.js', { stdio: 'inherit' });
|
||||||
|
|
||||||
|
// Copy the tiktoken WASM file
|
||||||
|
console.log('Copying tiktoken WASM file...');
|
||||||
|
execSync('shx cp node_modules/tiktoken/tiktoken_bg.wasm dist/tiktoken_bg.wasm', { stdio: 'inherit' });
|
||||||
|
|
||||||
|
// Build the UI
|
||||||
|
console.log('Building UI...');
|
||||||
|
// Check if node_modules exists in ui directory, if not install dependencies
|
||||||
|
if (!fs.existsSync('ui/node_modules')) {
|
||||||
|
console.log('Installing UI dependencies...');
|
||||||
|
execSync('cd ui && npm install', { stdio: 'inherit' });
|
||||||
|
}
|
||||||
|
execSync('cd ui && npm run build', { stdio: 'inherit' });
|
||||||
|
|
||||||
|
// Copy the built UI index.html to dist
|
||||||
|
console.log('Copying UI build artifacts...');
|
||||||
|
execSync('shx cp ui/dist/index.html dist/index.html', { stdio: 'inherit' });
|
||||||
|
|
||||||
|
console.log('Build completed successfully!');
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Build failed:', error.message);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
129
src/cli.ts
@@ -2,11 +2,12 @@
|
|||||||
import { run } from "./index";
|
import { run } from "./index";
|
||||||
import { showStatus } from "./utils/status";
|
import { showStatus } from "./utils/status";
|
||||||
import { executeCodeCommand } from "./utils/codeCommand";
|
import { executeCodeCommand } from "./utils/codeCommand";
|
||||||
import { cleanupPidFile, isServiceRunning } from "./utils/processCheck";
|
import { cleanupPidFile, isServiceRunning, getServiceInfo } from "./utils/processCheck";
|
||||||
import { version } from "../package.json";
|
import { version } from "../package.json";
|
||||||
import { spawn } from "child_process";
|
import { spawn, exec } from "child_process";
|
||||||
import { PID_FILE, REFERENCE_COUNT_FILE } from "./constants";
|
import { PID_FILE, REFERENCE_COUNT_FILE } from "./constants";
|
||||||
import { existsSync, readFileSync } from "fs";
|
import fs, { existsSync, readFileSync } from "fs";
|
||||||
|
import {join} from "path";
|
||||||
|
|
||||||
const command = process.argv[2];
|
const command = process.argv[2];
|
||||||
|
|
||||||
@@ -14,16 +15,19 @@ const HELP_TEXT = `
|
|||||||
Usage: ccr [command]
|
Usage: ccr [command]
|
||||||
|
|
||||||
Commands:
|
Commands:
|
||||||
start Start service
|
start Start server
|
||||||
stop Stop service
|
stop Stop server
|
||||||
status Show service status
|
restart Restart server
|
||||||
code Execute code command
|
status Show server status
|
||||||
|
code Execute claude command
|
||||||
|
ui Open the web UI in browser
|
||||||
-v, version Show version information
|
-v, version Show version information
|
||||||
-h, help Show help information
|
-h, help Show help information
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
ccr start
|
ccr start
|
||||||
ccr code "Write a Hello World"
|
ccr code "Write a Hello World"
|
||||||
|
ccr ui
|
||||||
`;
|
`;
|
||||||
|
|
||||||
async function waitForService(
|
async function waitForService(
|
||||||
@@ -57,7 +61,7 @@ async function main() {
|
|||||||
cleanupPidFile();
|
cleanupPidFile();
|
||||||
if (existsSync(REFERENCE_COUNT_FILE)) {
|
if (existsSync(REFERENCE_COUNT_FILE)) {
|
||||||
try {
|
try {
|
||||||
require("fs").unlinkSync(REFERENCE_COUNT_FILE);
|
fs.unlinkSync(REFERENCE_COUNT_FILE);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// Ignore cleanup errors
|
// Ignore cleanup errors
|
||||||
}
|
}
|
||||||
@@ -73,21 +77,34 @@ async function main() {
|
|||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case "status":
|
case "status":
|
||||||
showStatus();
|
await showStatus();
|
||||||
break;
|
break;
|
||||||
case "code":
|
case "code":
|
||||||
if (!isServiceRunning()) {
|
if (!isServiceRunning()) {
|
||||||
console.log("Service not running, starting service...");
|
console.log("Service not running, starting service...");
|
||||||
const startProcess = spawn("ccr", ["start"], {
|
const cliPath = join(__dirname, "cli.js");
|
||||||
|
const startProcess = spawn("node", [cliPath, "start"], {
|
||||||
detached: true,
|
detached: true,
|
||||||
stdio: "ignore",
|
stdio: "ignore",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// let errorMessage = "";
|
||||||
|
// startProcess.stderr?.on("data", (data) => {
|
||||||
|
// errorMessage += data.toString();
|
||||||
|
// });
|
||||||
|
|
||||||
startProcess.on("error", (error) => {
|
startProcess.on("error", (error) => {
|
||||||
console.error("Failed to start service:", error);
|
console.error("Failed to start service:", error.message);
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// startProcess.on("close", (code) => {
|
||||||
|
// if (code !== 0 && errorMessage) {
|
||||||
|
// console.error("Failed to start service:", errorMessage.trim());
|
||||||
|
// process.exit(1);
|
||||||
|
// }
|
||||||
|
// });
|
||||||
|
|
||||||
startProcess.unref();
|
startProcess.unref();
|
||||||
|
|
||||||
if (await waitForService()) {
|
if (await waitForService()) {
|
||||||
@@ -102,10 +119,100 @@ async function main() {
|
|||||||
executeCodeCommand(process.argv.slice(3));
|
executeCodeCommand(process.argv.slice(3));
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
case "ui":
|
||||||
|
// Check if service is running
|
||||||
|
if (!isServiceRunning()) {
|
||||||
|
console.log("Service not running, starting service...");
|
||||||
|
const cliPath = join(__dirname, "cli.js");
|
||||||
|
const startProcess = spawn("node", [cliPath, "start"], {
|
||||||
|
detached: true,
|
||||||
|
stdio: "ignore",
|
||||||
|
});
|
||||||
|
|
||||||
|
startProcess.on("error", (error) => {
|
||||||
|
console.error("Failed to start service:", error.message);
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
startProcess.unref();
|
||||||
|
|
||||||
|
if (!(await waitForService())) {
|
||||||
|
console.error(
|
||||||
|
"Service startup timeout, please manually run `ccr start` to start the service"
|
||||||
|
);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get service info and open UI
|
||||||
|
const serviceInfo = await getServiceInfo();
|
||||||
|
const uiUrl = `${serviceInfo.endpoint}/ui/`;
|
||||||
|
console.log(`Opening UI at ${uiUrl}`);
|
||||||
|
|
||||||
|
// Open URL in browser based on platform
|
||||||
|
const platform = process.platform;
|
||||||
|
let openCommand = "";
|
||||||
|
|
||||||
|
if (platform === "win32") {
|
||||||
|
// Windows
|
||||||
|
openCommand = `start ${uiUrl}`;
|
||||||
|
} else if (platform === "darwin") {
|
||||||
|
// macOS
|
||||||
|
openCommand = `open ${uiUrl}`;
|
||||||
|
} else if (platform === "linux") {
|
||||||
|
// Linux
|
||||||
|
openCommand = `xdg-open ${uiUrl}`;
|
||||||
|
} else {
|
||||||
|
console.error("Unsupported platform for opening browser");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
exec(openCommand, (error) => {
|
||||||
|
if (error) {
|
||||||
|
console.error("Failed to open browser:", error.message);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
break;
|
||||||
case "-v":
|
case "-v":
|
||||||
case "version":
|
case "version":
|
||||||
console.log(`claude-code-router version: ${version}`);
|
console.log(`claude-code-router version: ${version}`);
|
||||||
break;
|
break;
|
||||||
|
case "restart":
|
||||||
|
// Stop the service if it's running
|
||||||
|
try {
|
||||||
|
const pid = parseInt(readFileSync(PID_FILE, "utf-8"));
|
||||||
|
process.kill(pid);
|
||||||
|
cleanupPidFile();
|
||||||
|
if (existsSync(REFERENCE_COUNT_FILE)) {
|
||||||
|
try {
|
||||||
|
fs.unlinkSync(REFERENCE_COUNT_FILE);
|
||||||
|
} catch (e) {
|
||||||
|
// Ignore cleanup errors
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log("claude code router service has been stopped.");
|
||||||
|
} catch (e) {
|
||||||
|
console.log("Service was not running or failed to stop.");
|
||||||
|
cleanupPidFile();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start the service again in the background
|
||||||
|
console.log("Starting claude code router service...");
|
||||||
|
const cliPath = join(__dirname, "cli.js");
|
||||||
|
const startProcess = spawn("node", [cliPath, "start"], {
|
||||||
|
detached: true,
|
||||||
|
stdio: "ignore",
|
||||||
|
});
|
||||||
|
|
||||||
|
startProcess.on("error", (error) => {
|
||||||
|
console.error("Failed to start service:", error);
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
startProcess.unref();
|
||||||
|
console.log("✅ Service started successfully in the background.");
|
||||||
|
break;
|
||||||
case "-h":
|
case "-h":
|
||||||
case "help":
|
case "help":
|
||||||
console.log(HELP_TEXT);
|
console.log(HELP_TEXT);
|
||||||
|
|||||||
@@ -3,13 +3,13 @@ import os from "node:os";
|
|||||||
|
|
||||||
export const HOME_DIR = path.join(os.homedir(), ".claude-code-router");
|
export const HOME_DIR = path.join(os.homedir(), ".claude-code-router");
|
||||||
|
|
||||||
export const CONFIG_FILE = `${HOME_DIR}/config.json`;
|
export const CONFIG_FILE = path.join(HOME_DIR, "config.json");
|
||||||
|
|
||||||
export const PLUGINS_DIR = `${HOME_DIR}/plugins`;
|
export const PLUGINS_DIR = path.join(HOME_DIR, "plugins");
|
||||||
|
|
||||||
export const PID_FILE = path.join(HOME_DIR, '.claude-code-router.pid');
|
export const PID_FILE = path.join(HOME_DIR, '.claude-code-router.pid');
|
||||||
|
|
||||||
export const REFERENCE_COUNT_FILE = '/tmp/claude-code-reference-count.txt';
|
export const REFERENCE_COUNT_FILE = path.join(os.tmpdir(), "claude-code-reference-count.txt");
|
||||||
|
|
||||||
|
|
||||||
export const DEFAULT_CONFIG = {
|
export const DEFAULT_CONFIG = {
|
||||||
|
|||||||
36
src/index.ts
@@ -5,15 +5,17 @@ import { join } from "path";
|
|||||||
import { initConfig, initDir } from "./utils";
|
import { initConfig, initDir } from "./utils";
|
||||||
import { createServer } from "./server";
|
import { createServer } from "./server";
|
||||||
import { router } from "./utils/router";
|
import { router } from "./utils/router";
|
||||||
|
import { apiKeyAuth } from "./middleware/auth";
|
||||||
import {
|
import {
|
||||||
cleanupPidFile,
|
cleanupPidFile,
|
||||||
isServiceRunning,
|
isServiceRunning,
|
||||||
savePid,
|
savePid,
|
||||||
} from "./utils/processCheck";
|
} from "./utils/processCheck";
|
||||||
|
import { CONFIG_FILE } from "./constants";
|
||||||
|
|
||||||
async function initializeClaudeConfig() {
|
async function initializeClaudeConfig() {
|
||||||
const homeDir = process.env.HOME;
|
const homeDir = homedir();
|
||||||
const configPath = `${homeDir}/.claude.json`;
|
const configPath = join(homeDir, ".claude.json");
|
||||||
if (!existsSync(configPath)) {
|
if (!existsSync(configPath)) {
|
||||||
const userID = Array.from(
|
const userID = Array.from(
|
||||||
{ length: 64 },
|
{ length: 64 },
|
||||||
@@ -45,8 +47,16 @@ async function run(options: RunOptions = {}) {
|
|||||||
await initializeClaudeConfig();
|
await initializeClaudeConfig();
|
||||||
await initDir();
|
await initDir();
|
||||||
const config = await initConfig();
|
const config = await initConfig();
|
||||||
|
let HOST = config.HOST;
|
||||||
|
|
||||||
const port = options.port || 3456;
|
if (config.HOST && !config.APIKEY) {
|
||||||
|
HOST = "127.0.0.1";
|
||||||
|
console.warn(
|
||||||
|
"⚠️ API key is not set. HOST is forced to 127.0.0.1."
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const port = config.PORT || 3456;
|
||||||
|
|
||||||
// Save the PID of the background process
|
// Save the PID of the background process
|
||||||
savePid(process.pid);
|
savePid(process.pid);
|
||||||
@@ -63,20 +73,32 @@ async function run(options: RunOptions = {}) {
|
|||||||
cleanupPidFile();
|
cleanupPidFile();
|
||||||
process.exit(0);
|
process.exit(0);
|
||||||
});
|
});
|
||||||
|
console.log(HOST)
|
||||||
|
|
||||||
// Use port from environment variable if set (for background process)
|
// Use port from environment variable if set (for background process)
|
||||||
const servicePort = process.env.SERVICE_PORT
|
const servicePort = process.env.SERVICE_PORT
|
||||||
? parseInt(process.env.SERVICE_PORT)
|
? parseInt(process.env.SERVICE_PORT)
|
||||||
: port;
|
: port;
|
||||||
const server = createServer({
|
const server = createServer({
|
||||||
...config,
|
jsonPath: CONFIG_FILE,
|
||||||
|
initialConfig: {
|
||||||
|
// ...config,
|
||||||
providers: config.Providers || config.providers,
|
providers: config.Providers || config.providers,
|
||||||
|
HOST: HOST,
|
||||||
PORT: servicePort,
|
PORT: servicePort,
|
||||||
LOG_FILE: join(homedir(), ".claude-code-router", "claude-code-router.log"),
|
LOG_FILE: join(
|
||||||
|
homedir(),
|
||||||
|
".claude-code-router",
|
||||||
|
"claude-code-router.log"
|
||||||
|
),
|
||||||
|
},
|
||||||
});
|
});
|
||||||
server.addHook("preHandler", async (req, reply) =>
|
server.addHook("preHandler", apiKeyAuth(config));
|
||||||
|
server.addHook("preHandler", async (req, reply) => {
|
||||||
|
if(req.url.startsWith("/v1/messages")) {
|
||||||
router(req, reply, config)
|
router(req, reply, config)
|
||||||
);
|
}
|
||||||
|
});
|
||||||
server.start();
|
server.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
33
src/middleware/auth.ts
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
import { FastifyRequest, FastifyReply } from "fastify";
|
||||||
|
|
||||||
|
export const apiKeyAuth =
|
||||||
|
(config: any) =>
|
||||||
|
(req: FastifyRequest, reply: FastifyReply, done: () => void) => {
|
||||||
|
if (["/", "/health"].includes(req.url) || req.url.startsWith("/ui")) {
|
||||||
|
return done();
|
||||||
|
}
|
||||||
|
const apiKey = config.APIKEY;
|
||||||
|
|
||||||
|
if (!apiKey) {
|
||||||
|
return done();
|
||||||
|
}
|
||||||
|
|
||||||
|
const authKey: string =
|
||||||
|
req.headers.authorization || req.headers["x-api-key"];
|
||||||
|
if (!authKey) {
|
||||||
|
reply.status(401).send("APIKEY is missing");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let token = "";
|
||||||
|
if (authKey.startsWith("Bearer")) {
|
||||||
|
token = authKey.split(" ")[1];
|
||||||
|
} else {
|
||||||
|
token = authKey;
|
||||||
|
}
|
||||||
|
if (token !== apiKey) {
|
||||||
|
reply.status(401).send("Invalid API key");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
done();
|
||||||
|
};
|
||||||
@@ -1,8 +1,59 @@
|
|||||||
import Server from "@musistudio/llms";
|
import Server from "@musistudio/llms";
|
||||||
|
import { readConfigFile, writeConfigFile } from "./utils";
|
||||||
|
import { CONFIG_FILE } from "./constants";
|
||||||
|
import { join } from "path";
|
||||||
|
import { readFileSync } from "fs";
|
||||||
|
import fastifyStatic from "@fastify/static";
|
||||||
|
|
||||||
export const createServer = (config: any): Server => {
|
export const createServer = (config: any): Server => {
|
||||||
const server = new Server({
|
const server = new Server(config);
|
||||||
initialConfig: config,
|
|
||||||
|
// Add endpoint to read config.json
|
||||||
|
server.app.get("/api/config", async () => {
|
||||||
|
return await readConfigFile();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
server.app.get("/api/transformers", async () => {
|
||||||
|
const transformers =
|
||||||
|
server.app._server!.transformerService.getAllTransformers();
|
||||||
|
const transformerList = Array.from(transformers.entries()).map(
|
||||||
|
([name, transformer]: any) => ({
|
||||||
|
name,
|
||||||
|
endpoint: transformer.endPoint || null,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
return { transformers: transformerList };
|
||||||
|
});
|
||||||
|
|
||||||
|
// Add endpoint to save config.json
|
||||||
|
server.app.post("/api/config", async (req) => {
|
||||||
|
const newConfig = req.body;
|
||||||
|
await writeConfigFile(newConfig);
|
||||||
|
return { success: true, message: "Config saved successfully" };
|
||||||
|
});
|
||||||
|
|
||||||
|
// Add endpoint to restart the service
|
||||||
|
server.app.post("/api/restart", async (_, reply) => {
|
||||||
|
reply.send({ success: true, message: "Service restart initiated" });
|
||||||
|
|
||||||
|
// Restart the service after a short delay to allow response to be sent
|
||||||
|
setTimeout(() => {
|
||||||
|
const { spawn } = require("child_process");
|
||||||
|
spawn("ccr", ["restart"], { detached: true, stdio: "ignore" });
|
||||||
|
}, 1000);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Register static file serving with caching
|
||||||
|
server.app.register(fastifyStatic, {
|
||||||
|
root: join(__dirname, "..", "dist"),
|
||||||
|
prefix: "/ui/",
|
||||||
|
maxAge: "1h",
|
||||||
|
});
|
||||||
|
|
||||||
|
// Redirect /ui to /ui/ for proper static file serving
|
||||||
|
server.app.get("/ui", async (_, reply) => {
|
||||||
|
return reply.redirect("/ui/");
|
||||||
|
});
|
||||||
|
|
||||||
return server;
|
return server;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -4,16 +4,23 @@ import {
|
|||||||
decrementReferenceCount,
|
decrementReferenceCount,
|
||||||
} from "./processCheck";
|
} from "./processCheck";
|
||||||
import { closeService } from "./close";
|
import { closeService } from "./close";
|
||||||
|
import { readConfigFile } from ".";
|
||||||
|
|
||||||
export async function executeCodeCommand(args: string[] = []) {
|
export async function executeCodeCommand(args: string[] = []) {
|
||||||
// Set environment variables
|
// Set environment variables
|
||||||
|
const config = await readConfigFile();
|
||||||
const env = {
|
const env = {
|
||||||
...process.env,
|
...process.env,
|
||||||
ANTHROPIC_AUTH_TOKEN: "test",
|
ANTHROPIC_AUTH_TOKEN: "test",
|
||||||
ANTHROPIC_BASE_URL: `http://127.0.0.1:3456`,
|
ANTHROPIC_BASE_URL: `http://127.0.0.1:${config.PORT || 3456}`,
|
||||||
API_TIMEOUT_MS: "600000",
|
API_TIMEOUT_MS: "600000",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (config?.APIKEY) {
|
||||||
|
env.ANTHROPIC_API_KEY = config.APIKEY;
|
||||||
|
delete env.ANTHROPIC_AUTH_TOKEN;
|
||||||
|
}
|
||||||
|
|
||||||
// Increment reference count when command starts
|
// Increment reference count when command starts
|
||||||
incrementReferenceCount();
|
incrementReferenceCount();
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import fs from "node:fs/promises";
|
import fs from "node:fs/promises";
|
||||||
import readline from "node:readline";
|
import readline from "node:readline";
|
||||||
|
import JSON5 from "json5";
|
||||||
import {
|
import {
|
||||||
CONFIG_FILE,
|
CONFIG_FILE,
|
||||||
DEFAULT_CONFIG,
|
DEFAULT_CONFIG,
|
||||||
@@ -45,32 +46,49 @@ const confirm = async (query: string): Promise<boolean> => {
|
|||||||
export const readConfigFile = async () => {
|
export const readConfigFile = async () => {
|
||||||
try {
|
try {
|
||||||
const config = await fs.readFile(CONFIG_FILE, "utf-8");
|
const config = await fs.readFile(CONFIG_FILE, "utf-8");
|
||||||
return JSON.parse(config);
|
try {
|
||||||
} catch {
|
// Try to parse with JSON5 first (which also supports standard JSON)
|
||||||
const apiKey = await question("Enter OPENAI_API_KEY: ");
|
return JSON5.parse(config);
|
||||||
const baseUrl = await question("Enter OPENAI_BASE_URL: ");
|
} catch (parseError) {
|
||||||
const model = await question("Enter OPENAI_MODEL: ");
|
console.error(`Failed to parse config file at ${CONFIG_FILE}`);
|
||||||
|
console.error("Error details:", (parseError as Error).message);
|
||||||
|
console.error("Please check your config file syntax.");
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
} catch (readError: any) {
|
||||||
|
if (readError.code === "ENOENT") {
|
||||||
|
// Config file doesn't exist, prompt user for initial setup
|
||||||
|
const name = await question("Enter Provider Name: ");
|
||||||
|
const APIKEY = await question("Enter Provider API KEY: ");
|
||||||
|
const baseUrl = await question("Enter Provider URL: ");
|
||||||
|
const model = await question("Enter MODEL Name: ");
|
||||||
const config = Object.assign({}, DEFAULT_CONFIG, {
|
const config = Object.assign({}, DEFAULT_CONFIG, {
|
||||||
Providers: [
|
Providers: [
|
||||||
{
|
{
|
||||||
name: "openai",
|
name,
|
||||||
api_base_url: baseUrl,
|
api_base_url: baseUrl,
|
||||||
api_key: apiKey,
|
api_key: APIKEY,
|
||||||
models: [model],
|
models: [model],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
Router: {
|
Router: {
|
||||||
default: `openai,${model}`,
|
default: `${name},${model}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
await writeConfigFile(config);
|
await writeConfigFile(config);
|
||||||
return config;
|
return config;
|
||||||
|
} else {
|
||||||
|
console.error(`Failed to read config file at ${CONFIG_FILE}`);
|
||||||
|
console.error("Error details:", readError.message);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export const writeConfigFile = async (config: any) => {
|
export const writeConfigFile = async (config: any) => {
|
||||||
await ensureDir(HOME_DIR);
|
await ensureDir(HOME_DIR);
|
||||||
await fs.writeFile(CONFIG_FILE, JSON.stringify(config, null, 2));
|
const configWithComment = `${JSON.stringify(config, null, 2)}`;
|
||||||
|
await fs.writeFile(CONFIG_FILE, configWithComment);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const initConfig = async () => {
|
export const initConfig = async () => {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import { existsSync, readFileSync, writeFileSync } from 'fs';
|
import { existsSync, readFileSync, writeFileSync } from 'fs';
|
||||||
import { PID_FILE, REFERENCE_COUNT_FILE } from '../constants';
|
import { PID_FILE, REFERENCE_COUNT_FILE } from '../constants';
|
||||||
|
import { readConfigFile } from '.';
|
||||||
|
|
||||||
export function incrementReferenceCount() {
|
export function incrementReferenceCount() {
|
||||||
let count = 0;
|
let count = 0;
|
||||||
@@ -70,15 +71,16 @@ export function getServicePid(): number | null {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getServiceInfo() {
|
export async function getServiceInfo() {
|
||||||
const pid = getServicePid();
|
const pid = getServicePid();
|
||||||
const running = isServiceRunning();
|
const running = isServiceRunning();
|
||||||
|
const config = await readConfigFile();
|
||||||
|
|
||||||
return {
|
return {
|
||||||
running,
|
running,
|
||||||
pid,
|
pid,
|
||||||
port: 3456,
|
port: config.PORT,
|
||||||
endpoint: 'http://127.0.0.1:3456',
|
endpoint: `http://127.0.0.1:${config.PORT}`,
|
||||||
pidFile: PID_FILE,
|
pidFile: PID_FILE,
|
||||||
referenceCount: getReferenceCount()
|
referenceCount: getReferenceCount()
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,41 +1,25 @@
|
|||||||
import { MessageCreateParamsBase } from "@anthropic-ai/sdk/resources/messages";
|
import {
|
||||||
|
MessageCreateParamsBase,
|
||||||
|
MessageParam,
|
||||||
|
Tool,
|
||||||
|
} from "@anthropic-ai/sdk/resources/messages";
|
||||||
import { get_encoding } from "tiktoken";
|
import { get_encoding } from "tiktoken";
|
||||||
import { log } from "./log";
|
import { log } from "./log";
|
||||||
|
|
||||||
const enc = get_encoding("cl100k_base");
|
const enc = get_encoding("cl100k_base");
|
||||||
|
|
||||||
const getUseModel = (req: any, tokenCount: number, config: any) => {
|
const calculateTokenCount = (
|
||||||
if (req.body.model.includes(",")) {
|
messages: MessageParam[],
|
||||||
return req.body.model;
|
system: any,
|
||||||
}
|
tools: Tool[]
|
||||||
// if tokenCount is greater than 60K, use the long context model
|
) => {
|
||||||
if (tokenCount > 1000 * 60) {
|
|
||||||
log("Using long context model due to token count:", tokenCount);
|
|
||||||
return config.Router!.longContext;
|
|
||||||
}
|
|
||||||
// If the model is claude-3-5-haiku, use the background model
|
|
||||||
if (req.body.model?.startsWith("claude-3-5-haiku")) {
|
|
||||||
log("Using background model for ", req.body.model);
|
|
||||||
return config.Router!.background;
|
|
||||||
}
|
|
||||||
// if exits thinking, use the think model
|
|
||||||
if (req.body.thinking) {
|
|
||||||
log("Using think model for ", req.body.thinking);
|
|
||||||
return config.Router!.think;
|
|
||||||
}
|
|
||||||
return config.Router!.default;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const router = async (req: any, res: any, config: any) => {
|
|
||||||
const { messages, system = [], tools }: MessageCreateParamsBase = req.body;
|
|
||||||
try {
|
|
||||||
let tokenCount = 0;
|
let tokenCount = 0;
|
||||||
if (Array.isArray(messages)) {
|
if (Array.isArray(messages)) {
|
||||||
messages.forEach((message) => {
|
messages.forEach((message) => {
|
||||||
if (typeof message.content === "string") {
|
if (typeof message.content === "string") {
|
||||||
tokenCount += enc.encode(message.content).length;
|
tokenCount += enc.encode(message.content).length;
|
||||||
} else if (Array.isArray(message.content)) {
|
} else if (Array.isArray(message.content)) {
|
||||||
message.content.forEach((contentPart) => {
|
message.content.forEach((contentPart: any) => {
|
||||||
if (contentPart.type === "text") {
|
if (contentPart.type === "text") {
|
||||||
tokenCount += enc.encode(contentPart.text).length;
|
tokenCount += enc.encode(contentPart.text).length;
|
||||||
} else if (contentPart.type === "tool_use") {
|
} else if (contentPart.type === "tool_use") {
|
||||||
@@ -56,19 +40,19 @@ export const router = async (req: any, res: any, config: any) => {
|
|||||||
if (typeof system === "string") {
|
if (typeof system === "string") {
|
||||||
tokenCount += enc.encode(system).length;
|
tokenCount += enc.encode(system).length;
|
||||||
} else if (Array.isArray(system)) {
|
} else if (Array.isArray(system)) {
|
||||||
system.forEach((item) => {
|
system.forEach((item: any) => {
|
||||||
if (item.type !== "text") return;
|
if (item.type !== "text") return;
|
||||||
if (typeof item.text === "string") {
|
if (typeof item.text === "string") {
|
||||||
tokenCount += enc.encode(item.text).length;
|
tokenCount += enc.encode(item.text).length;
|
||||||
} else if (Array.isArray(item.text)) {
|
} else if (Array.isArray(item.text)) {
|
||||||
item.text.forEach((textPart) => {
|
item.text.forEach((textPart: any) => {
|
||||||
tokenCount += enc.encode(textPart || "").length;
|
tokenCount += enc.encode(textPart || "").length;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (tools) {
|
if (tools) {
|
||||||
tools.forEach((tool) => {
|
tools.forEach((tool: Tool) => {
|
||||||
if (tool.description) {
|
if (tool.description) {
|
||||||
tokenCount += enc.encode(tool.name + tool.description).length;
|
tokenCount += enc.encode(tool.name + tool.description).length;
|
||||||
}
|
}
|
||||||
@@ -77,7 +61,63 @@ export const router = async (req: any, res: any, config: any) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
const model = getUseModel(req, tokenCount, config);
|
return tokenCount;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getUseModel = async (req: any, tokenCount: number, config: any) => {
|
||||||
|
if (req.body.model.includes(",")) {
|
||||||
|
return req.body.model;
|
||||||
|
}
|
||||||
|
// if tokenCount is greater than the configured threshold, use the long context model
|
||||||
|
const longContextThreshold = config.Router.longContextThreshold || 60000;
|
||||||
|
if (tokenCount > longContextThreshold && config.Router.longContext) {
|
||||||
|
log("Using long context model due to token count:", tokenCount, "threshold:", longContextThreshold);
|
||||||
|
return config.Router.longContext;
|
||||||
|
}
|
||||||
|
// If the model is claude-3-5-haiku, use the background model
|
||||||
|
if (
|
||||||
|
req.body.model?.startsWith("claude-3-5-haiku") &&
|
||||||
|
config.Router.background
|
||||||
|
) {
|
||||||
|
log("Using background model for ", req.body.model);
|
||||||
|
return config.Router.background;
|
||||||
|
}
|
||||||
|
// if exits thinking, use the think model
|
||||||
|
if (req.body.thinking && config.Router.think) {
|
||||||
|
log("Using think model for ", req.body.thinking);
|
||||||
|
return config.Router.think;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
Array.isArray(req.body.tools) &&
|
||||||
|
req.body.tools.some((tool: any) => tool.type?.startsWith("web_search")) &&
|
||||||
|
config.Router.webSearch
|
||||||
|
) {
|
||||||
|
return config.Router.webSearch;
|
||||||
|
}
|
||||||
|
return config.Router!.default;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const router = async (req: any, _res: any, config: any) => {
|
||||||
|
const { messages, system = [], tools }: MessageCreateParamsBase = req.body;
|
||||||
|
try {
|
||||||
|
const tokenCount = calculateTokenCount(
|
||||||
|
messages as MessageParam[],
|
||||||
|
system,
|
||||||
|
tools as Tool[]
|
||||||
|
);
|
||||||
|
|
||||||
|
let model;
|
||||||
|
if (config.CUSTOM_ROUTER_PATH) {
|
||||||
|
try {
|
||||||
|
const customRouter = require(config.CUSTOM_ROUTER_PATH);
|
||||||
|
model = await customRouter(req, config);
|
||||||
|
} catch (e: any) {
|
||||||
|
log("failed to load custom router", e.message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!model) {
|
||||||
|
model = await getUseModel(req, tokenCount, config);
|
||||||
|
}
|
||||||
req.body.model = model;
|
req.body.model = model;
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
log("Error in router middleware:", error.message);
|
log("Error in router middleware:", error.message);
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { getServiceInfo } from './processCheck';
|
import { getServiceInfo } from './processCheck';
|
||||||
|
|
||||||
export function showStatus() {
|
export async function showStatus() {
|
||||||
const info = getServiceInfo();
|
const info = await getServiceInfo();
|
||||||
|
|
||||||
console.log('\n📊 Claude Code Router Status');
|
console.log('\n📊 Claude Code Router Status');
|
||||||
console.log('═'.repeat(40));
|
console.log('═'.repeat(40));
|
||||||
|
|||||||
33
ui/CLAUDE.md
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
# CLAUDE.md
|
||||||
|
|
||||||
|
This file provides guidance to Claude Code (claude.ai/code) when working with code in this repository.
|
||||||
|
|
||||||
|
## Project Overview
|
||||||
|
|
||||||
|
This is a frontend project for a configuration settings UI. The goal is to produce a single, self-contained HTML file with all JavaScript and CSS inlined. The application should be designed with a clean, modern UI and support both English and Chinese languages.
|
||||||
|
|
||||||
|
## Tech Stack
|
||||||
|
|
||||||
|
- **Package Manager:** pnpm
|
||||||
|
- **Build Tool:** Vite.js
|
||||||
|
- **Framework:** React.js
|
||||||
|
- **Styling:** Tailwind CSS with shadcn-ui
|
||||||
|
- **Languages:** TypeScript, English, Chinese
|
||||||
|
|
||||||
|
## Key Commands
|
||||||
|
|
||||||
|
- **Run development server:** `pnpm dev`
|
||||||
|
- **Build for production:** `pnpm build` (This produces a single HTML file)
|
||||||
|
- **Lint files:** `pnpm lint`
|
||||||
|
- **Preview production build:** `pnpm preview`
|
||||||
|
|
||||||
|
## Architecture & Development Notes
|
||||||
|
|
||||||
|
- **Configuration:** The application's configuration structure is defined in `config.example.json`. This file should be used as a reference for mocking data, as no backend APIs will be implemented.
|
||||||
|
- **Build Target:** The final build output must be a single HTML file. This is configured in `vite.config.ts` using `vite-plugin-singlefile`.
|
||||||
|
- **Internationalization (i18n):** The project uses `i18next` to support both English and Chinese. Locale files are located in `src/locales/`. When adding or changing text, ensure it is properly added to the translation files.
|
||||||
|
- **UI:** The UI is built with `shadcn-ui` components. Refer to existing components in `src/components/ui/` for styling conventions.
|
||||||
|
- **API Client:** The project uses a custom `ApiClient` class for handling HTTP requests with baseUrl and API key authentication. The class is defined in `src/lib/api.ts` and provides methods for GET, POST, PUT, and DELETE requests.
|
||||||
|
|
||||||
|
## 项目描述
|
||||||
|
参考`PROJECT.md`文件
|
||||||
23
ui/PROJECT.md
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# 项目指南
|
||||||
|
|
||||||
|
> 这是一个用于设置配置的前端项目,配置格式参考config.example.json
|
||||||
|
|
||||||
|
## 技术栈
|
||||||
|
1. 使用pnpm作为包管理工具
|
||||||
|
2. 使用vite.js作为构建工具
|
||||||
|
3. 使用react.js + tailwindcss + shadcn-ui构建前端界面
|
||||||
|
|
||||||
|
## UI设计
|
||||||
|
采用现代化的UI风格,让界面整体体现出呼吸感。整体配置应该简洁和通俗易懂,需要有必要的校验,易用的交互体验。
|
||||||
|
|
||||||
|
## 接口设计
|
||||||
|
不需要实现任何接口,但你需要根据config.example.json文件的内容mock数据
|
||||||
|
|
||||||
|
## 代码指引
|
||||||
|
在使用任何库之前你都需要使用websearch工具查找最新的文档,不要使用你知识库的内容,即使是显而易见的你以为的确定性的知识。
|
||||||
|
|
||||||
|
## 多语言设计
|
||||||
|
项目需要同时支持中文和英文
|
||||||
|
|
||||||
|
## 构建发布
|
||||||
|
最后需要构建出一个HTML文件,其中所有的js和css采用内联的方式,构建产物应该只包含一个html文件。
|
||||||
69
ui/README.md
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
# React + TypeScript + Vite
|
||||||
|
|
||||||
|
This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
|
||||||
|
|
||||||
|
Currently, two official plugins are available:
|
||||||
|
|
||||||
|
- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Babel](https://babeljs.io/) for Fast Refresh
|
||||||
|
- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/) for Fast Refresh
|
||||||
|
|
||||||
|
## Expanding the ESLint configuration
|
||||||
|
|
||||||
|
If you are developing a production application, we recommend updating the configuration to enable type-aware lint rules:
|
||||||
|
|
||||||
|
```js
|
||||||
|
export default tseslint.config([
|
||||||
|
globalIgnores(['dist']),
|
||||||
|
{
|
||||||
|
files: ['**/*.{ts,tsx}'],
|
||||||
|
extends: [
|
||||||
|
// Other configs...
|
||||||
|
|
||||||
|
// Remove tseslint.configs.recommended and replace with this
|
||||||
|
...tseslint.configs.recommendedTypeChecked,
|
||||||
|
// Alternatively, use this for stricter rules
|
||||||
|
...tseslint.configs.strictTypeChecked,
|
||||||
|
// Optionally, add this for stylistic rules
|
||||||
|
...tseslint.configs.stylisticTypeChecked,
|
||||||
|
|
||||||
|
// Other configs...
|
||||||
|
],
|
||||||
|
languageOptions: {
|
||||||
|
parserOptions: {
|
||||||
|
project: ['./tsconfig.node.json', './tsconfig.app.json'],
|
||||||
|
tsconfigRootDir: import.meta.dirname,
|
||||||
|
},
|
||||||
|
// other options...
|
||||||
|
},
|
||||||
|
},
|
||||||
|
])
|
||||||
|
```
|
||||||
|
|
||||||
|
You can also install [eslint-plugin-react-x](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-x) and [eslint-plugin-react-dom](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-dom) for React-specific lint rules:
|
||||||
|
|
||||||
|
```js
|
||||||
|
// eslint.config.js
|
||||||
|
import reactX from 'eslint-plugin-react-x'
|
||||||
|
import reactDom from 'eslint-plugin-react-dom'
|
||||||
|
|
||||||
|
export default tseslint.config([
|
||||||
|
globalIgnores(['dist']),
|
||||||
|
{
|
||||||
|
files: ['**/*.{ts,tsx}'],
|
||||||
|
extends: [
|
||||||
|
// Other configs...
|
||||||
|
// Enable lint rules for React
|
||||||
|
reactX.configs['recommended-typescript'],
|
||||||
|
// Enable lint rules for React DOM
|
||||||
|
reactDom.configs.recommended,
|
||||||
|
],
|
||||||
|
languageOptions: {
|
||||||
|
parserOptions: {
|
||||||
|
project: ['./tsconfig.node.json', './tsconfig.app.json'],
|
||||||
|
tsconfigRootDir: import.meta.dirname,
|
||||||
|
},
|
||||||
|
// other options...
|
||||||
|
},
|
||||||
|
},
|
||||||
|
])
|
||||||
|
```
|
||||||
21
ui/components.json
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://ui.shadcn.com/schema.json",
|
||||||
|
"style": "new-york",
|
||||||
|
"rsc": false,
|
||||||
|
"tsx": true,
|
||||||
|
"tailwind": {
|
||||||
|
"config": "",
|
||||||
|
"css": "src/index.css",
|
||||||
|
"baseColor": "neutral",
|
||||||
|
"cssVariables": true,
|
||||||
|
"prefix": ""
|
||||||
|
},
|
||||||
|
"aliases": {
|
||||||
|
"components": "@/components",
|
||||||
|
"utils": "@/lib/utils",
|
||||||
|
"ui": "@/components/ui",
|
||||||
|
"lib": "@/lib",
|
||||||
|
"hooks": "@/hooks"
|
||||||
|
},
|
||||||
|
"iconLibrary": "lucide"
|
||||||
|
}
|
||||||
177
ui/config.example.json
Normal file
@@ -0,0 +1,177 @@
|
|||||||
|
{
|
||||||
|
"LOG": true,
|
||||||
|
"CLAUDE_PATH": "/Users/jinhuilee/.claude/local/claude",
|
||||||
|
"HOST": "127.0.0.1",
|
||||||
|
"PORT": 8080,
|
||||||
|
"APIKEY": "1",
|
||||||
|
"transformers": [
|
||||||
|
{
|
||||||
|
"path": "/Users/abc/.claude-code-router/plugins/gemini-cli.js",
|
||||||
|
"options": {
|
||||||
|
"project": "x"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Providers": [
|
||||||
|
{
|
||||||
|
"name": "siliconflow",
|
||||||
|
"api_base_url": "https://api.moonshot.cn/v1/chat/completions",
|
||||||
|
"api_key": "sk-",
|
||||||
|
"models": [
|
||||||
|
"kimi-k2-0711-preview"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 130000
|
||||||
|
}
|
||||||
|
]
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "kimi",
|
||||||
|
"api_base_url": "https://api.moonshot.cn/v1/chat/completions",
|
||||||
|
"api_key": "sk-",
|
||||||
|
"models": [
|
||||||
|
"kimi-k2-0711-preview"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "groq",
|
||||||
|
"api_base_url": "https://api.groq.com/openai/v1/chat/completions",
|
||||||
|
"api_key": "",
|
||||||
|
"models": [
|
||||||
|
"moonshotai/kimi-k2-instruct"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
[
|
||||||
|
"maxtoken",
|
||||||
|
{
|
||||||
|
"max_tokens": 16384
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"groq"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "openrouter",
|
||||||
|
"api_base_url": "https://openrouter.ai/api/v1/chat/completions",
|
||||||
|
"api_key": "sk-or-v1-",
|
||||||
|
"models": [
|
||||||
|
"google/gemini-2.5-pro-preview",
|
||||||
|
"anthropic/claude-sonnet-4",
|
||||||
|
"anthropic/claude-3.5-sonnet",
|
||||||
|
"anthropic/claude-3.7-sonnet:thinking",
|
||||||
|
"deepseek/deepseek-chat-v3-0324",
|
||||||
|
"@preset/kimi"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
"openrouter"
|
||||||
|
],
|
||||||
|
"deepseek/deepseek-chat-v3-0324": {
|
||||||
|
"use": [
|
||||||
|
"tooluse"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "deepseek",
|
||||||
|
"api_base_url": "https://api.deepseek.com/chat/completions",
|
||||||
|
"api_key": "sk-",
|
||||||
|
"models": [
|
||||||
|
"deepseek-chat",
|
||||||
|
"deepseek-reasoner"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
"deepseek"
|
||||||
|
],
|
||||||
|
"deepseek-chat": {
|
||||||
|
"use": [
|
||||||
|
"tooluse"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "test",
|
||||||
|
"api_base_url": "https://tbai.xin/v1/chat/completions",
|
||||||
|
"api_key": "sk-",
|
||||||
|
"models": [
|
||||||
|
"gemini-2.5-pro"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "ollama",
|
||||||
|
"api_base_url": "http://localhost:11434/v1/chat/completions",
|
||||||
|
"api_key": "ollama",
|
||||||
|
"models": [
|
||||||
|
"qwen2.5-coder:latest"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "gemini",
|
||||||
|
"api_base_url": "https://generativelanguage.googleapis.com/v1beta/models/",
|
||||||
|
"api_key": "",
|
||||||
|
"models": [
|
||||||
|
"gemini-2.5-flash",
|
||||||
|
"gemini-2.5-pro"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
"gemini"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "volcengine",
|
||||||
|
"api_base_url": "https://ark.cn-beijing.volces.com/api/v3/chat/completions",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": [
|
||||||
|
"deepseek-v3-250324",
|
||||||
|
"deepseek-r1-250528"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
"deepseek"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "gemini-cli",
|
||||||
|
"api_base_url": "https://cloudcode-pa.googleapis.com/v1internal",
|
||||||
|
"api_key": "sk-xxx",
|
||||||
|
"models": [
|
||||||
|
"gemini-2.5-flash",
|
||||||
|
"gemini-2.5-pro"
|
||||||
|
],
|
||||||
|
"transformer": {
|
||||||
|
"use": [
|
||||||
|
"gemini-cli"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "azure",
|
||||||
|
"api_base_url": "https://your-resource-name.openai.azure.com/",
|
||||||
|
"api_key": "",
|
||||||
|
"models": [
|
||||||
|
"gpt-4"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Router": {
|
||||||
|
"default": "gemini-cli,gemini-2.5-pro",
|
||||||
|
"background": "gemini-cli,gemini-2.5-flash",
|
||||||
|
"think": "gemini-cli,gemini-2.5-pro",
|
||||||
|
"longContext": "gemini-cli,gemini-2.5-pro",
|
||||||
|
"webSearch": "gemini-cli,gemini-2.5-flash"
|
||||||
|
}
|
||||||
|
}
|
||||||
23
ui/eslint.config.js
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import js from '@eslint/js'
|
||||||
|
import globals from 'globals'
|
||||||
|
import reactHooks from 'eslint-plugin-react-hooks'
|
||||||
|
import reactRefresh from 'eslint-plugin-react-refresh'
|
||||||
|
import tseslint from 'typescript-eslint'
|
||||||
|
import { globalIgnores } from 'eslint/config'
|
||||||
|
|
||||||
|
export default tseslint.config([
|
||||||
|
globalIgnores(['dist']),
|
||||||
|
{
|
||||||
|
files: ['**/*.{ts,tsx}'],
|
||||||
|
extends: [
|
||||||
|
js.configs.recommended,
|
||||||
|
tseslint.configs.recommended,
|
||||||
|
reactHooks.configs['recommended-latest'],
|
||||||
|
reactRefresh.configs.vite,
|
||||||
|
],
|
||||||
|
languageOptions: {
|
||||||
|
ecmaVersion: 2020,
|
||||||
|
globals: globals.browser,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
])
|
||||||
13
ui/index.html
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8" />
|
||||||
|
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
|
<title>CCR UI</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="root"></div>
|
||||||
|
<script type="module" src="/src/main.tsx"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
5033
ui/package-lock.json
generated
Normal file
53
ui/package.json
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
{
|
||||||
|
"name": "temp-project",
|
||||||
|
"private": true,
|
||||||
|
"version": "0.0.0",
|
||||||
|
"type": "module",
|
||||||
|
"scripts": {
|
||||||
|
"dev": "vite",
|
||||||
|
"build": "tsc -b && vite build",
|
||||||
|
"lint": "eslint .",
|
||||||
|
"preview": "vite preview"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@monaco-editor/react": "^4.7.0",
|
||||||
|
"@radix-ui/react-dialog": "^1.1.14",
|
||||||
|
"@radix-ui/react-label": "^2.1.7",
|
||||||
|
"@radix-ui/react-popover": "^1.1.14",
|
||||||
|
"@radix-ui/react-slot": "^1.2.3",
|
||||||
|
"@radix-ui/react-switch": "^1.2.5",
|
||||||
|
"@tailwindcss/vite": "^4.1.11",
|
||||||
|
"class-variance-authority": "^0.7.1",
|
||||||
|
"clsx": "^2.1.1",
|
||||||
|
"cmdk": "^1.1.1",
|
||||||
|
"i18next": "^25.3.2",
|
||||||
|
"i18next-browser-languagedetector": "^8.2.0",
|
||||||
|
"lucide-react": "^0.525.0",
|
||||||
|
"react": "^19.1.0",
|
||||||
|
"react-dom": "^19.1.0",
|
||||||
|
"react-i18next": "^15.6.1",
|
||||||
|
"react-router-dom": "^7.7.0",
|
||||||
|
"tailwind-merge": "^3.3.1",
|
||||||
|
"tailwindcss-animate": "^1.0.7"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@eslint/js": "^9.30.1",
|
||||||
|
"@tailwindcss/postcss": "^4.1.11",
|
||||||
|
"@types/node": "^24.1.0",
|
||||||
|
"@types/react": "^19.1.8",
|
||||||
|
"@types/react-dom": "^19.1.6",
|
||||||
|
"@vitejs/plugin-react": "^4.6.0",
|
||||||
|
"autoprefixer": "^10.4.21",
|
||||||
|
"eslint": "^9.30.1",
|
||||||
|
"eslint-plugin-react-hooks": "^5.2.0",
|
||||||
|
"eslint-plugin-react-refresh": "^0.4.20",
|
||||||
|
"globals": "^16.3.0",
|
||||||
|
"postcss": "^8.5.6",
|
||||||
|
"tailwindcss": "^4.1.11",
|
||||||
|
"tw-animate-css": "^1.3.5",
|
||||||
|
"typescript": "~5.8.3",
|
||||||
|
"typescript-eslint": "^8.35.1",
|
||||||
|
"vite": "^7.0.4",
|
||||||
|
"vite-plugin-singlefile": "^2.3.0"
|
||||||
|
}
|
||||||
|
}
|
||||||
3459
ui/pnpm-lock.yaml
generated
Normal file
1
ui/public/vite.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="31.88" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 257"><defs><linearGradient id="IconifyId1813088fe1fbc01fb466" x1="-.828%" x2="57.636%" y1="7.652%" y2="78.411%"><stop offset="0%" stop-color="#41D1FF"></stop><stop offset="100%" stop-color="#BD34FE"></stop></linearGradient><linearGradient id="IconifyId1813088fe1fbc01fb467" x1="43.376%" x2="50.316%" y1="2.242%" y2="89.03%"><stop offset="0%" stop-color="#FFEA83"></stop><stop offset="8.333%" stop-color="#FFDD35"></stop><stop offset="100%" stop-color="#FFA800"></stop></linearGradient></defs><path fill="url(#IconifyId1813088fe1fbc01fb466)" d="M255.153 37.938L134.897 252.976c-2.483 4.44-8.862 4.466-11.382.048L.875 37.958c-2.746-4.814 1.371-10.646 6.827-9.67l120.385 21.517a6.537 6.537 0 0 0 2.322-.004l117.867-21.483c5.438-.991 9.574 4.796 6.877 9.62Z"></path><path fill="url(#IconifyId1813088fe1fbc01fb467)" d="M185.432.063L96.44 17.501a3.268 3.268 0 0 0-2.634 3.014l-5.474 92.456a3.268 3.268 0 0 0 3.997 3.378l24.777-5.718c2.318-.535 4.413 1.507 3.936 3.838l-7.361 36.047c-.495 2.426 1.782 4.5 4.151 3.78l15.304-4.649c2.372-.72 4.652 1.36 4.15 3.788l-11.698 56.621c-.732 3.542 3.979 5.473 5.943 2.437l1.313-2.028l72.516-144.72c1.215-2.423-.88-5.186-3.54-4.672l-25.505 4.922c-2.396.462-4.435-1.77-3.759-4.114l16.646-57.705c.677-2.35-1.37-4.583-3.769-4.113Z"></path></svg>
|
||||||
|
After Width: | Height: | Size: 1.5 KiB |
237
ui/src/App.tsx
Normal file
@@ -0,0 +1,237 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { useNavigate } from "react-router-dom";
|
||||||
|
import { SettingsDialog } from "@/components/SettingsDialog";
|
||||||
|
import { Transformers } from "@/components/Transformers";
|
||||||
|
import { Providers } from "@/components/Providers";
|
||||||
|
import { Router } from "@/components/Router";
|
||||||
|
import { JsonEditor } from "@/components/JsonEditor";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { useConfig } from "@/components/ConfigProvider";
|
||||||
|
import { api } from "@/lib/api";
|
||||||
|
import { Settings, Languages, Save, RefreshCw, FileJson } from "lucide-react";
|
||||||
|
import {
|
||||||
|
Popover,
|
||||||
|
PopoverContent,
|
||||||
|
PopoverTrigger,
|
||||||
|
} from "@/components/ui/popover";
|
||||||
|
import { Toast } from "@/components/ui/toast";
|
||||||
|
import "@/styles/animations.css";
|
||||||
|
|
||||||
|
function App() {
|
||||||
|
const { t, i18n } = useTranslation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const { config, error } = useConfig();
|
||||||
|
const [isSettingsOpen, setIsSettingsOpen] = useState(false);
|
||||||
|
const [isJsonEditorOpen, setIsJsonEditorOpen] = useState(false);
|
||||||
|
const [isCheckingAuth, setIsCheckingAuth] = useState(true);
|
||||||
|
const [toast, setToast] = useState<{ message: string; type: 'success' | 'error' | 'warning' } | null>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const checkAuth = async () => {
|
||||||
|
// If we already have a config, we're authenticated
|
||||||
|
if (config) {
|
||||||
|
setIsCheckingAuth(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// For empty API key, allow access without checking config
|
||||||
|
const apiKey = localStorage.getItem('apiKey');
|
||||||
|
if (!apiKey) {
|
||||||
|
setIsCheckingAuth(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If we don't have a config, try to fetch it
|
||||||
|
try {
|
||||||
|
await api.getConfig();
|
||||||
|
// If successful, we don't need to do anything special
|
||||||
|
// The ConfigProvider will handle setting the config
|
||||||
|
} catch (err) {
|
||||||
|
// If it's a 401, the API client will redirect to login
|
||||||
|
// For other errors, we still show the app to display the error
|
||||||
|
console.error('Error checking auth:', err);
|
||||||
|
// Redirect to login on authentication error
|
||||||
|
if ((err as Error).message === 'Unauthorized') {
|
||||||
|
navigate('/login');
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsCheckingAuth(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
checkAuth();
|
||||||
|
|
||||||
|
// Listen for unauthorized events
|
||||||
|
const handleUnauthorized = () => {
|
||||||
|
navigate('/login');
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('unauthorized', handleUnauthorized);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('unauthorized', handleUnauthorized);
|
||||||
|
};
|
||||||
|
}, [config, navigate]);
|
||||||
|
|
||||||
|
const saveConfig = async () => {
|
||||||
|
if (config) {
|
||||||
|
try {
|
||||||
|
// Save to API
|
||||||
|
const response = await api.updateConfig(config);
|
||||||
|
// Show success message or handle as needed
|
||||||
|
console.log('Config saved successfully');
|
||||||
|
|
||||||
|
// 根据响应信息进行提示
|
||||||
|
if (response && typeof response === 'object' && 'success' in response) {
|
||||||
|
const apiResponse = response as { success: boolean; message?: string };
|
||||||
|
if (apiResponse.success) {
|
||||||
|
setToast({ message: apiResponse.message || t('app.config_saved_success'), type: 'success' });
|
||||||
|
} else {
|
||||||
|
setToast({ message: apiResponse.message || t('app.config_saved_failed'), type: 'error' });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 默认成功提示
|
||||||
|
setToast({ message: t('app.config_saved_success'), type: 'success' });
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to save config:', error);
|
||||||
|
// Handle error appropriately
|
||||||
|
setToast({ message: t('app.config_saved_failed') + ': ' + (error as Error).message, type: 'error' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const saveConfigAndRestart = async () => {
|
||||||
|
if (config) {
|
||||||
|
try {
|
||||||
|
// Save to API
|
||||||
|
const response = await api.updateConfig(config);
|
||||||
|
|
||||||
|
// Check if save was successful before restarting
|
||||||
|
let saveSuccessful = true;
|
||||||
|
if (response && typeof response === 'object' && 'success' in response) {
|
||||||
|
const apiResponse = response as { success: boolean; message?: string };
|
||||||
|
if (!apiResponse.success) {
|
||||||
|
saveSuccessful = false;
|
||||||
|
setToast({ message: apiResponse.message || t('app.config_saved_failed'), type: 'error' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Only restart if save was successful
|
||||||
|
if (saveSuccessful) {
|
||||||
|
// Restart service
|
||||||
|
const response = await api.restartService();
|
||||||
|
|
||||||
|
// Show success message or handle as needed
|
||||||
|
console.log('Config saved and service restarted successfully');
|
||||||
|
|
||||||
|
// 根据响应信息进行提示
|
||||||
|
if (response && typeof response === 'object' && 'success' in response) {
|
||||||
|
const apiResponse = response as { success: boolean; message?: string };
|
||||||
|
if (apiResponse.success) {
|
||||||
|
setToast({ message: apiResponse.message || t('app.config_saved_restart_success'), type: 'success' });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 默认成功提示
|
||||||
|
setToast({ message: t('app.config_saved_restart_success'), type: 'success' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to save config and restart:', error);
|
||||||
|
// Handle error appropriately
|
||||||
|
setToast({ message: t('app.config_saved_restart_failed') + ': ' + (error as Error).message, type: 'error' });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
if (isCheckingAuth) {
|
||||||
|
return <div>Loading...</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (error) {
|
||||||
|
return <div>Error: {error.message}</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
return <div>Loading...</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-screen bg-gray-50 font-sans">
|
||||||
|
<header className="flex h-16 items-center justify-between border-b bg-white px-6">
|
||||||
|
<h1 className="text-xl font-semibold text-gray-800">{t('app.title')}</h1>
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Button variant="ghost" size="icon" onClick={() => setIsSettingsOpen(true)} className="transition-all-ease hover:scale-110">
|
||||||
|
<Settings className="h-5 w-5" />
|
||||||
|
</Button>
|
||||||
|
<Button variant="ghost" size="icon" onClick={() => setIsJsonEditorOpen(true)} className="transition-all-ease hover:scale-110">
|
||||||
|
<FileJson className="h-5 w-5" />
|
||||||
|
</Button>
|
||||||
|
<Popover>
|
||||||
|
<PopoverTrigger asChild>
|
||||||
|
<Button variant="ghost" size="icon" className="transition-all-ease hover:scale-110">
|
||||||
|
<Languages className="h-5 w-5" />
|
||||||
|
</Button>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent className="w-32 p-2">
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Button
|
||||||
|
variant={i18n.language.startsWith('en') ? 'default' : 'ghost'}
|
||||||
|
className="w-full justify-start transition-all-ease hover:scale-[1.02]"
|
||||||
|
onClick={() => i18n.changeLanguage('en')}
|
||||||
|
>
|
||||||
|
English
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant={i18n.language.startsWith('zh') ? 'default' : 'ghost'}
|
||||||
|
className="w-full justify-start transition-all-ease hover:scale-[1.02]"
|
||||||
|
onClick={() => i18n.changeLanguage('zh')}
|
||||||
|
>
|
||||||
|
中文
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
<Button onClick={saveConfig} variant="outline" className="transition-all-ease hover:scale-[1.02] active:scale-[0.98]">
|
||||||
|
<Save className="mr-2 h-4 w-4" />
|
||||||
|
{t('app.save')}
|
||||||
|
</Button>
|
||||||
|
<Button onClick={saveConfigAndRestart} className="transition-all-ease hover:scale-[1.02] active:scale-[0.98]">
|
||||||
|
<RefreshCw className="mr-2 h-4 w-4" />
|
||||||
|
{t('app.save_and_restart')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
<main className="flex h-[calc(100vh-4rem)] gap-4 p-4">
|
||||||
|
<div className="w-3/5">
|
||||||
|
<Providers />
|
||||||
|
</div>
|
||||||
|
<div className="flex w-2/5 flex-col gap-4">
|
||||||
|
<div className="h-3/5">
|
||||||
|
<Router />
|
||||||
|
</div>
|
||||||
|
<div className="flex-1">
|
||||||
|
<Transformers />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
<SettingsDialog isOpen={isSettingsOpen} onOpenChange={setIsSettingsOpen} />
|
||||||
|
<JsonEditor
|
||||||
|
open={isJsonEditorOpen}
|
||||||
|
onOpenChange={setIsJsonEditorOpen}
|
||||||
|
showToast={(message, type) => setToast({ message, type })}
|
||||||
|
/>
|
||||||
|
{toast && (
|
||||||
|
<Toast
|
||||||
|
message={toast.message}
|
||||||
|
type={toast.type}
|
||||||
|
onClose={() => setToast(null)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default App;
|
||||||
1
ui/src/assets/react.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" aria-hidden="true" role="img" class="iconify iconify--logos" width="35.93" height="32" preserveAspectRatio="xMidYMid meet" viewBox="0 0 256 228"><path fill="#00D8FF" d="M210.483 73.824a171.49 171.49 0 0 0-8.24-2.597c.465-1.9.893-3.777 1.273-5.621c6.238-30.281 2.16-54.676-11.769-62.708c-13.355-7.7-35.196.329-57.254 19.526a171.23 171.23 0 0 0-6.375 5.848a155.866 155.866 0 0 0-4.241-3.917C100.759 3.829 77.587-4.822 63.673 3.233C50.33 10.957 46.379 33.89 51.995 62.588a170.974 170.974 0 0 0 1.892 8.48c-3.28.932-6.445 1.924-9.474 2.98C17.309 83.498 0 98.307 0 113.668c0 15.865 18.582 31.778 46.812 41.427a145.52 145.52 0 0 0 6.921 2.165a167.467 167.467 0 0 0-2.01 9.138c-5.354 28.2-1.173 50.591 12.134 58.266c13.744 7.926 36.812-.22 59.273-19.855a145.567 145.567 0 0 0 5.342-4.923a168.064 168.064 0 0 0 6.92 6.314c21.758 18.722 43.246 26.282 56.54 18.586c13.731-7.949 18.194-32.003 12.4-61.268a145.016 145.016 0 0 0-1.535-6.842c1.62-.48 3.21-.974 4.76-1.488c29.348-9.723 48.443-25.443 48.443-41.52c0-15.417-17.868-30.326-45.517-39.844Zm-6.365 70.984c-1.4.463-2.836.91-4.3 1.345c-3.24-10.257-7.612-21.163-12.963-32.432c5.106-11 9.31-21.767 12.459-31.957c2.619.758 5.16 1.557 7.61 2.4c23.69 8.156 38.14 20.213 38.14 29.504c0 9.896-15.606 22.743-40.946 31.14Zm-10.514 20.834c2.562 12.94 2.927 24.64 1.23 33.787c-1.524 8.219-4.59 13.698-8.382 15.893c-8.067 4.67-25.32-1.4-43.927-17.412a156.726 156.726 0 0 1-6.437-5.87c7.214-7.889 14.423-17.06 21.459-27.246c12.376-1.098 24.068-2.894 34.671-5.345a134.17 134.17 0 0 1 1.386 6.193ZM87.276 214.515c-7.882 2.783-14.16 2.863-17.955.675c-8.075-4.657-11.432-22.636-6.853-46.752a156.923 156.923 0 0 1 1.869-8.499c10.486 2.32 22.093 3.988 34.498 4.994c7.084 9.967 14.501 19.128 21.976 27.15a134.668 134.668 0 0 1-4.877 4.492c-9.933 8.682-19.886 14.842-28.658 17.94ZM50.35 144.747c-12.483-4.267-22.792-9.812-29.858-15.863c-6.35-5.437-9.555-10.836-9.555-15.216c0-9.322 13.897-21.212 37.076-29.293c2.813-.98 5.757-1.905 8.812-2.773c3.204 10.42 7.406 21.315 12.477 32.332c-5.137 11.18-9.399 22.249-12.634 32.792a134.718 134.718 0 0 1-6.318-1.979Zm12.378-84.26c-4.811-24.587-1.616-43.134 6.425-47.789c8.564-4.958 27.502 2.111 47.463 19.835a144.318 144.318 0 0 1 3.841 3.545c-7.438 7.987-14.787 17.08-21.808 26.988c-12.04 1.116-23.565 2.908-34.161 5.309a160.342 160.342 0 0 1-1.76-7.887Zm110.427 27.268a347.8 347.8 0 0 0-7.785-12.803c8.168 1.033 15.994 2.404 23.343 4.08c-2.206 7.072-4.956 14.465-8.193 22.045a381.151 381.151 0 0 0-7.365-13.322Zm-45.032-43.861c5.044 5.465 10.096 11.566 15.065 18.186a322.04 322.04 0 0 0-30.257-.006c4.974-6.559 10.069-12.652 15.192-18.18ZM82.802 87.83a323.167 323.167 0 0 0-7.227 13.238c-3.184-7.553-5.909-14.98-8.134-22.152c7.304-1.634 15.093-2.97 23.209-3.984a321.524 321.524 0 0 0-7.848 12.897Zm8.081 65.352c-8.385-.936-16.291-2.203-23.593-3.793c2.26-7.3 5.045-14.885 8.298-22.6a321.187 321.187 0 0 0 7.257 13.246c2.594 4.48 5.28 8.868 8.038 13.147Zm37.542 31.03c-5.184-5.592-10.354-11.779-15.403-18.433c4.902.192 9.899.29 14.978.29c5.218 0 10.376-.117 15.453-.343c-4.985 6.774-10.018 12.97-15.028 18.486Zm52.198-57.817c3.422 7.8 6.306 15.345 8.596 22.52c-7.422 1.694-15.436 3.058-23.88 4.071a382.417 382.417 0 0 0 7.859-13.026a347.403 347.403 0 0 0 7.425-13.565Zm-16.898 8.101a358.557 358.557 0 0 1-12.281 19.815a329.4 329.4 0 0 1-23.444.823c-7.967 0-15.716-.248-23.178-.732a310.202 310.202 0 0 1-12.513-19.846h.001a307.41 307.41 0 0 1-10.923-20.627a310.278 310.278 0 0 1 10.89-20.637l-.001.001a307.318 307.318 0 0 1 12.413-19.761c7.613-.576 15.42-.876 23.31-.876H128c7.926 0 15.743.303 23.354.883a329.357 329.357 0 0 1 12.335 19.695a358.489 358.489 0 0 1 11.036 20.54a329.472 329.472 0 0 1-11 20.722Zm22.56-122.124c8.572 4.944 11.906 24.881 6.52 51.026c-.344 1.668-.73 3.367-1.15 5.09c-10.622-2.452-22.155-4.275-34.23-5.408c-7.034-10.017-14.323-19.124-21.64-27.008a160.789 160.789 0 0 1 5.888-5.4c18.9-16.447 36.564-22.941 44.612-18.3ZM128 90.808c12.625 0 22.86 10.235 22.86 22.86s-10.235 22.86-22.86 22.86s-22.86-10.235-22.86-22.86s10.235-22.86 22.86-22.86Z"></path></svg>
|
||||||
|
After Width: | Height: | Size: 4.0 KiB |
147
ui/src/components/ConfigProvider.tsx
Normal file
@@ -0,0 +1,147 @@
|
|||||||
|
import { createContext, useContext, useState, useEffect } from 'react';
|
||||||
|
import type { ReactNode, Dispatch, SetStateAction } from 'react';
|
||||||
|
import { api } from '@/lib/api';
|
||||||
|
|
||||||
|
export interface Transformer {
|
||||||
|
path: string;
|
||||||
|
options: {
|
||||||
|
[key: string]: string;
|
||||||
|
};
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
[key: string]: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ProviderTransformer {
|
||||||
|
use: (string | (string | Record<string, unknown> | { max_tokens: number })[])[];
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
[key: string]: any; // for model specific transformers
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Provider {
|
||||||
|
name: string;
|
||||||
|
api_base_url: string;
|
||||||
|
api_key: string;
|
||||||
|
models: string[];
|
||||||
|
transformer?: ProviderTransformer;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
[key: string]: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RouterConfig {
|
||||||
|
default: string;
|
||||||
|
background: string;
|
||||||
|
think: string;
|
||||||
|
longContext: string;
|
||||||
|
webSearch: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Config {
|
||||||
|
LOG: boolean;
|
||||||
|
CLAUDE_PATH: string;
|
||||||
|
HOST: string;
|
||||||
|
PORT: number;
|
||||||
|
APIKEY: string;
|
||||||
|
transformers: Transformer[];
|
||||||
|
Providers: Provider[];
|
||||||
|
Router: RouterConfig;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ConfigContextType {
|
||||||
|
config: Config | null;
|
||||||
|
setConfig: Dispatch<SetStateAction<Config | null>>;
|
||||||
|
error: Error | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ConfigContext = createContext<ConfigContextType | undefined>(undefined);
|
||||||
|
|
||||||
|
// eslint-disable-next-line react-refresh/only-export-components
|
||||||
|
export function useConfig() {
|
||||||
|
const context = useContext(ConfigContext);
|
||||||
|
if (context === undefined) {
|
||||||
|
throw new Error('useConfig must be used within a ConfigProvider');
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ConfigProviderProps {
|
||||||
|
children: ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ConfigProvider({ children }: ConfigProviderProps) {
|
||||||
|
const [config, setConfig] = useState<Config | null>(null);
|
||||||
|
const [error, setError] = useState<Error | null>(null);
|
||||||
|
const [hasFetched, setHasFetched] = useState<boolean>(false);
|
||||||
|
const [apiKey, setApiKey] = useState<string | null>(localStorage.getItem('apiKey'));
|
||||||
|
|
||||||
|
// Listen for localStorage changes
|
||||||
|
useEffect(() => {
|
||||||
|
const handleStorageChange = () => {
|
||||||
|
setApiKey(localStorage.getItem('apiKey'));
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('storage', handleStorageChange);
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('storage', handleStorageChange);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchConfig = async () => {
|
||||||
|
// Reset fetch state when API key changes
|
||||||
|
setHasFetched(false);
|
||||||
|
setConfig(null);
|
||||||
|
setError(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
fetchConfig();
|
||||||
|
}, [apiKey]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchConfig = async () => {
|
||||||
|
// Prevent duplicate API calls in React StrictMode
|
||||||
|
// Skip if we've already fetched
|
||||||
|
if (hasFetched) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setHasFetched(true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Try to fetch config regardless of API key presence
|
||||||
|
const data = await api.getConfig();
|
||||||
|
setConfig(data);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to fetch config:', err);
|
||||||
|
// If we get a 401, the API client will redirect to login
|
||||||
|
// Otherwise, set an empty config or error
|
||||||
|
if ((err as Error).message !== 'Unauthorized') {
|
||||||
|
// Set default empty config when fetch fails
|
||||||
|
setConfig({
|
||||||
|
LOG: false,
|
||||||
|
CLAUDE_PATH: '',
|
||||||
|
HOST: '127.0.0.1',
|
||||||
|
PORT: 3456,
|
||||||
|
APIKEY: '',
|
||||||
|
transformers: [],
|
||||||
|
Providers: [],
|
||||||
|
Router: {
|
||||||
|
default: '',
|
||||||
|
background: '',
|
||||||
|
think: '',
|
||||||
|
longContext: '',
|
||||||
|
webSearch: ''
|
||||||
|
}
|
||||||
|
});
|
||||||
|
setError(err as Error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
fetchConfig();
|
||||||
|
}, [hasFetched, apiKey]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ConfigContext.Provider value={{ config, setConfig, error }}>
|
||||||
|
{children}
|
||||||
|
</ConfigContext.Provider>
|
||||||
|
);
|
||||||
|
}
|
||||||
220
ui/src/components/JsonEditor.tsx
Normal file
@@ -0,0 +1,220 @@
|
|||||||
|
import { useState, useEffect, useRef } from 'react';
|
||||||
|
import Editor from '@monaco-editor/react';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { useConfig } from '@/components/ConfigProvider';
|
||||||
|
import { api } from '@/lib/api';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import { Save, X, RefreshCw } from 'lucide-react';
|
||||||
|
|
||||||
|
interface JsonEditorProps {
|
||||||
|
open: boolean;
|
||||||
|
onOpenChange: (open: boolean) => void;
|
||||||
|
showToast?: (message: string, type: 'success' | 'error' | 'warning') => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function JsonEditor({ open, onOpenChange, showToast }: JsonEditorProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { config } = useConfig();
|
||||||
|
const [jsonValue, setJsonValue] = useState<string>('');
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
|
const [isVisible, setIsVisible] = useState(false);
|
||||||
|
const [isAnimating, setIsAnimating] = useState(false);
|
||||||
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (config && open) {
|
||||||
|
setJsonValue(JSON.stringify(config, null, 2));
|
||||||
|
}
|
||||||
|
}, [config, open]);
|
||||||
|
|
||||||
|
// Handle open/close animations
|
||||||
|
useEffect(() => {
|
||||||
|
if (open) {
|
||||||
|
setIsVisible(true);
|
||||||
|
// Trigger the animation after a small delay to ensure the element is rendered
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
setIsAnimating(true);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setIsAnimating(false);
|
||||||
|
// Wait for the animation to complete before hiding
|
||||||
|
const timer = setTimeout(() => {
|
||||||
|
setIsVisible(false);
|
||||||
|
}, 300);
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}
|
||||||
|
}, [open]);
|
||||||
|
|
||||||
|
const handleSaveResponse = (response: unknown, successMessage: string, errorMessage: string) => {
|
||||||
|
// 根据响应信息进行提示
|
||||||
|
if (response && typeof response === 'object' && 'success' in response) {
|
||||||
|
const apiResponse = response as { success: boolean; message?: string };
|
||||||
|
if (apiResponse.success) {
|
||||||
|
if (showToast) {
|
||||||
|
showToast(apiResponse.message || successMessage, 'success');
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
if (showToast) {
|
||||||
|
showToast(apiResponse.message || errorMessage, 'error');
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 默认成功提示
|
||||||
|
if (showToast) {
|
||||||
|
showToast(successMessage, 'success');
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSave = async () => {
|
||||||
|
if (!jsonValue) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
setIsSaving(true);
|
||||||
|
const parsedConfig = JSON.parse(jsonValue);
|
||||||
|
const response = await api.updateConfig(parsedConfig);
|
||||||
|
|
||||||
|
const success = handleSaveResponse(
|
||||||
|
response,
|
||||||
|
t('app.config_saved_success'),
|
||||||
|
t('app.config_saved_failed')
|
||||||
|
);
|
||||||
|
|
||||||
|
if (success) {
|
||||||
|
onOpenChange(false);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to save config:', error);
|
||||||
|
if (showToast) {
|
||||||
|
showToast(t('app.config_saved_failed') + ': ' + (error as Error).message, 'error');
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsSaving(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSaveAndRestart = async () => {
|
||||||
|
if (!jsonValue) return;
|
||||||
|
|
||||||
|
try {
|
||||||
|
setIsSaving(true);
|
||||||
|
const parsedConfig = JSON.parse(jsonValue);
|
||||||
|
|
||||||
|
// Save config first
|
||||||
|
const saveResponse = await api.updateConfig(parsedConfig);
|
||||||
|
const saveSuccessful = handleSaveResponse(
|
||||||
|
saveResponse,
|
||||||
|
t('app.config_saved_success'),
|
||||||
|
t('app.config_saved_failed')
|
||||||
|
);
|
||||||
|
|
||||||
|
// Only restart if save was successful
|
||||||
|
if (saveSuccessful) {
|
||||||
|
// Restart service
|
||||||
|
const restartResponse = await api.restartService();
|
||||||
|
|
||||||
|
handleSaveResponse(
|
||||||
|
restartResponse,
|
||||||
|
t('app.config_saved_restart_success'),
|
||||||
|
t('app.config_saved_restart_failed')
|
||||||
|
);
|
||||||
|
|
||||||
|
onOpenChange(false);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to save config and restart:', error);
|
||||||
|
if (showToast) {
|
||||||
|
showToast(t('app.config_saved_restart_failed') + ': ' + (error as Error).message, 'error');
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
setIsSaving(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!isVisible && !open) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{(isVisible || open) && (
|
||||||
|
<div
|
||||||
|
className={`fixed inset-0 z-50 transition-all duration-300 ease-out ${
|
||||||
|
isAnimating && open ? 'bg-black/50 opacity-100' : 'bg-black/0 opacity-0 pointer-events-none'
|
||||||
|
}`}
|
||||||
|
onClick={() => onOpenChange(false)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div
|
||||||
|
ref={containerRef}
|
||||||
|
className={`fixed bottom-0 left-0 right-0 z-50 flex flex-col bg-white shadow-2xl transition-all duration-300 ease-out transform ${
|
||||||
|
isAnimating && open ? 'translate-y-0' : 'translate-y-full'
|
||||||
|
}`}
|
||||||
|
style={{
|
||||||
|
height: '100vh',
|
||||||
|
maxHeight: '100vh'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex items-center justify-between border-b p-4">
|
||||||
|
<h2 className="text-lg font-semibold">{t('json_editor.title')}</h2>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => onOpenChange(false)}
|
||||||
|
disabled={isSaving}
|
||||||
|
>
|
||||||
|
<X className="h-4 w-4 mr-2" />
|
||||||
|
{t('json_editor.cancel')}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={handleSave}
|
||||||
|
disabled={isSaving}
|
||||||
|
>
|
||||||
|
<Save className="h-4 w-4 mr-2" />
|
||||||
|
{isSaving ? t('json_editor.saving') : t('json_editor.save')}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="default"
|
||||||
|
size="sm"
|
||||||
|
onClick={handleSaveAndRestart}
|
||||||
|
disabled={isSaving}
|
||||||
|
>
|
||||||
|
<RefreshCw className="h-4 w-4 mr-2" />
|
||||||
|
{isSaving ? t('json_editor.saving') : t('json_editor.save_and_restart')}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex-1 min-h-0 bg-gray-50">
|
||||||
|
<Editor
|
||||||
|
height="100%"
|
||||||
|
defaultLanguage="json"
|
||||||
|
value={jsonValue}
|
||||||
|
onChange={(value) => setJsonValue(value || '')}
|
||||||
|
theme="vs"
|
||||||
|
options={{
|
||||||
|
minimap: { enabled: true },
|
||||||
|
fontSize: 14,
|
||||||
|
scrollBeyondLastLine: false,
|
||||||
|
automaticLayout: true,
|
||||||
|
wordWrap: 'on',
|
||||||
|
formatOnPaste: true,
|
||||||
|
formatOnType: true,
|
||||||
|
suggest: {
|
||||||
|
showKeywords: true,
|
||||||
|
showSnippets: true,
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
129
ui/src/components/Login.tsx
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
import { useState, useEffect } from 'react';
|
||||||
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
import { useTranslation } from 'react-i18next';
|
||||||
|
import { Button } from '@/components/ui/button';
|
||||||
|
import { Input } from '@/components/ui/input';
|
||||||
|
import { Label } from '@/components/ui/label';
|
||||||
|
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@/components/ui/card';
|
||||||
|
import { api } from '@/lib/api';
|
||||||
|
|
||||||
|
export function Login() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const [apiKey, setApiKey] = useState('');
|
||||||
|
const [error, setError] = useState('');
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
|
||||||
|
// Check if user is already authenticated
|
||||||
|
useEffect(() => {
|
||||||
|
const checkAuth = async () => {
|
||||||
|
const apiKey = localStorage.getItem('apiKey');
|
||||||
|
if (apiKey) {
|
||||||
|
setIsLoading(true);
|
||||||
|
// Verify the API key is still valid
|
||||||
|
try {
|
||||||
|
await api.getConfig();
|
||||||
|
navigate('/dashboard');
|
||||||
|
} catch (err) {
|
||||||
|
// If verification fails, remove the API key
|
||||||
|
localStorage.removeItem('apiKey');
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
checkAuth();
|
||||||
|
|
||||||
|
// Listen for unauthorized events
|
||||||
|
const handleUnauthorized = () => {
|
||||||
|
navigate('/login');
|
||||||
|
};
|
||||||
|
|
||||||
|
window.addEventListener('unauthorized', handleUnauthorized);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('unauthorized', handleUnauthorized);
|
||||||
|
};
|
||||||
|
}, [navigate]);
|
||||||
|
|
||||||
|
const handleLogin = async (e: React.FormEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Set the API key
|
||||||
|
api.setApiKey(apiKey);
|
||||||
|
|
||||||
|
// Dispatch storage event to notify other components of the change
|
||||||
|
window.dispatchEvent(new StorageEvent('storage', {
|
||||||
|
key: 'apiKey',
|
||||||
|
newValue: apiKey,
|
||||||
|
url: window.location.href
|
||||||
|
}));
|
||||||
|
|
||||||
|
// Test the API key by fetching config (skip if apiKey is empty)
|
||||||
|
if (apiKey) {
|
||||||
|
await api.getConfig();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Navigate to dashboard
|
||||||
|
// The ConfigProvider will handle fetching the config
|
||||||
|
navigate('/dashboard');
|
||||||
|
} catch (err) {
|
||||||
|
// Clear the API key on failure
|
||||||
|
api.setApiKey('');
|
||||||
|
setError(t('login.invalidApiKey'));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if (isLoading) {
|
||||||
|
return (
|
||||||
|
<div className="flex min-h-screen items-center justify-center bg-gray-50">
|
||||||
|
<Card className="w-full max-w-md">
|
||||||
|
<CardHeader className="space-y-1">
|
||||||
|
<CardTitle className="text-2xl">{t('login.title')}</CardTitle>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent>
|
||||||
|
<div className="flex justify-center py-8">
|
||||||
|
<div className="h-8 w-8 animate-spin rounded-full border-4 border-primary border-t-transparent"></div>
|
||||||
|
</div>
|
||||||
|
<p className="text-center text-sm text-gray-500">{t('login.validating')}</p>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex min-h-screen items-center justify-center bg-gray-50">
|
||||||
|
<Card className="w-full max-w-md">
|
||||||
|
<CardHeader className="space-y-1">
|
||||||
|
<CardTitle className="text-2xl">{t('login.title')}</CardTitle>
|
||||||
|
<CardDescription>
|
||||||
|
{t('login.description')}
|
||||||
|
</CardDescription>
|
||||||
|
</CardHeader>
|
||||||
|
<form onSubmit={handleLogin}>
|
||||||
|
<CardContent className="space-y-4">
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="apiKey">{t('login.apiKey')}</Label>
|
||||||
|
<Input
|
||||||
|
id="apiKey"
|
||||||
|
type="password"
|
||||||
|
value={apiKey}
|
||||||
|
onChange={(e) => setApiKey(e.target.value)}
|
||||||
|
placeholder={t('login.apiKeyPlaceholder')}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{error && <div className="text-sm text-red-500">{error}</div>}
|
||||||
|
</CardContent>
|
||||||
|
<CardFooter>
|
||||||
|
<Button className="w-full" type="submit">
|
||||||
|
{t('login.signIn')}
|
||||||
|
</Button>
|
||||||
|
</CardFooter>
|
||||||
|
</form>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
38
ui/src/components/ProviderList.tsx
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
import { Pencil, Trash2 } from "lucide-react";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { Badge } from "@/components/ui/badge";
|
||||||
|
import { type Provider } from "./ConfigProvider";
|
||||||
|
|
||||||
|
interface ProviderListProps {
|
||||||
|
providers: Provider[];
|
||||||
|
onEdit: (index: number) => void;
|
||||||
|
onRemove: (index: number) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ProviderList({ providers, onEdit, onRemove }: ProviderListProps) {
|
||||||
|
return (
|
||||||
|
<div className="space-y-3">
|
||||||
|
{providers.map((provider, index) => (
|
||||||
|
<div key={index} className="flex items-start justify-between rounded-md border bg-white p-4 transition-all hover:shadow-md animate-slide-in hover:scale-[1.01]">
|
||||||
|
<div className="flex-1 space-y-1.5">
|
||||||
|
<p className="text-md font-semibold text-gray-800">{provider.name}</p>
|
||||||
|
<p className="text-sm text-gray-500">{provider.api_base_url}</p>
|
||||||
|
<div className="flex flex-wrap gap-2 pt-2">
|
||||||
|
{provider.models.map((model) => (
|
||||||
|
<Badge key={model} variant="outline" className="font-normal transition-all-ease hover:scale-105">{model}</Badge>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="ml-4 flex flex-shrink-0 items-center gap-2">
|
||||||
|
<Button variant="ghost" size="icon" onClick={() => onEdit(index)} className="transition-all-ease hover:scale-110">
|
||||||
|
<Pencil className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
<Button variant="destructive" size="icon" onClick={() => onRemove(index)} className="transition-all duration-200 hover:scale-110">
|
||||||
|
<Trash2 className="h-4 w-4 text-current transition-colors duration-200" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
791
ui/src/components/Providers.tsx
Normal file
@@ -0,0 +1,791 @@
|
|||||||
|
import { useState, useRef, useEffect } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { useConfig } from "./ConfigProvider";
|
||||||
|
import { ProviderList } from "./ProviderList";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
import { X, Trash2, Plus } from "lucide-react";
|
||||||
|
import { Badge } from "@/components/ui/badge";
|
||||||
|
import { Combobox } from "@/components/ui/combobox";
|
||||||
|
import { ComboInput } from "@/components/ui/combo-input";
|
||||||
|
import { api } from "@/lib/api";
|
||||||
|
|
||||||
|
|
||||||
|
export function Providers() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
const [editingProviderIndex, setEditingProviderIndex] = useState<number | null>(null);
|
||||||
|
const [deletingProviderIndex, setDeletingProviderIndex] = useState<number | null>(null);
|
||||||
|
const [hasFetchedModels, setHasFetchedModels] = useState<Record<number, boolean>>({});
|
||||||
|
const [providerParamInputs, setProviderParamInputs] = useState<Record<string, {name: string, value: string}>>({});
|
||||||
|
const [modelParamInputs, setModelParamInputs] = useState<Record<string, {name: string, value: string}>>({});
|
||||||
|
const [availableTransformers, setAvailableTransformers] = useState<{name: string; endpoint: string | null;}[]>([]);
|
||||||
|
const comboInputRef = useRef<HTMLInputElement>(null);
|
||||||
|
|
||||||
|
// Fetch available transformers when component mounts
|
||||||
|
useEffect(() => {
|
||||||
|
const fetchTransformers = async () => {
|
||||||
|
try {
|
||||||
|
const response = await api.get<{transformers: {name: string; endpoint: string | null;}[]}>('/transformers');
|
||||||
|
setAvailableTransformers(response.transformers);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to fetch transformers:', error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
fetchTransformers();
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const handleAddProvider = () => {
|
||||||
|
const newProviders = [...config.Providers, { name: "", api_base_url: "", api_key: "", models: [] }];
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
setEditingProviderIndex(newProviders.length - 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSaveProvider = () => {
|
||||||
|
setEditingProviderIndex(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleCancelAddProvider = () => {
|
||||||
|
// If we're adding a new provider, remove it regardless of content
|
||||||
|
if (editingProviderIndex !== null && editingProviderIndex === config.Providers.length - 1) {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
newProviders.pop();
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
}
|
||||||
|
// Reset fetched models state for this provider
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
setHasFetchedModels(prev => {
|
||||||
|
const newState = { ...prev };
|
||||||
|
delete newState[editingProviderIndex];
|
||||||
|
return newState;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
setEditingProviderIndex(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRemoveProvider = (index: number) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
newProviders.splice(index, 1);
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
setDeletingProviderIndex(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleProviderChange = (index: number, field: string, value: string) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
newProviders[index][field] = value;
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleProviderTransformerChange = (index: number, transformerPath: string) => {
|
||||||
|
if (!transformerPath) return; // Don't add empty transformers
|
||||||
|
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (!newProviders[index].transformer) {
|
||||||
|
newProviders[index].transformer = { use: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add transformer to the use array
|
||||||
|
newProviders[index].transformer!.use = [...newProviders[index].transformer!.use, transformerPath];
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
const removeProviderTransformerAtIndex = (index: number, transformerIndex: number) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (newProviders[index].transformer) {
|
||||||
|
const newUseArray = [...newProviders[index].transformer!.use];
|
||||||
|
newUseArray.splice(transformerIndex, 1);
|
||||||
|
newProviders[index].transformer!.use = newUseArray;
|
||||||
|
|
||||||
|
// If use array is now empty and no other properties, remove transformer entirely
|
||||||
|
if (newUseArray.length === 0 && Object.keys(newProviders[index].transformer!).length === 1) {
|
||||||
|
delete newProviders[index].transformer;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleModelTransformerChange = (providerIndex: number, model: string, transformerPath: string) => {
|
||||||
|
if (!transformerPath) return; // Don't add empty transformers
|
||||||
|
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (!newProviders[providerIndex].transformer) {
|
||||||
|
newProviders[providerIndex].transformer = { use: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize model transformer if it doesn't exist
|
||||||
|
if (!newProviders[providerIndex].transformer![model]) {
|
||||||
|
newProviders[providerIndex].transformer![model] = { use: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add transformer to the use array
|
||||||
|
newProviders[providerIndex].transformer![model].use = [...newProviders[providerIndex].transformer![model].use, transformerPath];
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
const removeModelTransformerAtIndex = (providerIndex: number, model: string, transformerIndex: number) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (newProviders[providerIndex].transformer && newProviders[providerIndex].transformer![model]) {
|
||||||
|
const newUseArray = [...newProviders[providerIndex].transformer![model].use];
|
||||||
|
newUseArray.splice(transformerIndex, 1);
|
||||||
|
newProviders[providerIndex].transformer![model].use = newUseArray;
|
||||||
|
|
||||||
|
// If use array is now empty and no other properties, remove model transformer entirely
|
||||||
|
if (newUseArray.length === 0 && Object.keys(newProviders[providerIndex].transformer![model]).length === 1) {
|
||||||
|
delete newProviders[providerIndex].transformer![model];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const addProviderTransformerParameter = (providerIndex: number, transformerIndex: number, paramName: string, paramValue: string) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (!newProviders[providerIndex].transformer) {
|
||||||
|
newProviders[providerIndex].transformer = { use: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add parameter to the specified transformer in use array
|
||||||
|
if (newProviders[providerIndex].transformer!.use && newProviders[providerIndex].transformer!.use.length > transformerIndex) {
|
||||||
|
const targetTransformer = newProviders[providerIndex].transformer!.use[transformerIndex];
|
||||||
|
|
||||||
|
// If it's already an array with parameters, update it
|
||||||
|
if (Array.isArray(targetTransformer)) {
|
||||||
|
const transformerArray = [...targetTransformer];
|
||||||
|
// Check if the second element is an object (parameters object)
|
||||||
|
if (transformerArray.length > 1 && typeof transformerArray[1] === 'object' && transformerArray[1] !== null) {
|
||||||
|
// Update the existing parameters object
|
||||||
|
const existingParams = transformerArray[1] as Record<string, unknown>;
|
||||||
|
const paramsObj: Record<string, unknown> = { ...existingParams, [paramName]: paramValue };
|
||||||
|
transformerArray[1] = paramsObj;
|
||||||
|
} else if (transformerArray.length > 1) {
|
||||||
|
// If there are other elements, add the parameters object
|
||||||
|
const paramsObj = { [paramName]: paramValue };
|
||||||
|
transformerArray.splice(1, transformerArray.length - 1, paramsObj);
|
||||||
|
} else {
|
||||||
|
// Add a new parameters object
|
||||||
|
const paramsObj = { [paramName]: paramValue };
|
||||||
|
transformerArray.push(paramsObj);
|
||||||
|
}
|
||||||
|
|
||||||
|
newProviders[providerIndex].transformer!.use[transformerIndex] = transformerArray as any;
|
||||||
|
} else {
|
||||||
|
// Convert to array format with parameters
|
||||||
|
const paramsObj = { [paramName]: paramValue };
|
||||||
|
newProviders[providerIndex].transformer!.use[transformerIndex] = [targetTransformer as string, paramsObj] as any;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const removeProviderTransformerParameterAtIndex = (providerIndex: number, transformerIndex: number, paramName: string) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (!newProviders[providerIndex].transformer?.use || newProviders[providerIndex].transformer.use.length <= transformerIndex) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetTransformer = newProviders[providerIndex].transformer.use[transformerIndex];
|
||||||
|
if (Array.isArray(targetTransformer) && targetTransformer.length > 1) {
|
||||||
|
const transformerArray = [...targetTransformer];
|
||||||
|
// Check if the second element is an object (parameters object)
|
||||||
|
if (typeof transformerArray[1] === 'object' && transformerArray[1] !== null) {
|
||||||
|
const paramsObj = { ...(transformerArray[1] as Record<string, unknown>) };
|
||||||
|
delete paramsObj[paramName];
|
||||||
|
|
||||||
|
// If the parameters object is now empty, remove it
|
||||||
|
if (Object.keys(paramsObj).length === 0) {
|
||||||
|
transformerArray.splice(1, 1);
|
||||||
|
} else {
|
||||||
|
transformerArray[1] = paramsObj;
|
||||||
|
}
|
||||||
|
|
||||||
|
newProviders[providerIndex].transformer!.use[transformerIndex] = transformerArray;
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const addModelTransformerParameter = (providerIndex: number, model: string, transformerIndex: number, paramName: string, paramValue: string) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (!newProviders[providerIndex].transformer) {
|
||||||
|
newProviders[providerIndex].transformer = { use: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!newProviders[providerIndex].transformer![model]) {
|
||||||
|
newProviders[providerIndex].transformer![model] = { use: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add parameter to the specified transformer in use array
|
||||||
|
if (newProviders[providerIndex].transformer![model].use && newProviders[providerIndex].transformer![model].use.length > transformerIndex) {
|
||||||
|
const targetTransformer = newProviders[providerIndex].transformer![model].use[transformerIndex];
|
||||||
|
|
||||||
|
// If it's already an array with parameters, update it
|
||||||
|
if (Array.isArray(targetTransformer)) {
|
||||||
|
const transformerArray = [...targetTransformer];
|
||||||
|
// Check if the second element is an object (parameters object)
|
||||||
|
if (transformerArray.length > 1 && typeof transformerArray[1] === 'object' && transformerArray[1] !== null) {
|
||||||
|
// Update the existing parameters object
|
||||||
|
const existingParams = transformerArray[1] as Record<string, unknown>;
|
||||||
|
const paramsObj: Record<string, unknown> = { ...existingParams, [paramName]: paramValue };
|
||||||
|
transformerArray[1] = paramsObj;
|
||||||
|
} else if (transformerArray.length > 1) {
|
||||||
|
// If there are other elements, add the parameters object
|
||||||
|
const paramsObj = { [paramName]: paramValue };
|
||||||
|
transformerArray.splice(1, transformerArray.length - 1, paramsObj);
|
||||||
|
} else {
|
||||||
|
// Add a new parameters object
|
||||||
|
const paramsObj = { [paramName]: paramValue };
|
||||||
|
transformerArray.push(paramsObj);
|
||||||
|
}
|
||||||
|
|
||||||
|
newProviders[providerIndex].transformer![model].use[transformerIndex] = transformerArray as any;
|
||||||
|
} else {
|
||||||
|
// Convert to array format with parameters
|
||||||
|
const paramsObj = { [paramName]: paramValue };
|
||||||
|
newProviders[providerIndex].transformer![model].use[transformerIndex] = [targetTransformer as string, paramsObj] as any;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
const removeModelTransformerParameterAtIndex = (providerIndex: number, model: string, transformerIndex: number, paramName: string) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
|
||||||
|
if (!newProviders[providerIndex].transformer?.[model]?.use || newProviders[providerIndex].transformer[model].use.length <= transformerIndex) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetTransformer = newProviders[providerIndex].transformer[model].use[transformerIndex];
|
||||||
|
if (Array.isArray(targetTransformer) && targetTransformer.length > 1) {
|
||||||
|
const transformerArray = [...targetTransformer];
|
||||||
|
// Check if the second element is an object (parameters object)
|
||||||
|
if (typeof transformerArray[1] === 'object' && transformerArray[1] !== null) {
|
||||||
|
const paramsObj = { ...(transformerArray[1] as Record<string, unknown>) };
|
||||||
|
delete paramsObj[paramName];
|
||||||
|
|
||||||
|
// If the parameters object is now empty, remove it
|
||||||
|
if (Object.keys(paramsObj).length === 0) {
|
||||||
|
transformerArray.splice(1, 1);
|
||||||
|
} else {
|
||||||
|
transformerArray[1] = paramsObj;
|
||||||
|
}
|
||||||
|
|
||||||
|
newProviders[providerIndex].transformer![model].use[transformerIndex] = transformerArray;
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddModel = (index: number, model: string) => {
|
||||||
|
if (!model.trim()) return;
|
||||||
|
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
const models = [...newProviders[index].models];
|
||||||
|
|
||||||
|
// Check if model already exists
|
||||||
|
if (!models.includes(model.trim())) {
|
||||||
|
models.push(model.trim());
|
||||||
|
newProviders[index].models = models;
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRemoveModel = (providerIndex: number, modelIndex: number) => {
|
||||||
|
const newProviders = [...config.Providers];
|
||||||
|
const models = [...newProviders[providerIndex].models];
|
||||||
|
models.splice(modelIndex, 1);
|
||||||
|
newProviders[providerIndex].models = models;
|
||||||
|
setConfig({ ...config, Providers: newProviders });
|
||||||
|
};
|
||||||
|
|
||||||
|
const editingProvider = editingProviderIndex !== null ? config.Providers[editingProviderIndex] : null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card className="flex h-full flex-col rounded-lg border shadow-sm">
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between border-b p-4">
|
||||||
|
<CardTitle className="text-lg">{t("providers.title")} <span className="text-sm font-normal text-gray-500">({config.Providers.length})</span></CardTitle>
|
||||||
|
<Button onClick={handleAddProvider}>{t("providers.add")}</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="flex-grow overflow-y-auto p-4">
|
||||||
|
<ProviderList
|
||||||
|
providers={config.Providers}
|
||||||
|
onEdit={setEditingProviderIndex}
|
||||||
|
onRemove={setDeletingProviderIndex}
|
||||||
|
/>
|
||||||
|
</CardContent>
|
||||||
|
|
||||||
|
{/* Edit Dialog */}
|
||||||
|
<Dialog open={editingProviderIndex !== null} onOpenChange={(open) => {
|
||||||
|
if (!open) {
|
||||||
|
handleCancelAddProvider();
|
||||||
|
}
|
||||||
|
}}>
|
||||||
|
<DialogContent className="max-h-[80vh] flex flex-col sm:max-w-2xl">
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>{t("providers.edit")}</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
{editingProvider && editingProviderIndex !== null && (
|
||||||
|
<div className="space-y-4 p-4 overflow-y-auto flex-grow">
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="name">{t("providers.name")}</Label>
|
||||||
|
<Input id="name" value={editingProvider.name} onChange={(e) => handleProviderChange(editingProviderIndex, 'name', e.target.value)} />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="api_base_url">{t("providers.api_base_url")}</Label>
|
||||||
|
<Input id="api_base_url" value={editingProvider.api_base_url} onChange={(e) => handleProviderChange(editingProviderIndex, 'api_base_url', e.target.value)} />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="api_key">{t("providers.api_key")}</Label>
|
||||||
|
<Input id="api_key" type="password" value={editingProvider.api_key} onChange={(e) => handleProviderChange(editingProviderIndex, 'api_key', e.target.value)} />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="models">{t("providers.models")}</Label>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<div className="flex-1">
|
||||||
|
{hasFetchedModels[editingProviderIndex] ? (
|
||||||
|
<ComboInput
|
||||||
|
ref={comboInputRef}
|
||||||
|
options={editingProvider.models.map(model => ({ label: model, value: model }))}
|
||||||
|
value=""
|
||||||
|
onChange={(_) => {
|
||||||
|
// 只更新输入值,不添加模型
|
||||||
|
}}
|
||||||
|
onEnter={(value) => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
handleAddModel(editingProviderIndex, value);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
inputPlaceholder={t("providers.models_placeholder")}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<Input
|
||||||
|
id="models"
|
||||||
|
placeholder={t("providers.models_placeholder")}
|
||||||
|
onKeyDown={(e) => {
|
||||||
|
if (e.key === 'Enter' && e.currentTarget.value.trim() && editingProviderIndex !== null) {
|
||||||
|
handleAddModel(editingProviderIndex, e.currentTarget.value);
|
||||||
|
e.currentTarget.value = '';
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
if (hasFetchedModels[editingProviderIndex] && comboInputRef.current) {
|
||||||
|
// 使用ComboInput的逻辑
|
||||||
|
const comboInput = comboInputRef.current as any;
|
||||||
|
const currentValue = comboInput.getCurrentValue();
|
||||||
|
if (currentValue && currentValue.trim() && editingProviderIndex !== null) {
|
||||||
|
handleAddModel(editingProviderIndex, currentValue.trim());
|
||||||
|
// 清空ComboInput
|
||||||
|
comboInput.clearInput();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// 使用普通Input的逻辑
|
||||||
|
const input = document.getElementById('models') as HTMLInputElement;
|
||||||
|
if (input && input.value.trim() && editingProviderIndex !== null) {
|
||||||
|
handleAddModel(editingProviderIndex, input.value);
|
||||||
|
input.value = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{t("providers.add_model")}
|
||||||
|
</Button>
|
||||||
|
{/* <Button
|
||||||
|
onClick={() => editingProvider && fetchAvailableModels(editingProvider)}
|
||||||
|
disabled={isFetchingModels}
|
||||||
|
variant="outline"
|
||||||
|
>
|
||||||
|
{isFetchingModels ? t("providers.fetching_models") : t("providers.fetch_available_models")}
|
||||||
|
</Button> */}
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-wrap gap-2 pt-2">
|
||||||
|
{editingProvider.models.map((model, modelIndex) => (
|
||||||
|
<Badge key={modelIndex} variant="outline" className="font-normal flex items-center gap-1">
|
||||||
|
{model}
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
className="ml-1 rounded-full hover:bg-gray-200"
|
||||||
|
onClick={() => editingProviderIndex !== null && handleRemoveModel(editingProviderIndex, modelIndex)}
|
||||||
|
>
|
||||||
|
<X className="h-3 w-3" />
|
||||||
|
</button>
|
||||||
|
</Badge>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Provider Transformer Selection */}
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("providers.provider_transformer")}</Label>
|
||||||
|
|
||||||
|
{/* Add new transformer */}
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Combobox
|
||||||
|
options={availableTransformers.map(t => ({
|
||||||
|
label: t.name,
|
||||||
|
value: t.name
|
||||||
|
}))}
|
||||||
|
value=""
|
||||||
|
onChange={(value) => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
handleProviderTransformerChange(editingProviderIndex, value);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
placeholder={t("providers.select_transformer")}
|
||||||
|
emptyPlaceholder={t("providers.no_transformers")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Display existing transformers */}
|
||||||
|
{editingProvider.transformer?.use && editingProvider.transformer.use.length > 0 && (
|
||||||
|
<div className="space-y-2 mt-2">
|
||||||
|
<div className="text-sm font-medium text-gray-700">{t("providers.selected_transformers")}</div>
|
||||||
|
{editingProvider.transformer.use.map((transformer: any, transformerIndex: number) => (
|
||||||
|
<div key={transformerIndex} className="border rounded-md p-3">
|
||||||
|
<div className="flex gap-2 items-center mb-2">
|
||||||
|
<div className="flex-1 bg-gray-50 rounded p-2 text-sm">
|
||||||
|
{typeof transformer === 'string' ? transformer : Array.isArray(transformer) ? String(transformer[0]) : String(transformer)}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
removeProviderTransformerAtIndex(editingProviderIndex, transformerIndex);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Trash2 className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Transformer-specific Parameters */}
|
||||||
|
<div className="mt-2 pl-4 border-l-2 border-gray-200">
|
||||||
|
<Label className="text-sm">{t("providers.transformer_parameters")}</Label>
|
||||||
|
<div className="space-y-2 mt-1">
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Input
|
||||||
|
placeholder={t("providers.parameter_name")}
|
||||||
|
value={providerParamInputs[`provider-${editingProviderIndex}-transformer-${transformerIndex}`]?.name || ""}
|
||||||
|
onChange={(e) => {
|
||||||
|
const key = `provider-${editingProviderIndex}-transformer-${transformerIndex}`;
|
||||||
|
setProviderParamInputs(prev => ({
|
||||||
|
...prev,
|
||||||
|
[key]: {
|
||||||
|
...prev[key] || {name: "", value: ""},
|
||||||
|
name: e.target.value
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
placeholder={t("providers.parameter_value")}
|
||||||
|
value={providerParamInputs[`provider-${editingProviderIndex}-transformer-${transformerIndex}`]?.value || ""}
|
||||||
|
onChange={(e) => {
|
||||||
|
const key = `provider-${editingProviderIndex}-transformer-${transformerIndex}`;
|
||||||
|
setProviderParamInputs(prev => ({
|
||||||
|
...prev,
|
||||||
|
[key]: {
|
||||||
|
...prev[key] || {name: "", value: ""},
|
||||||
|
value: e.target.value
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
const key = `provider-${editingProviderIndex}-transformer-${transformerIndex}`;
|
||||||
|
const paramInput = providerParamInputs[key];
|
||||||
|
if (paramInput && paramInput.name && paramInput.value) {
|
||||||
|
addProviderTransformerParameter(editingProviderIndex, transformerIndex, paramInput.name, paramInput.value);
|
||||||
|
setProviderParamInputs(prev => ({
|
||||||
|
...prev,
|
||||||
|
[key]: {name: "", value: ""}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Plus className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Display existing parameters for this transformer */}
|
||||||
|
{(() => {
|
||||||
|
// Get parameters for this specific transformer
|
||||||
|
if (!editingProvider.transformer?.use || editingProvider.transformer.use.length <= transformerIndex) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetTransformer = editingProvider.transformer.use[transformerIndex];
|
||||||
|
let params = {};
|
||||||
|
|
||||||
|
if (Array.isArray(targetTransformer) && targetTransformer.length > 1) {
|
||||||
|
// Check if the second element is an object (parameters object)
|
||||||
|
if (typeof targetTransformer[1] === 'object' && targetTransformer[1] !== null) {
|
||||||
|
params = targetTransformer[1] as Record<string, unknown>;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.keys(params).length > 0 ? (
|
||||||
|
<div className="space-y-1">
|
||||||
|
{Object.entries(params).map(([key, value]) => (
|
||||||
|
<div key={key} className="flex items-center justify-between bg-gray-50 rounded p-2">
|
||||||
|
<div className="text-sm">
|
||||||
|
<span className="font-medium">{key}:</span> {String(value)}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
className="h-6 w-6 p-0"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
// We need a function to remove parameters from a specific transformer
|
||||||
|
removeProviderTransformerParameterAtIndex(editingProviderIndex, transformerIndex, key);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<X className="h-3 w-3" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
) : null;
|
||||||
|
})()}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Model-specific Transformers */}
|
||||||
|
{editingProvider.models.length > 0 && (
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("providers.model_transformers")}</Label>
|
||||||
|
<div className="space-y-3">
|
||||||
|
{editingProvider.models.map((model, modelIndex) => (
|
||||||
|
<div key={modelIndex} className="border rounded-md p-3">
|
||||||
|
<div className="font-medium text-sm mb-2">{model}</div>
|
||||||
|
{/* Add new transformer */}
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<div className="flex-1 flex gap-2">
|
||||||
|
<Combobox
|
||||||
|
options={availableTransformers.map(t => ({
|
||||||
|
label: t.name,
|
||||||
|
value: t.name
|
||||||
|
}))}
|
||||||
|
value=""
|
||||||
|
onChange={(value) => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
handleModelTransformerChange(editingProviderIndex, model, value);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
placeholder={t("providers.select_transformer")}
|
||||||
|
emptyPlaceholder={t("providers.no_transformers")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Display existing transformers */}
|
||||||
|
{editingProvider.transformer?.[model]?.use && editingProvider.transformer[model].use.length > 0 && (
|
||||||
|
<div className="space-y-2 mt-2">
|
||||||
|
<div className="text-sm font-medium text-gray-700">{t("providers.selected_transformers")}</div>
|
||||||
|
{editingProvider.transformer[model].use.map((transformer: any, transformerIndex: number) => (
|
||||||
|
<div key={transformerIndex} className="border rounded-md p-3">
|
||||||
|
<div className="flex gap-2 items-center mb-2">
|
||||||
|
<div className="flex-1 bg-gray-50 rounded p-2 text-sm">
|
||||||
|
{typeof transformer === 'string' ? transformer : Array.isArray(transformer) ? String(transformer[0]) : String(transformer)}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
removeModelTransformerAtIndex(editingProviderIndex, model, transformerIndex);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Trash2 className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Transformer-specific Parameters */}
|
||||||
|
<div className="mt-2 pl-4 border-l-2 border-gray-200">
|
||||||
|
<Label className="text-sm">{t("providers.transformer_parameters")}</Label>
|
||||||
|
<div className="space-y-2 mt-1">
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Input
|
||||||
|
placeholder={t("providers.parameter_name")}
|
||||||
|
value={modelParamInputs[`model-${editingProviderIndex}-${model}-transformer-${transformerIndex}`]?.name || ""}
|
||||||
|
onChange={(e) => {
|
||||||
|
const key = `model-${editingProviderIndex}-${model}-transformer-${transformerIndex}`;
|
||||||
|
setModelParamInputs(prev => ({
|
||||||
|
...prev,
|
||||||
|
[key]: {
|
||||||
|
...prev[key] || {name: "", value: ""},
|
||||||
|
name: e.target.value
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
placeholder={t("providers.parameter_value")}
|
||||||
|
value={modelParamInputs[`model-${editingProviderIndex}-${model}-transformer-${transformerIndex}`]?.value || ""}
|
||||||
|
onChange={(e) => {
|
||||||
|
const key = `model-${editingProviderIndex}-${model}-transformer-${transformerIndex}`;
|
||||||
|
setModelParamInputs(prev => ({
|
||||||
|
...prev,
|
||||||
|
[key]: {
|
||||||
|
...prev[key] || {name: "", value: ""},
|
||||||
|
value: e.target.value
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
const key = `model-${editingProviderIndex}-${model}-transformer-${transformerIndex}`;
|
||||||
|
const paramInput = modelParamInputs[key];
|
||||||
|
if (paramInput && paramInput.name && paramInput.value) {
|
||||||
|
addModelTransformerParameter(editingProviderIndex, model, transformerIndex, paramInput.name, paramInput.value);
|
||||||
|
setModelParamInputs(prev => ({
|
||||||
|
...prev,
|
||||||
|
[key]: {name: "", value: ""}
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Plus className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Display existing parameters for this transformer */}
|
||||||
|
{(() => {
|
||||||
|
// Get parameters for this specific transformer
|
||||||
|
if (!editingProvider.transformer?.[model]?.use || editingProvider.transformer[model].use.length <= transformerIndex) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const targetTransformer = editingProvider.transformer[model].use[transformerIndex];
|
||||||
|
let params = {};
|
||||||
|
|
||||||
|
if (Array.isArray(targetTransformer) && targetTransformer.length > 1) {
|
||||||
|
// Check if the second element is an object (parameters object)
|
||||||
|
if (typeof targetTransformer[1] === 'object' && targetTransformer[1] !== null) {
|
||||||
|
params = targetTransformer[1] as Record<string, unknown>;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.keys(params).length > 0 ? (
|
||||||
|
<div className="space-y-1">
|
||||||
|
{Object.entries(params).map(([key, value]) => (
|
||||||
|
<div key={key} className="flex items-center justify-between bg-gray-50 rounded p-2">
|
||||||
|
<div className="text-sm">
|
||||||
|
<span className="font-medium">{key}:</span> {String(value)}
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
className="h-6 w-6 p-0"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingProviderIndex !== null) {
|
||||||
|
// We need a function to remove parameters from a specific transformer
|
||||||
|
removeModelTransformerParameterAtIndex(editingProviderIndex, model, transformerIndex, key);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<X className="h-3 w-3" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
) : null;
|
||||||
|
})()}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="space-y-3 mt-auto">
|
||||||
|
<div className="flex justify-end gap-2">
|
||||||
|
{/* <Button
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => editingProvider && testConnectivity(editingProvider)}
|
||||||
|
disabled={isTestingConnectivity || !editingProvider}
|
||||||
|
>
|
||||||
|
<Wifi className="mr-2 h-4 w-4" />
|
||||||
|
{isTestingConnectivity ? t("providers.testing") : t("providers.test_connectivity")}
|
||||||
|
</Button> */}
|
||||||
|
<Button onClick={handleSaveProvider}>{t("app.save")}</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
|
{/* Delete Confirmation Dialog */}
|
||||||
|
<Dialog open={deletingProviderIndex !== null} onOpenChange={() => setDeletingProviderIndex(null)}>
|
||||||
|
<DialogContent>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>{t("providers.delete")}</DialogTitle>
|
||||||
|
<DialogDescription>
|
||||||
|
{t("providers.delete_provider_confirm")}
|
||||||
|
</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
<DialogFooter>
|
||||||
|
<Button variant="outline" onClick={() => setDeletingProviderIndex(null)}>{t("providers.cancel")}</Button>
|
||||||
|
<Button variant="destructive" onClick={() => deletingProviderIndex !== null && handleRemoveProvider(deletingProviderIndex)}>{t("providers.delete")}</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
91
ui/src/components/Router.tsx
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { useConfig } from "./ConfigProvider";
|
||||||
|
import { Combobox } from "./ui/combobox";
|
||||||
|
|
||||||
|
export function Router() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleRouterChange = (field: string, value: string) => {
|
||||||
|
const newRouter = { ...config.Router, [field]: value };
|
||||||
|
setConfig({ ...config, Router: newRouter });
|
||||||
|
};
|
||||||
|
|
||||||
|
const modelOptions = config.Providers.flatMap((provider) =>
|
||||||
|
provider.models.map((model) => ({
|
||||||
|
value: `${provider.name},${model}`,
|
||||||
|
label: `${provider.name}, ${model}`,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card className="flex h-full flex-col rounded-lg border shadow-sm">
|
||||||
|
<CardHeader className="border-b p-4">
|
||||||
|
<CardTitle className="text-lg">{t("router.title")}</CardTitle>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="flex-grow space-y-5 overflow-y-auto p-4">
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("router.default")}</Label>
|
||||||
|
<Combobox
|
||||||
|
options={modelOptions}
|
||||||
|
value={config.Router.default}
|
||||||
|
onChange={(value) => handleRouterChange("default", value)}
|
||||||
|
placeholder={t("router.selectModel")}
|
||||||
|
searchPlaceholder={t("router.searchModel")}
|
||||||
|
emptyPlaceholder={t("router.noModelFound")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("router.background")}</Label>
|
||||||
|
<Combobox
|
||||||
|
options={modelOptions}
|
||||||
|
value={config.Router.background}
|
||||||
|
onChange={(value) => handleRouterChange("background", value)}
|
||||||
|
placeholder={t("router.selectModel")}
|
||||||
|
searchPlaceholder={t("router.searchModel")}
|
||||||
|
emptyPlaceholder={t("router.noModelFound")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("router.think")}</Label>
|
||||||
|
<Combobox
|
||||||
|
options={modelOptions}
|
||||||
|
value={config.Router.think}
|
||||||
|
onChange={(value) => handleRouterChange("think", value)}
|
||||||
|
placeholder={t("router.selectModel")}
|
||||||
|
searchPlaceholder={t("router.searchModel")}
|
||||||
|
emptyPlaceholder={t("router.noModelFound")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("router.longContext")}</Label>
|
||||||
|
<Combobox
|
||||||
|
options={modelOptions}
|
||||||
|
value={config.Router.longContext}
|
||||||
|
onChange={(value) => handleRouterChange("longContext", value)}
|
||||||
|
placeholder={t("router.selectModel")}
|
||||||
|
searchPlaceholder={t("router.searchModel")}
|
||||||
|
emptyPlaceholder={t("router.noModelFound")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label>{t("router.webSearch")}</Label>
|
||||||
|
<Combobox
|
||||||
|
options={modelOptions}
|
||||||
|
value={config.Router.webSearch}
|
||||||
|
onChange={(value) => handleRouterChange("webSearch", value)}
|
||||||
|
placeholder={t("router.selectModel")}
|
||||||
|
searchPlaceholder={t("router.searchModel")}
|
||||||
|
emptyPlaceholder={t("router.noModelFound")}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
71
ui/src/components/SettingsDialog.tsx
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
DialogFooter,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { Switch } from "@/components/ui/switch";
|
||||||
|
import { useConfig } from "./ConfigProvider";
|
||||||
|
|
||||||
|
interface SettingsDialogProps {
|
||||||
|
isOpen: boolean;
|
||||||
|
onOpenChange: (isOpen: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function SettingsDialog({ isOpen, onOpenChange }: SettingsDialogProps) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleLogChange = (checked: boolean) => {
|
||||||
|
setConfig({ ...config, LOG: checked });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handlePathChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
setConfig({ ...config, CLAUDE_PATH: e.target.value });
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog open={isOpen} onOpenChange={onOpenChange}>
|
||||||
|
<DialogContent>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>{t("toplevel.title")}</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
<div className="space-y-4 py-4">
|
||||||
|
<div className="flex items-center space-x-2">
|
||||||
|
<Switch id="log" checked={config.LOG} onCheckedChange={handleLogChange} />
|
||||||
|
<Label htmlFor="log" className="transition-all-ease hover:scale-[1.02] cursor-pointer">{t("toplevel.log")}</Label>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="claude-path" className="transition-all-ease hover:scale-[1.01] cursor-pointer">{t("toplevel.claude_path")}</Label>
|
||||||
|
<Input id="claude-path" value={config.CLAUDE_PATH} onChange={handlePathChange} className="transition-all-ease focus:scale-[1.01]" />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="host" className="transition-all-ease hover:scale-[1.01] cursor-pointer">{t("toplevel.host")}</Label>
|
||||||
|
<Input id="host" value={config.HOST} onChange={(e) => setConfig({ ...config, HOST: e.target.value })} className="transition-all-ease focus:scale-[1.01]" />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="port" className="transition-all-ease hover:scale-[1.01] cursor-pointer">{t("toplevel.port")}</Label>
|
||||||
|
<Input id="port" type="number" value={config.PORT} onChange={(e) => setConfig({ ...config, PORT: parseInt(e.target.value, 10) })} className="transition-all-ease focus:scale-[1.01]" />
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="apikey" className="transition-all-ease hover:scale-[1.01] cursor-pointer">{t("toplevel.apikey")}</Label>
|
||||||
|
<Input id="apikey" type="password" value={config.APIKEY} onChange={(e) => setConfig({ ...config, APIKEY: e.target.value })} className="transition-all-ease focus:scale-[1.01]" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<DialogFooter>
|
||||||
|
<Button onClick={() => onOpenChange(false)} className="transition-all-ease hover:scale-[1.02] active:scale-[0.98]">{t("app.save")}</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
}
|
||||||
32
ui/src/components/TransformerList.tsx
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import { Pencil, Trash2 } from "lucide-react";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { type Transformer } from "./ConfigProvider";
|
||||||
|
|
||||||
|
interface TransformerListProps {
|
||||||
|
transformers: Transformer[];
|
||||||
|
onEdit: (index: number) => void;
|
||||||
|
onRemove: (index: number) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function TransformerList({ transformers, onEdit, onRemove }: TransformerListProps) {
|
||||||
|
return (
|
||||||
|
<div className="space-y-3">
|
||||||
|
{transformers.map((transformer, index) => (
|
||||||
|
<div key={index} className="flex items-start justify-between rounded-md border bg-white p-4 transition-all hover:shadow-md animate-slide-in hover:scale-[1.01]">
|
||||||
|
<div className="flex-1 space-y-1.5">
|
||||||
|
<p className="text-md font-semibold text-gray-800">{transformer.path}</p>
|
||||||
|
<p className="text-sm text-gray-500">{transformer.options.project}</p>
|
||||||
|
</div>
|
||||||
|
<div className="ml-4 flex flex-shrink-0 items-center gap-2">
|
||||||
|
<Button variant="ghost" size="icon" onClick={() => onEdit(index)} className="transition-all-ease hover:scale-110">
|
||||||
|
<Pencil className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
<Button variant="destructive" size="icon" onClick={() => onRemove(index)} className="transition-all duration-200 hover:scale-110">
|
||||||
|
<Trash2 className="h-4 w-4 text-current transition-colors duration-200" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
220
ui/src/components/Transformers.tsx
Normal file
@@ -0,0 +1,220 @@
|
|||||||
|
import { useState } from "react";
|
||||||
|
import { useTranslation } from "react-i18next";
|
||||||
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
||||||
|
import { Input } from "@/components/ui/input";
|
||||||
|
import { Label } from "@/components/ui/label";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { Plus, Trash2 } from "lucide-react";
|
||||||
|
import { useConfig } from "./ConfigProvider";
|
||||||
|
import { TransformerList } from "./TransformerList";
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog";
|
||||||
|
|
||||||
|
export function Transformers() {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
const [editingTransformerIndex, setEditingTransformerIndex] = useState<number | null>(null);
|
||||||
|
const [deletingTransformerIndex, setDeletingTransformerIndex] = useState<number | null>(null);
|
||||||
|
const [newTransformer, setNewTransformer] = useState<{ path: string; options: { [key: string]: string } } | null>(null);
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleAddTransformer = () => {
|
||||||
|
const newTransformer = { path: "", options: {} };
|
||||||
|
setNewTransformer(newTransformer);
|
||||||
|
setEditingTransformerIndex(config.transformers.length); // Use the length as index for the new item
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRemoveTransformer = (index: number) => {
|
||||||
|
const newTransformers = [...config.transformers];
|
||||||
|
newTransformers.splice(index, 1);
|
||||||
|
setConfig({ ...config, transformers: newTransformers });
|
||||||
|
setDeletingTransformerIndex(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleTransformerChange = (index: number, field: string, value: string, optionKey?: string) => {
|
||||||
|
if (index < config.transformers.length) {
|
||||||
|
// Editing an existing transformer
|
||||||
|
const newTransformers = [...config.transformers];
|
||||||
|
if (optionKey !== undefined) {
|
||||||
|
newTransformers[index].options[optionKey] = value;
|
||||||
|
} else {
|
||||||
|
(newTransformers[index] as Record<string, unknown>)[field] = value;
|
||||||
|
}
|
||||||
|
setConfig({ ...config, transformers: newTransformers });
|
||||||
|
} else {
|
||||||
|
// Editing the new transformer
|
||||||
|
if (newTransformer) {
|
||||||
|
const updatedTransformer = { ...newTransformer };
|
||||||
|
if (optionKey !== undefined) {
|
||||||
|
updatedTransformer.options[optionKey] = value;
|
||||||
|
} else {
|
||||||
|
(updatedTransformer as Record<string, unknown>)[field] = value;
|
||||||
|
}
|
||||||
|
setNewTransformer(updatedTransformer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const editingTransformer = editingTransformerIndex !== null ?
|
||||||
|
(editingTransformerIndex < config.transformers.length ?
|
||||||
|
config.transformers[editingTransformerIndex] :
|
||||||
|
newTransformer) :
|
||||||
|
null;
|
||||||
|
|
||||||
|
const handleSaveTransformer = () => {
|
||||||
|
if (newTransformer && editingTransformerIndex === config.transformers.length) {
|
||||||
|
// Saving a new transformer
|
||||||
|
const newTransformers = [...config.transformers, newTransformer];
|
||||||
|
setConfig({ ...config, transformers: newTransformers });
|
||||||
|
}
|
||||||
|
// Close the dialog
|
||||||
|
setEditingTransformerIndex(null);
|
||||||
|
setNewTransformer(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleCancelTransformer = () => {
|
||||||
|
// Close the dialog without saving
|
||||||
|
setEditingTransformerIndex(null);
|
||||||
|
setNewTransformer(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Card className="flex h-full flex-col rounded-lg border shadow-sm">
|
||||||
|
<CardHeader className="flex flex-row items-center justify-between border-b p-4">
|
||||||
|
<CardTitle className="text-lg">{t("transformers.title")} <span className="text-sm font-normal text-gray-500">({config.transformers.length})</span></CardTitle>
|
||||||
|
<Button onClick={handleAddTransformer}>{t("transformers.add")}</Button>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="flex-grow overflow-y-auto p-4">
|
||||||
|
<TransformerList
|
||||||
|
transformers={config.transformers}
|
||||||
|
onEdit={setEditingTransformerIndex}
|
||||||
|
onRemove={setDeletingTransformerIndex}
|
||||||
|
/>
|
||||||
|
</CardContent>
|
||||||
|
|
||||||
|
{/* Edit Dialog */}
|
||||||
|
<Dialog open={editingTransformerIndex !== null} onOpenChange={handleCancelTransformer}>
|
||||||
|
<DialogContent>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>{t("transformers.edit")}</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
{editingTransformer && editingTransformerIndex !== null && (
|
||||||
|
<div className="space-y-4 py-4 px-6 max-h-96 overflow-y-auto">
|
||||||
|
<div className="space-y-2">
|
||||||
|
<Label htmlFor="transformer-path">{t("transformers.path")}</Label>
|
||||||
|
<Input
|
||||||
|
id="transformer-path"
|
||||||
|
value={editingTransformer.path}
|
||||||
|
onChange={(e) => handleTransformerChange(editingTransformerIndex, "path", e.target.value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="space-y-2">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<Label>{t("transformers.parameters")}</Label>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="sm"
|
||||||
|
onClick={() => {
|
||||||
|
const newKey = `param${Object.keys(editingTransformer.options).length + 1}`;
|
||||||
|
if (editingTransformerIndex !== null) {
|
||||||
|
const newOptions = { ...editingTransformer.options, [newKey]: "" };
|
||||||
|
if (editingTransformerIndex < config.transformers.length) {
|
||||||
|
const newTransformers = [...config.transformers];
|
||||||
|
newTransformers[editingTransformerIndex].options = newOptions;
|
||||||
|
setConfig({ ...config, transformers: newTransformers });
|
||||||
|
} else if (newTransformer) {
|
||||||
|
setNewTransformer({ ...newTransformer, options: newOptions });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Plus className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
{Object.entries(editingTransformer.options).map(([key, value]) => (
|
||||||
|
<div key={key} className="flex items-center gap-2">
|
||||||
|
<Input
|
||||||
|
value={key}
|
||||||
|
onChange={(e) => {
|
||||||
|
const newOptions = { ...editingTransformer.options };
|
||||||
|
delete newOptions[key];
|
||||||
|
newOptions[e.target.value] = value;
|
||||||
|
if (editingTransformerIndex !== null) {
|
||||||
|
if (editingTransformerIndex < config.transformers.length) {
|
||||||
|
const newTransformers = [...config.transformers];
|
||||||
|
newTransformers[editingTransformerIndex].options = newOptions;
|
||||||
|
setConfig({ ...config, transformers: newTransformers });
|
||||||
|
} else if (newTransformer) {
|
||||||
|
setNewTransformer({ ...newTransformer, options: newOptions });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
className="flex-1"
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
value={value}
|
||||||
|
onChange={(e) => {
|
||||||
|
if (editingTransformerIndex !== null) {
|
||||||
|
handleTransformerChange(editingTransformerIndex, "options", e.target.value, key);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
className="flex-1"
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
size="icon"
|
||||||
|
onClick={() => {
|
||||||
|
if (editingTransformerIndex !== null) {
|
||||||
|
const newOptions = { ...editingTransformer.options };
|
||||||
|
delete newOptions[key];
|
||||||
|
if (editingTransformerIndex < config.transformers.length) {
|
||||||
|
const newTransformers = [...config.transformers];
|
||||||
|
newTransformers[editingTransformerIndex].options = newOptions;
|
||||||
|
setConfig({ ...config, transformers: newTransformers });
|
||||||
|
} else if (newTransformer) {
|
||||||
|
setNewTransformer({ ...newTransformer, options: newOptions });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Trash2 className="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<DialogFooter>
|
||||||
|
<Button variant="outline" onClick={handleCancelTransformer}>{t("app.cancel")}</Button>
|
||||||
|
<Button onClick={handleSaveTransformer}>{t("app.save")}</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
|
{/* Delete Confirmation Dialog */}
|
||||||
|
<Dialog open={deletingTransformerIndex !== null} onOpenChange={() => setDeletingTransformerIndex(null)}>
|
||||||
|
<DialogContent>
|
||||||
|
<DialogHeader>
|
||||||
|
<DialogTitle>{t("transformers.delete")}</DialogTitle>
|
||||||
|
<DialogDescription>
|
||||||
|
{t("transformers.delete_transformer_confirm")}
|
||||||
|
</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
<DialogFooter>
|
||||||
|
<Button variant="outline" onClick={() => setDeletingTransformerIndex(null)}>{t("app.cancel")}</Button>
|
||||||
|
<Button variant="destructive" onClick={() => deletingTransformerIndex !== null && handleRemoveTransformer(deletingTransformerIndex)}>{t("app.delete")}</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
38
ui/src/components/ui/badge.tsx
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const badgeVariants = cva(
|
||||||
|
"inline-flex items-center rounded-full border px-2.5 py-0.5 text-xs font-semibold transition-colors focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2",
|
||||||
|
{
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
default:
|
||||||
|
"border-transparent bg-primary text-primary-foreground hover:bg-primary/80",
|
||||||
|
secondary:
|
||||||
|
"border-transparent bg-secondary text-secondary-foreground hover:bg-secondary/80",
|
||||||
|
destructive:
|
||||||
|
"border-transparent bg-destructive text-destructive-foreground hover:bg-destructive/80",
|
||||||
|
outline: "text-foreground",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
variant: "default",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
export interface BadgeProps
|
||||||
|
extends React.HTMLAttributes<HTMLDivElement>,
|
||||||
|
VariantProps<typeof badgeVariants> {}
|
||||||
|
|
||||||
|
function Badge({ className, variant, ...props }: BadgeProps) {
|
||||||
|
return (
|
||||||
|
<div className={cn(badgeVariants({ variant }), className)} {...props} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line react-refresh/only-export-components
|
||||||
|
export { Badge, badgeVariants }
|
||||||
57
ui/src/components/ui/button.tsx
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import { Slot } from "@radix-ui/react-slot"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const buttonVariants = cva(
|
||||||
|
"inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0",
|
||||||
|
{
|
||||||
|
variants: {
|
||||||
|
variant: {
|
||||||
|
default: "bg-primary text-primary-foreground hover:bg-primary/90",
|
||||||
|
destructive:
|
||||||
|
"border border-destructive/50 text-destructive hover:bg-destructive hover:text-destructive-foreground",
|
||||||
|
outline:
|
||||||
|
"border border-input bg-background hover:bg-accent hover:text-accent-foreground",
|
||||||
|
secondary:
|
||||||
|
"bg-secondary text-secondary-foreground hover:bg-secondary/80",
|
||||||
|
ghost: "hover:bg-accent hover:text-accent-foreground",
|
||||||
|
link: "text-primary underline-offset-4 hover:underline",
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
default: "h-10 px-4 py-2",
|
||||||
|
sm: "h-9 rounded-md px-3",
|
||||||
|
lg: "h-11 rounded-md px-8",
|
||||||
|
icon: "h-10 w-10",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaultVariants: {
|
||||||
|
variant: "default",
|
||||||
|
size: "default",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
export interface ButtonProps
|
||||||
|
extends React.ButtonHTMLAttributes<HTMLButtonElement>,
|
||||||
|
VariantProps<typeof buttonVariants> {
|
||||||
|
asChild?: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
|
||||||
|
({ className, variant, size, asChild = false, ...props }, ref) => {
|
||||||
|
const Comp = asChild ? Slot : "button"
|
||||||
|
return (
|
||||||
|
<Comp
|
||||||
|
className={cn(buttonVariants({ variant, size, className }))}
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
Button.displayName = "Button"
|
||||||
|
|
||||||
|
// eslint-disable-next-line react-refresh/only-export-components
|
||||||
|
export { Button, buttonVariants }
|
||||||
79
ui/src/components/ui/card.tsx
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Card = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"rounded-lg border bg-card text-card-foreground shadow-sm",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Card.displayName = "Card"
|
||||||
|
|
||||||
|
const CardHeader = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("flex flex-col space-y-1.5 p-6", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardHeader.displayName = "CardHeader"
|
||||||
|
|
||||||
|
const CardTitle = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"text-2xl font-semibold leading-none tracking-tight",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardTitle.displayName = "CardTitle"
|
||||||
|
|
||||||
|
const CardDescription = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm text-muted-foreground", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardDescription.displayName = "CardDescription"
|
||||||
|
|
||||||
|
const CardContent = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div ref={ref} className={cn("p-6 pt-0", className)} {...props} />
|
||||||
|
))
|
||||||
|
CardContent.displayName = "CardContent"
|
||||||
|
|
||||||
|
const CardFooter = React.forwardRef<
|
||||||
|
HTMLDivElement,
|
||||||
|
React.HTMLAttributes<HTMLDivElement>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<div
|
||||||
|
ref={ref}
|
||||||
|
className={cn("flex items-center p-6 pt-0", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
CardFooter.displayName = "CardFooter"
|
||||||
|
|
||||||
|
export { Card, CardHeader, CardFooter, CardTitle, CardDescription, CardContent }
|
||||||
139
ui/src/components/ui/combo-input.tsx
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import { Check, ChevronsUpDown } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
|
import { Input } from "@/components/ui/input"
|
||||||
|
import {
|
||||||
|
Command,
|
||||||
|
CommandEmpty,
|
||||||
|
CommandGroup,
|
||||||
|
CommandInput,
|
||||||
|
CommandItem,
|
||||||
|
CommandList,
|
||||||
|
} from "@/components/ui/command"
|
||||||
|
import {
|
||||||
|
Popover,
|
||||||
|
PopoverContent,
|
||||||
|
PopoverTrigger,
|
||||||
|
} from "@/components/ui/popover"
|
||||||
|
|
||||||
|
interface ComboInputProps {
|
||||||
|
options: { label: string; value: string }[];
|
||||||
|
value?: string;
|
||||||
|
onChange: (value: string) => void;
|
||||||
|
onEnter?: (value: string) => void;
|
||||||
|
searchPlaceholder?: string;
|
||||||
|
emptyPlaceholder?: string;
|
||||||
|
inputPlaceholder?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ComboInput = React.forwardRef<HTMLInputElement, ComboInputProps>(({
|
||||||
|
options,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
onEnter,
|
||||||
|
searchPlaceholder = "Search...",
|
||||||
|
emptyPlaceholder = "No options found.",
|
||||||
|
inputPlaceholder = "Type or select...",
|
||||||
|
}, ref) => {
|
||||||
|
const [open, setOpen] = React.useState(false)
|
||||||
|
const [inputValue, setInputValue] = React.useState(value || "")
|
||||||
|
const internalInputRef = React.useRef<HTMLInputElement>(null)
|
||||||
|
|
||||||
|
// Forward ref to the internal input
|
||||||
|
React.useImperativeHandle(ref, () => internalInputRef.current as HTMLInputElement)
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
setInputValue(value || "")
|
||||||
|
}, [value])
|
||||||
|
|
||||||
|
const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
|
const newValue = e.target.value
|
||||||
|
setInputValue(newValue)
|
||||||
|
onChange(newValue)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||||
|
if (e.key === 'Enter' && inputValue.trim() && onEnter) {
|
||||||
|
onEnter(inputValue.trim())
|
||||||
|
setInputValue("")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleSelect = (selectedValue: string) => {
|
||||||
|
setInputValue(selectedValue)
|
||||||
|
onChange(selectedValue)
|
||||||
|
if (onEnter) {
|
||||||
|
onEnter(selectedValue)
|
||||||
|
setInputValue("")
|
||||||
|
}
|
||||||
|
setOpen(false)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Function to get current value for external access
|
||||||
|
const getCurrentValue = () => inputValue
|
||||||
|
|
||||||
|
// Expose methods through the ref
|
||||||
|
React.useImperativeHandle(ref, () => ({
|
||||||
|
...internalInputRef.current!,
|
||||||
|
value: inputValue,
|
||||||
|
getCurrentValue,
|
||||||
|
clearInput: () => {
|
||||||
|
setInputValue("")
|
||||||
|
onChange("")
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="relative">
|
||||||
|
<Input
|
||||||
|
ref={internalInputRef}
|
||||||
|
value={inputValue}
|
||||||
|
onChange={handleInputChange}
|
||||||
|
onKeyDown={handleKeyDown}
|
||||||
|
placeholder={inputPlaceholder}
|
||||||
|
className="pr-10"
|
||||||
|
/>
|
||||||
|
<Popover open={open} onOpenChange={setOpen}>
|
||||||
|
<PopoverTrigger asChild>
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
className="absolute right-0 top-0 h-full px-3 py-2 hover:bg-transparent"
|
||||||
|
>
|
||||||
|
<ChevronsUpDown className="h-4 w-4 opacity-50" />
|
||||||
|
</Button>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent className="w-[--radix-popover-trigger-width] p-0 animate-fade-in">
|
||||||
|
<Command>
|
||||||
|
<CommandInput placeholder={searchPlaceholder} />
|
||||||
|
<CommandList>
|
||||||
|
<CommandEmpty>{emptyPlaceholder}</CommandEmpty>
|
||||||
|
<CommandGroup>
|
||||||
|
{options.map((option) => (
|
||||||
|
<CommandItem
|
||||||
|
key={option.value}
|
||||||
|
value={option.value}
|
||||||
|
onSelect={() => handleSelect(option.value)}
|
||||||
|
className="transition-all-ease hover:bg-accent hover:text-accent-foreground"
|
||||||
|
>
|
||||||
|
<Check
|
||||||
|
className={cn(
|
||||||
|
"mr-2 h-4 w-4 transition-opacity",
|
||||||
|
value === option.value ? "opacity-100" : "opacity-0"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
{option.label}
|
||||||
|
</CommandItem>
|
||||||
|
))}
|
||||||
|
</CommandGroup>
|
||||||
|
</CommandList>
|
||||||
|
</Command>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
})
|
||||||
87
ui/src/components/ui/combobox.tsx
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import { Check, ChevronsUpDown } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
|
import {
|
||||||
|
Command,
|
||||||
|
CommandEmpty,
|
||||||
|
CommandGroup,
|
||||||
|
CommandInput,
|
||||||
|
CommandItem,
|
||||||
|
CommandList,
|
||||||
|
} from "@/components/ui/command"
|
||||||
|
import {
|
||||||
|
Popover,
|
||||||
|
PopoverContent,
|
||||||
|
PopoverTrigger,
|
||||||
|
} from "@/components/ui/popover"
|
||||||
|
|
||||||
|
interface ComboboxProps {
|
||||||
|
options: { label: string; value: string }[];
|
||||||
|
value?: string;
|
||||||
|
onChange: (value: string) => void;
|
||||||
|
placeholder?: string;
|
||||||
|
searchPlaceholder?: string;
|
||||||
|
emptyPlaceholder?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Combobox({
|
||||||
|
options,
|
||||||
|
value,
|
||||||
|
onChange,
|
||||||
|
placeholder = "Select an option...",
|
||||||
|
searchPlaceholder = "Search...",
|
||||||
|
emptyPlaceholder = "No options found.",
|
||||||
|
}: ComboboxProps) {
|
||||||
|
const [open, setOpen] = React.useState(false)
|
||||||
|
|
||||||
|
const selectedOption = options.find((option) => option.value === value)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Popover open={open} onOpenChange={setOpen}>
|
||||||
|
<PopoverTrigger asChild>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
role="combobox"
|
||||||
|
aria-expanded={open}
|
||||||
|
className="w-full justify-between transition-all-ease hover:scale-[1.02] active:scale-[0.98]"
|
||||||
|
>
|
||||||
|
{selectedOption ? selectedOption.label : placeholder}
|
||||||
|
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50 transition-transform duration-200 group-data-[state=open]:rotate-180" />
|
||||||
|
</Button>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent className="w-[--radix-popover-trigger-width] p-0 animate-fade-in">
|
||||||
|
<Command>
|
||||||
|
<CommandInput placeholder={searchPlaceholder} />
|
||||||
|
<CommandList>
|
||||||
|
<CommandEmpty>{emptyPlaceholder}</CommandEmpty>
|
||||||
|
<CommandGroup>
|
||||||
|
{options.map((option) => (
|
||||||
|
<CommandItem
|
||||||
|
key={option.value}
|
||||||
|
value={option.value}
|
||||||
|
onSelect={(currentValue) => {
|
||||||
|
onChange(currentValue === value ? "" : currentValue)
|
||||||
|
setOpen(false)
|
||||||
|
}}
|
||||||
|
className="transition-all-ease hover:bg-accent hover:text-accent-foreground"
|
||||||
|
>
|
||||||
|
<Check
|
||||||
|
className={cn(
|
||||||
|
"mr-2 h-4 w-4 transition-opacity",
|
||||||
|
value === option.value ? "opacity-100" : "opacity-0"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
{option.label}
|
||||||
|
</CommandItem>
|
||||||
|
))}
|
||||||
|
</CommandGroup>
|
||||||
|
</CommandList>
|
||||||
|
</Command>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
)
|
||||||
|
}
|
||||||
181
ui/src/components/ui/command.tsx
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import { Command as CommandPrimitive } from "cmdk"
|
||||||
|
import { SearchIcon } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
} from "@/components/ui/dialog"
|
||||||
|
|
||||||
|
function Command({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive>) {
|
||||||
|
return (
|
||||||
|
<CommandPrimitive
|
||||||
|
data-slot="command"
|
||||||
|
className={cn(
|
||||||
|
"bg-popover text-popover-foreground flex h-full w-full flex-col overflow-hidden rounded-md",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandDialog({
|
||||||
|
title = "Command Palette",
|
||||||
|
description = "Search for a command to run...",
|
||||||
|
children,
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof Dialog> & {
|
||||||
|
title?: string
|
||||||
|
description?: string
|
||||||
|
className?: string
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<Dialog {...props}>
|
||||||
|
<DialogHeader className="sr-only">
|
||||||
|
<DialogTitle>{title}</DialogTitle>
|
||||||
|
<DialogDescription>{description}</DialogDescription>
|
||||||
|
</DialogHeader>
|
||||||
|
<DialogContent
|
||||||
|
className={cn("overflow-hidden p-0", className)}
|
||||||
|
>
|
||||||
|
<Command className="[&_[cmdk-group-heading]]:text-muted-foreground **:data-[slot=command-input-wrapper]:h-12 [&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group]]:px-2 [&_[cmdk-group]:not([hidden])_~[cmdk-group]]:pt-0 [&_[cmdk-input-wrapper]_svg]:h-5 [&_[cmdk-input-wrapper]_svg]:w-5 [&_[cmdk-input]]:h-12 [&_[cmdk-item]]:px-2 [&_[cmdk-item]]:py-3 [&_[cmdk-item]_svg]:h-5 [&_[cmdk-item]_svg]:w-5">
|
||||||
|
{children}
|
||||||
|
</Command>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandInput({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive.Input>) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
data-slot="command-input-wrapper"
|
||||||
|
className="flex h-9 items-center gap-2 border-b px-3"
|
||||||
|
>
|
||||||
|
<SearchIcon className="size-4 shrink-0 opacity-50" />
|
||||||
|
<CommandPrimitive.Input
|
||||||
|
data-slot="command-input"
|
||||||
|
className={cn(
|
||||||
|
"placeholder:text-muted-foreground flex h-10 w-full rounded-md bg-transparent py-3 text-sm outline-hidden disabled:cursor-not-allowed disabled:opacity-50",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandList({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive.List>) {
|
||||||
|
return (
|
||||||
|
<CommandPrimitive.List
|
||||||
|
data-slot="command-list"
|
||||||
|
className={cn(
|
||||||
|
"max-h-[300px] scroll-py-1 overflow-x-hidden overflow-y-auto",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandEmpty({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive.Empty>) {
|
||||||
|
return (
|
||||||
|
<CommandPrimitive.Empty
|
||||||
|
data-slot="command-empty"
|
||||||
|
className="py-6 text-center text-sm"
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandGroup({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive.Group>) {
|
||||||
|
return (
|
||||||
|
<CommandPrimitive.Group
|
||||||
|
data-slot="command-group"
|
||||||
|
className={cn(
|
||||||
|
"text-foreground [&_[cmdk-group-heading]]:text-muted-foreground overflow-hidden p-1 [&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandSeparator({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive.Separator>) {
|
||||||
|
return (
|
||||||
|
<CommandPrimitive.Separator
|
||||||
|
data-slot="command-separator"
|
||||||
|
className={cn("bg-border -mx-1 h-px", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandItem({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof CommandPrimitive.Item>) {
|
||||||
|
return (
|
||||||
|
<CommandPrimitive.Item
|
||||||
|
data-slot="command-item"
|
||||||
|
className={cn(
|
||||||
|
"data-[selected=true]:bg-accent data-[selected=true]:text-accent-foreground [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled=true]:pointer-events-none data-[disabled=true]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function CommandShortcut({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<"span">) {
|
||||||
|
return (
|
||||||
|
<span
|
||||||
|
data-slot="command-shortcut"
|
||||||
|
className={cn(
|
||||||
|
"text-muted-foreground ml-auto text-xs tracking-widest",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export {
|
||||||
|
Command,
|
||||||
|
CommandDialog,
|
||||||
|
CommandInput,
|
||||||
|
CommandList,
|
||||||
|
CommandEmpty,
|
||||||
|
CommandGroup,
|
||||||
|
CommandItem,
|
||||||
|
CommandShortcut,
|
||||||
|
CommandSeparator,
|
||||||
|
}
|
||||||
125
ui/src/components/ui/dialog.tsx
Normal file
@@ -0,0 +1,125 @@
|
|||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as DialogPrimitive from "@radix-ui/react-dialog"
|
||||||
|
import { X } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Dialog = DialogPrimitive.Root
|
||||||
|
|
||||||
|
const DialogTrigger = DialogPrimitive.Trigger
|
||||||
|
|
||||||
|
const DialogPortal = DialogPrimitive.Portal
|
||||||
|
|
||||||
|
const DialogClose = DialogPrimitive.Close
|
||||||
|
|
||||||
|
const DialogOverlay = React.forwardRef<(
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Overlay>
|
||||||
|
), (
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay>
|
||||||
|
)>(({ className, ...props }, ref) => (
|
||||||
|
<DialogPrimitive.Overlay
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
DialogOverlay.displayName = DialogPrimitive.Overlay.displayName
|
||||||
|
|
||||||
|
const DialogContent = React.forwardRef<(
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Content>
|
||||||
|
), (
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content>
|
||||||
|
)>(({ className, children, ...props }, ref) => (
|
||||||
|
<DialogPortal>
|
||||||
|
<DialogOverlay />
|
||||||
|
<DialogPrimitive.Content
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"fixed left-[50%] top-[50%] z-50 grid w-full max-w-lg translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-300 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg animate-scale-in",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
<DialogPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground transition-all-ease hover:scale-110">
|
||||||
|
<X className="h-4 w-4" />
|
||||||
|
<span className="sr-only">Close</span>
|
||||||
|
</DialogPrimitive.Close>
|
||||||
|
</DialogPrimitive.Content>
|
||||||
|
</DialogPortal>
|
||||||
|
))
|
||||||
|
DialogContent.displayName = DialogPrimitive.Content.displayName
|
||||||
|
|
||||||
|
const DialogHeader = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLDivElement>) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"flex flex-col space-y-1.5 text-center sm:text-left",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
DialogHeader.displayName = "DialogHeader"
|
||||||
|
|
||||||
|
const DialogFooter = ({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLDivElement>) => (
|
||||||
|
<div
|
||||||
|
className={cn(
|
||||||
|
"flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
DialogFooter.displayName = "DialogFooter"
|
||||||
|
|
||||||
|
const DialogTitle = React.forwardRef<(
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Title>
|
||||||
|
), (
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Title>
|
||||||
|
)>(({ className, ...props }, ref) => (
|
||||||
|
<DialogPrimitive.Title
|
||||||
|
ref={ref}
|
||||||
|
className={cn(
|
||||||
|
"text-lg font-semibold leading-none tracking-tight",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
DialogTitle.displayName = DialogPrimitive.Title.displayName
|
||||||
|
|
||||||
|
const DialogDescription = React.forwardRef<(
|
||||||
|
React.ElementRef<typeof DialogPrimitive.Description>
|
||||||
|
), (
|
||||||
|
React.ComponentPropsWithoutRef<typeof DialogPrimitive.Description>
|
||||||
|
)>(({ className, ...props }, ref) => (
|
||||||
|
<DialogPrimitive.Description
|
||||||
|
ref={ref}
|
||||||
|
className={cn("text-sm text-muted-foreground", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
DialogDescription.displayName = DialogPrimitive.Description.displayName
|
||||||
|
|
||||||
|
export {
|
||||||
|
Dialog,
|
||||||
|
DialogPortal,
|
||||||
|
DialogOverlay,
|
||||||
|
DialogTrigger,
|
||||||
|
DialogClose,
|
||||||
|
DialogContent,
|
||||||
|
DialogHeader,
|
||||||
|
DialogFooter,
|
||||||
|
DialogTitle,
|
||||||
|
DialogDescription,
|
||||||
|
}
|
||||||
22
ui/src/components/ui/input.tsx
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Input = React.forwardRef<HTMLInputElement, React.ComponentProps<"input">>(
|
||||||
|
({ className, type, ...props }, ref) => {
|
||||||
|
return (
|
||||||
|
<input
|
||||||
|
type={type}
|
||||||
|
className={cn(
|
||||||
|
"flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-base ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium file:text-foreground placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
)
|
||||||
|
Input.displayName = "Input"
|
||||||
|
|
||||||
|
export { Input }
|
||||||
24
ui/src/components/ui/label.tsx
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import * as LabelPrimitive from "@radix-ui/react-label"
|
||||||
|
import { cva, type VariantProps } from "class-variance-authority"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const labelVariants = cva(
|
||||||
|
"text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
|
||||||
|
)
|
||||||
|
|
||||||
|
const Label = React.forwardRef<
|
||||||
|
React.ElementRef<typeof LabelPrimitive.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof LabelPrimitive.Root> &
|
||||||
|
VariantProps<typeof labelVariants>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<LabelPrimitive.Root
|
||||||
|
ref={ref}
|
||||||
|
className={cn(labelVariants(), className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
Label.displayName = LabelPrimitive.Root.displayName
|
||||||
|
|
||||||
|
export { Label }
|
||||||
114
ui/src/components/ui/multi-combobox.tsx
Normal file
@@ -0,0 +1,114 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import { Check, ChevronsUpDown, X } from "lucide-react"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
|
import {
|
||||||
|
Command,
|
||||||
|
CommandEmpty,
|
||||||
|
CommandGroup,
|
||||||
|
CommandInput,
|
||||||
|
CommandItem,
|
||||||
|
CommandList,
|
||||||
|
} from "@/components/ui/command"
|
||||||
|
import {
|
||||||
|
Popover,
|
||||||
|
PopoverContent,
|
||||||
|
PopoverTrigger,
|
||||||
|
} from "@/components/ui/popover"
|
||||||
|
import { Badge } from "@/components/ui/badge"
|
||||||
|
|
||||||
|
interface MultiComboboxProps {
|
||||||
|
options: { label: string; value: string }[];
|
||||||
|
value?: string[];
|
||||||
|
onChange: (value: string[]) => void;
|
||||||
|
placeholder?: string;
|
||||||
|
searchPlaceholder?: string;
|
||||||
|
emptyPlaceholder?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function MultiCombobox({
|
||||||
|
options,
|
||||||
|
value = [],
|
||||||
|
onChange,
|
||||||
|
placeholder = "Select options...",
|
||||||
|
searchPlaceholder = "Search...",
|
||||||
|
emptyPlaceholder = "No options found.",
|
||||||
|
}: MultiComboboxProps) {
|
||||||
|
const [open, setOpen] = React.useState(false)
|
||||||
|
|
||||||
|
const handleSelect = (currentValue: string) => {
|
||||||
|
if (value.includes(currentValue)) {
|
||||||
|
onChange(value.filter(v => v !== currentValue))
|
||||||
|
} else {
|
||||||
|
onChange([...value, currentValue])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const removeValue = (val: string, e: React.MouseEvent) => {
|
||||||
|
e.stopPropagation()
|
||||||
|
onChange(value.filter(v => v !== val))
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="space-y-2">
|
||||||
|
<div className="flex flex-wrap gap-1">
|
||||||
|
{value.map((val) => {
|
||||||
|
const option = options.find(opt => opt.value === val)
|
||||||
|
return (
|
||||||
|
<Badge key={val} variant="outline" className="font-normal">
|
||||||
|
{option?.label || val}
|
||||||
|
<button
|
||||||
|
onClick={(e) => removeValue(val, e)}
|
||||||
|
className="ml-1 rounded-full hover:bg-gray-200"
|
||||||
|
>
|
||||||
|
<X className="h-3 w-3" />
|
||||||
|
</button>
|
||||||
|
</Badge>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
<Popover open={open} onOpenChange={setOpen}>
|
||||||
|
<PopoverTrigger asChild>
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
role="combobox"
|
||||||
|
aria-expanded={open}
|
||||||
|
className="w-full justify-between transition-all-ease hover:scale-[1.02] active:scale-[0.98]"
|
||||||
|
>
|
||||||
|
{value.length > 0 ? `${value.length} selected` : placeholder}
|
||||||
|
<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50 transition-transform duration-200 group-data-[state=open]:rotate-180" />
|
||||||
|
</Button>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent className="w-[--radix-popover-trigger-width] p-0 animate-fade-in">
|
||||||
|
<Command>
|
||||||
|
<CommandInput placeholder={searchPlaceholder} />
|
||||||
|
<CommandList>
|
||||||
|
<CommandEmpty>{emptyPlaceholder}</CommandEmpty>
|
||||||
|
<CommandGroup>
|
||||||
|
{options.map((option) => (
|
||||||
|
<CommandItem
|
||||||
|
key={option.value}
|
||||||
|
value={option.value}
|
||||||
|
onSelect={() => handleSelect(option.value)}
|
||||||
|
className="transition-all-ease hover:bg-accent hover:text-accent-foreground"
|
||||||
|
>
|
||||||
|
<Check
|
||||||
|
className={cn(
|
||||||
|
"mr-2 h-4 w-4 transition-opacity",
|
||||||
|
value.includes(option.value) ? "opacity-100" : "opacity-0"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
{option.label}
|
||||||
|
</CommandItem>
|
||||||
|
))}
|
||||||
|
</CommandGroup>
|
||||||
|
</CommandList>
|
||||||
|
</Command>
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
46
ui/src/components/ui/popover.tsx
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
import * as React from "react"
|
||||||
|
import * as PopoverPrimitive from "@radix-ui/react-popover"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
function Popover({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof PopoverPrimitive.Root>) {
|
||||||
|
return <PopoverPrimitive.Root data-slot="popover" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function PopoverTrigger({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof PopoverPrimitive.Trigger>) {
|
||||||
|
return <PopoverPrimitive.Trigger data-slot="popover-trigger" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
function PopoverContent({
|
||||||
|
className,
|
||||||
|
align = "center",
|
||||||
|
sideOffset = 4,
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof PopoverPrimitive.Content>) {
|
||||||
|
return (
|
||||||
|
<PopoverPrimitive.Portal>
|
||||||
|
<PopoverPrimitive.Content
|
||||||
|
data-slot="popover-content"
|
||||||
|
align={align}
|
||||||
|
sideOffset={sideOffset}
|
||||||
|
className={cn(
|
||||||
|
"bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-72 origin-(--radix-popover-content-transform-origin) rounded-md border p-4 shadow-md outline-hidden animate-fade-in",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
</PopoverPrimitive.Portal>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
function PopoverAnchor({
|
||||||
|
...props
|
||||||
|
}: React.ComponentProps<typeof PopoverPrimitive.Anchor>) {
|
||||||
|
return <PopoverPrimitive.Anchor data-slot="popover-anchor" {...props} />
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Popover, PopoverTrigger, PopoverContent, PopoverAnchor }
|
||||||
29
ui/src/components/ui/switch.tsx
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
"use client"
|
||||||
|
|
||||||
|
import * as React from "react"
|
||||||
|
import * as SwitchPrimitives from "@radix-ui/react-switch"
|
||||||
|
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
const Switch = React.forwardRef<
|
||||||
|
React.ElementRef<typeof SwitchPrimitives.Root>,
|
||||||
|
React.ComponentPropsWithoutRef<typeof SwitchPrimitives.Root>
|
||||||
|
>(({ className, ...props }, ref) => (
|
||||||
|
<SwitchPrimitives.Root
|
||||||
|
className={cn(
|
||||||
|
"peer inline-flex h-6 w-11 shrink-0 cursor-pointer items-center rounded-full border-2 border-transparent transition-all-ease focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 focus-visible:ring-offset-background disabled:cursor-not-allowed disabled:opacity-50 data-[state=checked]:bg-primary data-[state=unchecked]:bg-input",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...props}
|
||||||
|
ref={ref}
|
||||||
|
>
|
||||||
|
<SwitchPrimitives.Thumb
|
||||||
|
className={cn(
|
||||||
|
"pointer-events-none block h-5 w-5 rounded-full bg-background shadow-lg ring-0 transition-transform data-[state=checked]:translate-x-5 data-[state=unchecked]:translate-x-0 transition-all-ease"
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</SwitchPrimitives.Root>
|
||||||
|
))
|
||||||
|
Switch.displayName = SwitchPrimitives.Root.displayName
|
||||||
|
|
||||||
|
export { Switch }
|
||||||
59
ui/src/components/ui/toast.tsx
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
import { useEffect } from 'react';
|
||||||
|
import { CheckCircle, XCircle, AlertCircle, X } from 'lucide-react';
|
||||||
|
|
||||||
|
interface ToastProps {
|
||||||
|
message: string;
|
||||||
|
type: 'success' | 'error' | 'warning';
|
||||||
|
onClose: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function Toast({ message, type, onClose }: ToastProps) {
|
||||||
|
useEffect(() => {
|
||||||
|
const timer = setTimeout(() => {
|
||||||
|
onClose();
|
||||||
|
}, 3000);
|
||||||
|
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}, [onClose]);
|
||||||
|
|
||||||
|
const getIcon = () => {
|
||||||
|
switch (type) {
|
||||||
|
case 'success':
|
||||||
|
return <CheckCircle className="h-5 w-5 text-green-500" />;
|
||||||
|
case 'error':
|
||||||
|
return <XCircle className="h-5 w-5 text-red-500" />;
|
||||||
|
case 'warning':
|
||||||
|
return <AlertCircle className="h-5 w-5 text-yellow-500" />;
|
||||||
|
default:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const getBackgroundColor = () => {
|
||||||
|
switch (type) {
|
||||||
|
case 'success':
|
||||||
|
return 'bg-green-100 border-green-200';
|
||||||
|
case 'error':
|
||||||
|
return 'bg-red-100 border-red-200';
|
||||||
|
case 'warning':
|
||||||
|
return 'bg-yellow-100 border-yellow-200';
|
||||||
|
default:
|
||||||
|
return 'bg-gray-100 border-gray-200';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={`fixed top-4 right-4 z-50 flex items-center justify-between p-4 rounded-lg border shadow-lg ${getBackgroundColor()} transition-all duration-300 ease-in-out`}>
|
||||||
|
<div className="flex items-center space-x-2">
|
||||||
|
{getIcon()}
|
||||||
|
<span className="text-sm font-medium">{message}</span>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={onClose}
|
||||||
|
className="ml-4 text-gray-500 hover:text-gray-700 focus:outline-none"
|
||||||
|
>
|
||||||
|
<X className="h-4 w-4" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
28
ui/src/i18n.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import i18n from "i18next";
|
||||||
|
import { initReactI18next } from "react-i18next";
|
||||||
|
import LanguageDetector from "i18next-browser-languagedetector";
|
||||||
|
|
||||||
|
import en from "./locales/en.json";
|
||||||
|
import zh from "./locales/zh.json";
|
||||||
|
|
||||||
|
const resources = {
|
||||||
|
en: {
|
||||||
|
translation: en,
|
||||||
|
},
|
||||||
|
zh: {
|
||||||
|
translation: zh,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
i18n
|
||||||
|
.use(LanguageDetector)
|
||||||
|
.use(initReactI18next)
|
||||||
|
.init({
|
||||||
|
resources,
|
||||||
|
fallbackLng: "en",
|
||||||
|
interpolation: {
|
||||||
|
escapeValue: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export default i18n;
|
||||||
122
ui/src/index.css
Normal file
@@ -0,0 +1,122 @@
|
|||||||
|
@import "tailwindcss";
|
||||||
|
|
||||||
|
@custom-variant dark (&:is(.dark *));
|
||||||
|
|
||||||
|
@theme inline {
|
||||||
|
--radius-sm: calc(var(--radius) - 4px);
|
||||||
|
--radius-md: calc(var(--radius) - 2px);
|
||||||
|
--radius-lg: var(--radius);
|
||||||
|
--radius-xl: calc(var(--radius) + 4px);
|
||||||
|
--color-background: var(--background);
|
||||||
|
--color-foreground: var(--foreground);
|
||||||
|
--color-card: var(--card);
|
||||||
|
--color-card-foreground: var(--card-foreground);
|
||||||
|
--color-popover: var(--popover);
|
||||||
|
--color-popover-foreground: var(--popover-foreground);
|
||||||
|
--color-primary: var(--primary);
|
||||||
|
--color-primary-foreground: var(--primary-foreground);
|
||||||
|
--color-secondary: var(--secondary);
|
||||||
|
--color-secondary-foreground: var(--secondary-foreground);
|
||||||
|
--color-muted: var(--muted);
|
||||||
|
--color-muted-foreground: var(--muted-foreground);
|
||||||
|
--color-accent: var(--accent);
|
||||||
|
--color-accent-foreground: var(--accent-foreground);
|
||||||
|
--color-destructive: var(--destructive);
|
||||||
|
--color-destructive-foreground: var(--destructive-foreground);
|
||||||
|
--color-border: var(--border);
|
||||||
|
--color-input: var(--input);
|
||||||
|
--color-ring: var(--ring);
|
||||||
|
--color-chart-1: var(--chart-1);
|
||||||
|
--color-chart-2: var(--chart-2);
|
||||||
|
--color-chart-3: var(--chart-3);
|
||||||
|
--color-chart-4: var(--chart-4);
|
||||||
|
--color-chart-5: var(--chart-5);
|
||||||
|
--color-sidebar: var(--sidebar);
|
||||||
|
--color-sidebar-foreground: var(--sidebar-foreground);
|
||||||
|
--color-sidebar-primary: var(--sidebar-primary);
|
||||||
|
--color-sidebar-primary-foreground: var(--sidebar-primary-foreground);
|
||||||
|
--color-sidebar-accent: var(--sidebar-accent);
|
||||||
|
--color-sidebar-accent-foreground: var(--sidebar-accent-foreground);
|
||||||
|
--color-sidebar-border: var(--sidebar-border);
|
||||||
|
--color-sidebar-ring: var(--sidebar-ring);
|
||||||
|
}
|
||||||
|
|
||||||
|
:root {
|
||||||
|
--radius: 0.625rem;
|
||||||
|
--background: oklch(1 0 0);
|
||||||
|
--foreground: oklch(0.145 0 0);
|
||||||
|
--card: oklch(1 0 0);
|
||||||
|
--card-foreground: oklch(0.145 0 0);
|
||||||
|
--popover: oklch(1 0 0);
|
||||||
|
--popover-foreground: oklch(0.145 0 0);
|
||||||
|
--primary: oklch(0.205 0 0);
|
||||||
|
--primary-foreground: oklch(0.985 0 0);
|
||||||
|
--secondary: oklch(0.97 0 0);
|
||||||
|
--secondary-foreground: oklch(0.205 0 0);
|
||||||
|
--muted: oklch(0.97 0 0);
|
||||||
|
--muted-foreground: oklch(0.556 0 0);
|
||||||
|
--accent: oklch(0.97 0 0);
|
||||||
|
--accent-foreground: oklch(0.205 0 0);
|
||||||
|
--destructive: oklch(0.577 0.245 27.325);
|
||||||
|
--destructive-foreground: oklch(0.985 0 0);
|
||||||
|
--border: oklch(0.922 0 0);
|
||||||
|
--input: oklch(0.922 0 0);
|
||||||
|
--ring: oklch(0.708 0 0);
|
||||||
|
--chart-1: oklch(0.646 0.222 41.116);
|
||||||
|
--chart-2: oklch(0.6 0.118 184.704);
|
||||||
|
--chart-3: oklch(0.398 0.07 227.392);
|
||||||
|
--chart-4: oklch(0.828 0.189 84.429);
|
||||||
|
--chart-5: oklch(0.769 0.188 70.08);
|
||||||
|
--sidebar: oklch(0.985 0 0);
|
||||||
|
--sidebar-foreground: oklch(0.145 0 0);
|
||||||
|
--sidebar-primary: oklch(0.205 0 0);
|
||||||
|
--sidebar-primary-foreground: oklch(0.985 0 0);
|
||||||
|
--sidebar-accent: oklch(0.97 0 0);
|
||||||
|
--sidebar-accent-foreground: oklch(0.205 0 0);
|
||||||
|
--sidebar-border: oklch(0.922 0 0);
|
||||||
|
--sidebar-ring: oklch(0.708 0 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
.dark {
|
||||||
|
--background: oklch(0.145 0 0);
|
||||||
|
--foreground: oklch(0.985 0 0);
|
||||||
|
--card: oklch(0.205 0 0);
|
||||||
|
--card-foreground: oklch(0.985 0 0);
|
||||||
|
--popover: oklch(0.205 0 0);
|
||||||
|
--popover-foreground: oklch(0.985 0 0);
|
||||||
|
--primary: oklch(0.922 0 0);
|
||||||
|
--primary-foreground: oklch(0.205 0 0);
|
||||||
|
--secondary: oklch(0.269 0 0);
|
||||||
|
--secondary-foreground: oklch(0.985 0 0);
|
||||||
|
--muted: oklch(0.269 0 0);
|
||||||
|
--muted-foreground: oklch(0.708 0 0);
|
||||||
|
--accent: oklch(0.269 0 0);
|
||||||
|
--accent-foreground: oklch(0.985 0 0);
|
||||||
|
--destructive: oklch(0.704 0.191 22.216);
|
||||||
|
--destructive-foreground: oklch(0.985 0 0);
|
||||||
|
--border: oklch(1 0 0 / 10%);
|
||||||
|
--input: oklch(1 0 0 / 15%);
|
||||||
|
--ring: oklch(0.556 0 0);
|
||||||
|
--chart-1: oklch(0.488 0.243 264.376);
|
||||||
|
--chart-2: oklch(0.696 0.17 162.48);
|
||||||
|
--chart-3: oklch(0.769 0.188 70.08);
|
||||||
|
--chart-4: oklch(0.627 0.265 303.9);
|
||||||
|
--chart-5: oklch(0.645 0.246 16.439);
|
||||||
|
--sidebar: oklch(0.205 0 0);
|
||||||
|
--sidebar-foreground: oklch(0.985 0 0);
|
||||||
|
--sidebar-primary: oklch(0.488 0.243 264.376);
|
||||||
|
--sidebar-primary-foreground: oklch(0.985 0 0);
|
||||||
|
--sidebar-accent: oklch(0.269 0 0);
|
||||||
|
--sidebar-accent-foreground: oklch(0.985 0 0);
|
||||||
|
--sidebar-border: oklch(1 0 0 / 10%);
|
||||||
|
--sidebar-ring: oklch(0.556 0 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@layer base {
|
||||||
|
* {
|
||||||
|
@apply border-border outline-ring/50;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
@apply bg-background text-foreground;
|
||||||
|
}
|
||||||
|
}
|
||||||
189
ui/src/lib/api.ts
Normal file
@@ -0,0 +1,189 @@
|
|||||||
|
import type { Config, Provider, Transformer } from '@/components/ConfigProvider';
|
||||||
|
|
||||||
|
// API Client Class for handling requests with baseUrl and apikey authentication
|
||||||
|
class ApiClient {
|
||||||
|
private baseUrl: string;
|
||||||
|
private apiKey: string;
|
||||||
|
|
||||||
|
constructor(baseUrl: string = 'http://127.0.0.1:3456/api', apiKey: string = '') {
|
||||||
|
this.baseUrl = baseUrl;
|
||||||
|
// Load API key from localStorage if available
|
||||||
|
this.apiKey = apiKey || localStorage.getItem('apiKey') || '';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update base URL
|
||||||
|
setBaseUrl(url: string) {
|
||||||
|
this.baseUrl = url;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update API key
|
||||||
|
setApiKey(apiKey: string) {
|
||||||
|
this.apiKey = apiKey;
|
||||||
|
// Save API key to localStorage
|
||||||
|
if (apiKey) {
|
||||||
|
localStorage.setItem('apiKey', apiKey);
|
||||||
|
} else {
|
||||||
|
localStorage.removeItem('apiKey');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create headers with API key authentication
|
||||||
|
private createHeaders(contentType: string = 'application/json'): HeadersInit {
|
||||||
|
const headers: Record<string, string> = {
|
||||||
|
'X-API-Key': this.apiKey,
|
||||||
|
'Accept': 'application/json',
|
||||||
|
};
|
||||||
|
|
||||||
|
if (contentType) {
|
||||||
|
headers['Content-Type'] = contentType;
|
||||||
|
}
|
||||||
|
|
||||||
|
return headers;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Generic fetch wrapper with base URL and authentication
|
||||||
|
private async apiFetch<T>(endpoint: string, options: RequestInit = {}): Promise<T> {
|
||||||
|
const url = `${this.baseUrl}${endpoint}`;
|
||||||
|
|
||||||
|
const config: RequestInit = {
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
...this.createHeaders(),
|
||||||
|
...options.headers,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(url, config);
|
||||||
|
|
||||||
|
// Handle 401 Unauthorized responses
|
||||||
|
if (response.status === 401) {
|
||||||
|
// Remove API key when it's invalid
|
||||||
|
localStorage.removeItem('apiKey');
|
||||||
|
// Redirect to login page if not already there
|
||||||
|
// For memory router, we need to use the router instance
|
||||||
|
// We'll dispatch a custom event that the app can listen to
|
||||||
|
window.dispatchEvent(new CustomEvent('unauthorized'));
|
||||||
|
// Return a promise that never resolves to prevent further execution
|
||||||
|
return new Promise(() => {}) as Promise<T>;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`API request failed: ${response.status} ${response.statusText}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.status === 204) {
|
||||||
|
return {} as T;
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = await response.text();
|
||||||
|
return text ? JSON.parse(text) : ({} as T);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('API request error:', error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// GET request
|
||||||
|
async get<T>(endpoint: string): Promise<T> {
|
||||||
|
return this.apiFetch<T>(endpoint, {
|
||||||
|
method: 'GET',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// POST request
|
||||||
|
async post<T>(endpoint: string, data: unknown): Promise<T> {
|
||||||
|
return this.apiFetch<T>(endpoint, {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// PUT request
|
||||||
|
async put<T>(endpoint: string, data: unknown): Promise<T> {
|
||||||
|
return this.apiFetch<T>(endpoint, {
|
||||||
|
method: 'PUT',
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// DELETE request
|
||||||
|
async delete<T>(endpoint: string): Promise<T> {
|
||||||
|
return this.apiFetch<T>(endpoint, {
|
||||||
|
method: 'DELETE',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// API methods for configuration
|
||||||
|
// Get current configuration
|
||||||
|
async getConfig(): Promise<Config> {
|
||||||
|
return this.get<Config>('/config');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update entire configuration
|
||||||
|
async updateConfig(config: Config): Promise<Config> {
|
||||||
|
return this.post<Config>('/config', config);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get providers
|
||||||
|
async getProviders(): Promise<Provider[]> {
|
||||||
|
return this.get<Provider[]>('/api/providers');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add a new provider
|
||||||
|
async addProvider(provider: Provider): Promise<Provider> {
|
||||||
|
return this.post<Provider>('/api/providers', provider);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update a provider
|
||||||
|
async updateProvider(index: number, provider: Provider): Promise<Provider> {
|
||||||
|
return this.post<Provider>(`/api/providers/${index}`, provider);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete a provider
|
||||||
|
async deleteProvider(index: number): Promise<void> {
|
||||||
|
return this.delete<void>(`/api/providers/${index}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get transformers
|
||||||
|
async getTransformers(): Promise<Transformer[]> {
|
||||||
|
return this.get<Transformer[]>('/api/transformers');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add a new transformer
|
||||||
|
async addTransformer(transformer: Transformer): Promise<Transformer> {
|
||||||
|
return this.post<Transformer>('/api/transformers', transformer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update a transformer
|
||||||
|
async updateTransformer(index: number, transformer: Transformer): Promise<Transformer> {
|
||||||
|
return this.post<Transformer>(`/api/transformers/${index}`, transformer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Delete a transformer
|
||||||
|
async deleteTransformer(index: number): Promise<void> {
|
||||||
|
return this.delete<void>(`/api/transformers/${index}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get configuration (new endpoint)
|
||||||
|
async getConfigNew(): Promise<Config> {
|
||||||
|
return this.get<Config>('/config');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save configuration (new endpoint)
|
||||||
|
async saveConfig(config: Config): Promise<unknown> {
|
||||||
|
return this.post<Config>('/config', config);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restart service
|
||||||
|
async restartService(): Promise<unknown> {
|
||||||
|
return this.post<void>('/restart', {});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a default instance of the API client
|
||||||
|
export const api = new ApiClient();
|
||||||
|
|
||||||
|
// Export the class for creating custom instances
|
||||||
|
export default ApiClient;
|
||||||
6
ui/src/lib/utils.ts
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
import { clsx, type ClassValue } from "clsx"
|
||||||
|
import { twMerge } from "tailwind-merge"
|
||||||
|
|
||||||
|
export function cn(...inputs: ClassValue[]) {
|
||||||
|
return twMerge(clsx(inputs))
|
||||||
|
}
|
||||||
99
ui/src/locales/en.json
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
{
|
||||||
|
"app": {
|
||||||
|
"title": "Claude Code Router",
|
||||||
|
"save": "Save",
|
||||||
|
"save_and_restart": "Save and Restart",
|
||||||
|
"cancel": "Cancel",
|
||||||
|
"edit": "Edit",
|
||||||
|
"remove": "Remove",
|
||||||
|
"delete": "Delete",
|
||||||
|
"settings": "Settings",
|
||||||
|
"selectFile": "Select File",
|
||||||
|
"config_saved_success": "Config saved successfully",
|
||||||
|
"config_saved_failed": "Failed to save config",
|
||||||
|
"config_saved_restart_success": "Config saved and service restarted successfully",
|
||||||
|
"config_saved_restart_failed": "Failed to save config and restart service"
|
||||||
|
},
|
||||||
|
"login": {
|
||||||
|
"title": "Sign in to your account",
|
||||||
|
"description": "Enter your API key to access the configuration panel",
|
||||||
|
"apiKey": "API Key",
|
||||||
|
"apiKeyPlaceholder": "Enter your API key",
|
||||||
|
"signIn": "Sign In",
|
||||||
|
"invalidApiKey": "Invalid API key",
|
||||||
|
"configError": "Configuration not loaded",
|
||||||
|
"validating": "Validating API key..."
|
||||||
|
},
|
||||||
|
"toplevel": {
|
||||||
|
"title": "General Settings",
|
||||||
|
"log": "Enable Logging",
|
||||||
|
"claude_path": "Claude Path",
|
||||||
|
"host": "Host",
|
||||||
|
"port": "Port",
|
||||||
|
"apikey": "API Key"
|
||||||
|
},
|
||||||
|
"transformers": {
|
||||||
|
"title": "Custom Transformers",
|
||||||
|
"path": "Path",
|
||||||
|
"project": "Project",
|
||||||
|
"remove": "Remove",
|
||||||
|
"add": "Add Custom Transformer",
|
||||||
|
"edit": "Edit Custom Transformer",
|
||||||
|
"delete": "Delete Custom Transformer",
|
||||||
|
"delete_transformer_confirm": "Are you sure you want to delete this custom transformer?",
|
||||||
|
"parameters": "Parameters"
|
||||||
|
},
|
||||||
|
"providers": {
|
||||||
|
"title": "Providers",
|
||||||
|
"name": "Name",
|
||||||
|
"api_base_url": "API Base URL",
|
||||||
|
"api_key": "API Key",
|
||||||
|
"models": "Models",
|
||||||
|
"models_placeholder": "Enter model name and press Enter to add",
|
||||||
|
"add_model": "Add Model",
|
||||||
|
"select_models": "Select Models",
|
||||||
|
"remove": "Remove",
|
||||||
|
"add": "Add Provider",
|
||||||
|
"edit": "Edit Provider",
|
||||||
|
"delete": "Delete",
|
||||||
|
"cancel": "Cancel",
|
||||||
|
"delete_provider_confirm": "Are you sure you want to delete this provider?",
|
||||||
|
"test_connectivity": "Test Connectivity",
|
||||||
|
"testing": "Testing...",
|
||||||
|
"connection_successful": "Connection successful!",
|
||||||
|
"connection_failed": "Connection failed!",
|
||||||
|
"missing_credentials": "Missing API base URL or API key",
|
||||||
|
"fetch_available_models": "Fetch available models",
|
||||||
|
"fetching_models": "Fetching models...",
|
||||||
|
"fetch_models_failed": "Failed to fetch models",
|
||||||
|
"transformers": "Transformers",
|
||||||
|
"select_transformer": "Select Transformer",
|
||||||
|
"no_transformers": "No transformers available",
|
||||||
|
"provider_transformer": "Provider Transformer",
|
||||||
|
"model_transformers": "Model Transformers",
|
||||||
|
"transformer_parameters": "Transformer Parameters",
|
||||||
|
"add_parameter": "Add Parameter",
|
||||||
|
"parameter_name": "Parameter Name",
|
||||||
|
"parameter_value": "Parameter Value",
|
||||||
|
"selected_transformers": "Selected Transformers"
|
||||||
|
},
|
||||||
|
"router": {
|
||||||
|
"title": "Router",
|
||||||
|
"default": "Default",
|
||||||
|
"background": "Background",
|
||||||
|
"think": "Think",
|
||||||
|
"longContext": "Long Context",
|
||||||
|
"webSearch": "Web Search",
|
||||||
|
"selectModel": "Select a model...",
|
||||||
|
"searchModel": "Search model...",
|
||||||
|
"noModelFound": "No model found."
|
||||||
|
},
|
||||||
|
"json_editor": {
|
||||||
|
"title": "JSON Editor",
|
||||||
|
"save": "Save",
|
||||||
|
"saving": "Saving...",
|
||||||
|
"cancel": "Cancel",
|
||||||
|
"save_failed": "Failed to save config",
|
||||||
|
"save_and_restart": "Save & Restart"
|
||||||
|
}
|
||||||
|
}
|
||||||
99
ui/src/locales/zh.json
Normal file
@@ -0,0 +1,99 @@
|
|||||||
|
{
|
||||||
|
"app": {
|
||||||
|
"title": "Claude Code Router",
|
||||||
|
"save": "保存",
|
||||||
|
"save_and_restart": "保存并重启",
|
||||||
|
"cancel": "取消",
|
||||||
|
"edit": "编辑",
|
||||||
|
"remove": "移除",
|
||||||
|
"delete": "删除",
|
||||||
|
"settings": "设置",
|
||||||
|
"selectFile": "选择文件",
|
||||||
|
"config_saved_success": "配置保存成功",
|
||||||
|
"config_saved_failed": "配置保存失败",
|
||||||
|
"config_saved_restart_success": "配置保存并服务重启成功",
|
||||||
|
"config_saved_restart_failed": "配置保存并服务重启失败"
|
||||||
|
},
|
||||||
|
"login": {
|
||||||
|
"title": "登录到您的账户",
|
||||||
|
"description": "请输入您的API密钥以访问配置面板",
|
||||||
|
"apiKey": "API密钥",
|
||||||
|
"apiKeyPlaceholder": "请输入您的API密钥",
|
||||||
|
"signIn": "登录",
|
||||||
|
"invalidApiKey": "API密钥无效",
|
||||||
|
"configError": "配置未加载",
|
||||||
|
"validating": "正在验证API密钥..."
|
||||||
|
},
|
||||||
|
"toplevel": {
|
||||||
|
"title": "通用设置",
|
||||||
|
"log": "启用日志",
|
||||||
|
"claude_path": "Claude 路径",
|
||||||
|
"host": "主机",
|
||||||
|
"port": "端口",
|
||||||
|
"apikey": "API 密钥"
|
||||||
|
},
|
||||||
|
"transformers": {
|
||||||
|
"title": "自定义转换器",
|
||||||
|
"path": "路径",
|
||||||
|
"project": "项目",
|
||||||
|
"remove": "移除",
|
||||||
|
"add": "添加自定义转换器",
|
||||||
|
"edit": "编辑自定义转换器",
|
||||||
|
"delete": "删除自定义转换器",
|
||||||
|
"delete_transformer_confirm": "您确定要删除此自定义转换器吗?",
|
||||||
|
"parameters": "参数"
|
||||||
|
},
|
||||||
|
"providers": {
|
||||||
|
"title": "供应商",
|
||||||
|
"name": "名称",
|
||||||
|
"api_base_url": "API 基础地址",
|
||||||
|
"api_key": "API 密钥",
|
||||||
|
"models": "模型",
|
||||||
|
"models_placeholder": "输入模型名称并按回车键添加",
|
||||||
|
"add_model": "添加模型",
|
||||||
|
"select_models": "选择模型",
|
||||||
|
"remove": "移除",
|
||||||
|
"add": "添加供应商",
|
||||||
|
"edit": "编辑供应商",
|
||||||
|
"delete": "删除",
|
||||||
|
"cancel": "取消",
|
||||||
|
"delete_provider_confirm": "您确定要删除此供应商吗?",
|
||||||
|
"test_connectivity": "测试连通性",
|
||||||
|
"testing": "测试中...",
|
||||||
|
"connection_successful": "连接成功!",
|
||||||
|
"connection_failed": "连接失败!",
|
||||||
|
"missing_credentials": "缺少 API 基础地址或 API 密钥",
|
||||||
|
"fetch_available_models": "获取可用模型",
|
||||||
|
"fetching_models": "获取模型中...",
|
||||||
|
"fetch_models_failed": "获取模型失败",
|
||||||
|
"transformers": "转换器",
|
||||||
|
"select_transformer": "选择转换器",
|
||||||
|
"no_transformers": "无可用转换器",
|
||||||
|
"provider_transformer": "供应商转换器",
|
||||||
|
"model_transformers": "模型转换器",
|
||||||
|
"transformer_parameters": "转换器参数",
|
||||||
|
"add_parameter": "添加参数",
|
||||||
|
"parameter_name": "参数名称",
|
||||||
|
"parameter_value": "参数值",
|
||||||
|
"selected_transformers": "已选转换器"
|
||||||
|
},
|
||||||
|
"router": {
|
||||||
|
"title": "路由",
|
||||||
|
"default": "默认",
|
||||||
|
"background": "后台",
|
||||||
|
"think": "思考",
|
||||||
|
"longContext": "长上下文",
|
||||||
|
"webSearch": "网络搜索",
|
||||||
|
"selectModel": "选择一个模型...",
|
||||||
|
"searchModel": "搜索模型...",
|
||||||
|
"noModelFound": "未找到模型."
|
||||||
|
},
|
||||||
|
"json_editor": {
|
||||||
|
"title": "JSON 编辑器",
|
||||||
|
"save": "保存",
|
||||||
|
"saving": "保存中...",
|
||||||
|
"cancel": "取消",
|
||||||
|
"save_failed": "配置保存失败",
|
||||||
|
"save_and_restart": "保存并重启"
|
||||||
|
}
|
||||||
|
}
|
||||||
15
ui/src/main.tsx
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import './i18n';
|
||||||
|
import { StrictMode } from 'react'
|
||||||
|
import { createRoot } from 'react-dom/client'
|
||||||
|
import './index.css'
|
||||||
|
import { RouterProvider } from 'react-router-dom';
|
||||||
|
import { router } from './routes';
|
||||||
|
import { ConfigProvider } from '@/components/ConfigProvider';
|
||||||
|
|
||||||
|
createRoot(document.getElementById('root')!).render(
|
||||||
|
<StrictMode>
|
||||||
|
<ConfigProvider>
|
||||||
|
<RouterProvider router={router} />
|
||||||
|
</ConfigProvider>
|
||||||
|
</StrictMode>,
|
||||||
|
)
|
||||||
32
ui/src/routes.tsx
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import { createMemoryRouter, Navigate } from 'react-router-dom';
|
||||||
|
import App from './App';
|
||||||
|
import { Login } from '@/components/Login';
|
||||||
|
|
||||||
|
const ProtectedRoute = ({ children }: { children: React.ReactNode }) => {
|
||||||
|
// For this application, we allow access without an API key
|
||||||
|
// The App component will handle loading and error states
|
||||||
|
return children;
|
||||||
|
};
|
||||||
|
|
||||||
|
const PublicRoute = ({ children }: { children: React.ReactNode }) => {
|
||||||
|
// Always show login page
|
||||||
|
// The login page will handle empty API keys appropriately
|
||||||
|
return children;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const router = createMemoryRouter([
|
||||||
|
{
|
||||||
|
path: '/',
|
||||||
|
element: <Navigate to="/dashboard" replace />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
element: <PublicRoute><Login /></PublicRoute>,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/dashboard',
|
||||||
|
element: <ProtectedRoute><App /></ProtectedRoute>,
|
||||||
|
},
|
||||||
|
], {
|
||||||
|
initialEntries: ['/dashboard']
|
||||||
|
});
|
||||||
48
ui/src/styles/animations.css
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
@keyframes fadeIn {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateY(10px);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateY(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes scaleIn {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: scale(0.95);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: scale(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes slideIn {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: translateX(20px);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: translateX(0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.animate-fade-in {
|
||||||
|
animation: fadeIn 0.2s ease-out forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
.animate-scale-in {
|
||||||
|
animation: scaleIn 0.2s cubic-bezier(0.34, 1.56, 0.64, 1) forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
.animate-slide-in {
|
||||||
|
animation: slideIn 0.3s cubic-bezier(0.34, 1.56, 0.64, 1) forwards;
|
||||||
|
}
|
||||||
|
|
||||||
|
.transition-all-ease {
|
||||||
|
transition: all 0.2s cubic-bezier(0.34, 1.56, 0.64, 1);
|
||||||
|
}
|
||||||
1
ui/src/vite-env.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
/// <reference types="vite/client" />
|
||||||
26
ui/tsconfig.app.json
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "ES2022",
|
||||||
|
"useDefineForClassFields": true,
|
||||||
|
"lib": ["ES2022", "DOM", "DOM.Iterable"],
|
||||||
|
"module": "ESNext",
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"moduleResolution": "bundler",
|
||||||
|
"allowImportingTsExtensions": true,
|
||||||
|
"verbatimModuleSyntax": true,
|
||||||
|
"moduleDetection": "force",
|
||||||
|
"noEmit": true,
|
||||||
|
"jsx": "react-jsx",
|
||||||
|
"strict": true,
|
||||||
|
"noUnusedLocals": true,
|
||||||
|
"noUnusedParameters": true,
|
||||||
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
"baseUrl": ".",
|
||||||
|
"paths": {
|
||||||
|
"@/*": [
|
||||||
|
"./src/*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"include": ["src"]
|
||||||
|
}
|
||||||
26
ui/tsconfig.json
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
{
|
||||||
|
"compilerOptions": {
|
||||||
|
"target": "ES2022",
|
||||||
|
"useDefineForClassFields": true,
|
||||||
|
"lib": ["ES2022", "DOM", "DOM.Iterable"],
|
||||||
|
"module": "ESNext",
|
||||||
|
"skipLibCheck": true,
|
||||||
|
"moduleResolution": "bundler",
|
||||||
|
"allowImportingTsExtensions": true,
|
||||||
|
"verbatimModuleSyntax": true,
|
||||||
|
"moduleDetection": "force",
|
||||||
|
"noEmit": true,
|
||||||
|
"jsx": "react-jsx",
|
||||||
|
"strict": true,
|
||||||
|
"noUnusedLocals": true,
|
||||||
|
"noUnusedParameters": true,
|
||||||
|
"noFallthroughCasesInSwitch": true,
|
||||||
|
"baseUrl": ".",
|
||||||
|
"paths": {
|
||||||
|
"@/*": [
|
||||||
|
"./src/*"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"include": ["src"]
|
||||||
|
}
|
||||||
1
ui/tsconfig.tsbuildinfo
Normal file
@@ -0,0 +1 @@
|
|||||||
|
{"root":["./src/app.tsx","./src/i18n.ts","./src/main.tsx","./src/routes.tsx","./src/vite-env.d.ts","./src/components/configprovider.tsx","./src/components/jsoneditor.tsx","./src/components/login.tsx","./src/components/providerlist.tsx","./src/components/providers.tsx","./src/components/router.tsx","./src/components/settingsdialog.tsx","./src/components/transformerlist.tsx","./src/components/transformers.tsx","./src/components/ui/badge.tsx","./src/components/ui/button.tsx","./src/components/ui/card.tsx","./src/components/ui/combo-input.tsx","./src/components/ui/combobox.tsx","./src/components/ui/command.tsx","./src/components/ui/dialog.tsx","./src/components/ui/input.tsx","./src/components/ui/label.tsx","./src/components/ui/multi-combobox.tsx","./src/components/ui/popover.tsx","./src/components/ui/switch.tsx","./src/components/ui/toast.tsx","./src/lib/api.ts","./src/lib/utils.ts"],"version":"5.8.3"}
|
||||||
16
ui/vite.config.ts
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import path from "path"
|
||||||
|
import react from "@vitejs/plugin-react"
|
||||||
|
import { defineConfig } from "vite"
|
||||||
|
import { viteSingleFile } from "vite-plugin-singlefile"
|
||||||
|
import tailwindcss from "@tailwindcss/vite"
|
||||||
|
|
||||||
|
|
||||||
|
export default defineConfig({
|
||||||
|
base: './',
|
||||||
|
plugins: [react(), tailwindcss(), viteSingleFile()],
|
||||||
|
resolve: {
|
||||||
|
alias: {
|
||||||
|
"@": path.resolve(__dirname, "./src"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||