together-ai-mcp
A Node.js Model Context Protocol (MCP) server that exposes Together AI's inference endpoints — chat completions, image generation, vision, and embeddings — as tools callable from Claude Desktop, Cursor, VS Code, and any other MCP-compatible client.
Why this exists
I created this MCP due to several issues I was having accessing models through Together AI.
1. Reasoning model silent empty responses
Together AI's largest reasoning models (GLM-5, Qwen3.5-397B, MiniMax M2.5, Kimi K2.5) write their chain-of-thought into non-standard response fields, and they exhaust the OpenAI SDK's default token budget before producing a final answer.
Two problems compound each other:
Token budget exhaustion. The OpenAI SDK sets a default max_tokens of 2048. For reasoning models, this budget is consumed entirely by the thinking phase — message.content is never populated. You get charged for tokens, no error is raised, and the response is silently empty.
Fragmented response fields. Different model families on Together AI write their output to different fields:
| Field | Used by |
|---|---|
message.content | Standard models; Qwen (inline <think> tags) |
message.reasoning_content | DeepSeek-style format |
message.reasoning | Together AI format (GLM-5, MiniMax, Kimi) |
Any code that only reads message.content — or even message.content \|\| message.reasoning — silently returns an empty string for some models.
// Broken — misses reasoning_content (DeepSeek format):
const text = message.content || message.reasoning || '';
// Fixed — covers all Together AI reasoning model formats:
const text = message.content || message.reasoning_content || message.reasoning || '';
The default max_tokens is raised to 8192 to give reasoning models enough budget to complete their chain of thought before producing a final answer.
2. Vision model failures
Using the OpenAI SDK's chat.completions.create() for vision requests fails silently against Together AI's vision API. Together AI requires stream: false to be set explicitly; the SDK may not send it. When it does fail, the SDK error contains no response body, making the root cause invisible.
// Broken — SDK may omit stream:false; errors are opaque:
const response = await openai.chat.completions.create({ model, messages });
// Fixed — raw fetch, explicit stream:false, full error body in exception:
const response = await fetch('https://api.together.xyz/v1/chat/completions', {
method: 'POST',
headers: { Authorization: `Bearer ${apiKey}`, 'Content-Type': 'application/json' },
body: JSON.stringify({ model, messages, max_tokens, stream: false }),
});
if (!response.ok) {
const body = await response.text();
throw new Error(`Vision API error ${response.status}: ${body.slice(0, 200)}`);
}
Features
- Chat completions — any Together AI text or reasoning model, with full prompt and multi-turn message support
- Reasoning model support — correctly handles GLM-5, Qwen3.5-397B, MiniMax M2.5, Kimi K2.5 (see above)
- Image generation — FLUX.1-dev, FLUX.1-schnell, Stable Diffusion XL; images saved to disk
- Vision — analyse images via Llama 3.2 Vision or Qwen 2.5 VL
- Embeddings — generate vectors for RAG/retrieval pipelines via BGE and Snowflake Arctic models
Installation
Prerequisites
- Node.js 18+
- A Together AI API key — get one at api.together.ai
Setup
git clone https://github.com/your-username/together-ai-mcp
cd together-ai-mcp
npm install
cp .env.example .env
# Edit .env and add your TOGETHER_API_KEY
Add to Claude Desktop
Edit ~/Library/Application Support/Claude/claude_desktop_config.json:
{
"mcpServers": {
"together-ai": {
"command": "node",
"args": ["/absolute/path/to/together-ai-mcp/index.js"],
"env": {
"TOGETHER_API_KEY": "your_api_key_here",
"IMAGE_OUTPUT_DIR": "/path/to/save/images"
}
}
}
}
See examples/claude-config.md for Cursor and VS Code configuration.
Tools
together_chat
Call any Together AI chat or reasoning model.
| Parameter | Type | Default | Description |
|---|---|---|---|
model | string | meta-llama/Llama-3.3-70B-Instruct-Turbo | Model ID |
prompt | string | — | User message (use this OR messages) |
messages | array | — | Multi-turn [{role, content}] array |
system | string | — | System prompt (used with prompt only) |
temperature | number | 0.7 | 0.0–2.0 |
max_tokens | integer | 8192 | Raised from SDK default to give reasoning models enough budget for chain-of-thought |
together_generate_image
Generate images using FLUX or SDXL models.
| Parameter | Type | Default | Description |
|---|---|---|---|
prompt | string | required | Image description |
model | string | black-forest-labs/FLUX.1-schnell | Model ID |
width | integer | 1024 | Image width in pixels |
height | integer | 1024 | Image height in pixels |
steps | integer | 4 | Diffusion steps |
n | integer | 1 | Number of images |
negative_prompt | string | — | What to exclude |
Images are saved as PNG files to IMAGE_OUTPUT_DIR.
Note: Image generation uses a direct
fetchcall rather than the OpenAI SDK'simages.generate()because the SDK strips custom parameters likestepswhen calling Together AI's endpoint.
together_vision
Analyse an image using a vision model.
| Parameter | Type | Default | Description |
|---|---|---|---|
prompt | string | required | Question or instruction |
model | string | meta-llama/Llama-3.2-11B-Vision-Instruct | Model ID |
image_url | string | — | Public image URL |
image_path | string | — | Local file path (converted to base64) |
max_tokens | integer | 1024 | Max response length |
together_embed
Generate text embeddings for RAG and retrieval pipelines.
| Parameter | Type | Default | Description |
|---|---|---|---|
input | string | string[] | required | Text to embed |
model | string | BAAI/bge-large-en-v1.5 | Embedding model ID |
Models
The server works with any model available on Together AI's serverless API — just pass its model ID. No configuration changes are needed.
The tables below list the models I personally use. They are provided as a reference, not as a hard limit.
Finding model IDs
Browse all available models at api.together.ai/models. Each model's page shows its exact ID string. Pass that ID as the model parameter to any tool:
{
"tool": "together_chat",
"params": {
"model": "any-model-id-from-together-ai",
"prompt": "Hello"
}
}
The only constraint is that image generation models must be called via together_generate_image, vision models via together_vision, and embedding models via together_embed — you cannot call an image model through together_chat.
Dedicated endpoints: Some models (e.g.
meta-llama/Llama-4-Maverick-17B-128E-Instruct-FP8) require a dedicated endpoint rather than the serverless API. Calling these via this server will return a 400 error from Together AI.
Models I use
Chat / Reasoning
| Model | ID | Notes |
|---|---|---|
| Llama 3.3 70B | meta-llama/Llama-3.3-70B-Instruct-Turbo | Default — fast general-purpose |
| DeepSeek V3 | deepseek-ai/DeepSeek-V3 | Strong at code and reasoning |
| DeepSeek R1 | deepseek-ai/DeepSeek-R1 | Reasoning model |
| GLM-5 (744B) | zai-org/GLM-5 | Reasoning model — requires fix above |
| Qwen3.5 397B | Qwen/Qwen3.5-397B-A17B | Reasoning model — requires fix above |
| MiniMax M2.5 | MiniMaxAI/MiniMax-M2.5 | Reasoning model — requires fix above |
| Kimi K2.5 | moonshotai/Kimi-K2.5 | Reasoning model — requires fix above |
| Qwen 2.5 7B | Qwen/Qwen2.5-7B-Instruct-Turbo | Lightweight / low cost |
Image generation
| Model | ID |
|---|---|
| FLUX.1-schnell | black-forest-labs/FLUX.1-schnell |
| FLUX.1-dev | black-forest-labs/FLUX.1-dev |
| Stable Diffusion XL | stabilityai/stable-diffusion-xl-base-1.0 |
Vision
| Model | ID |
|---|---|
| Llama 3.2 11B Vision | meta-llama/Llama-3.2-11B-Vision-Instruct |
| Qwen 2.5 VL 72B | Qwen/Qwen2.5-VL-72B-Instruct |
Embeddings
| Model | ID |
|---|---|
| BGE Large | BAAI/bge-large-en-v1.5 |
| M2-BERT 32K | togethercomputer/m2-bert-80M-32k-retrieval |
| Snowflake Arctic | Snowflake/snowflake-arctic-embed-m |
Running tests
npm test
The test suite uses Node.js's built-in test runner and mocks all external dependencies — no API key required to run tests.
Project structure
together-ai-mcp/
├── index.js # MCP server and handler logic
├── package.json
├── .env.example
├── test/
│ └── index.test.js # Full test suite (node:test, no external framework)
└── examples/
├── chat.md # Example prompts for each tool and model
└── claude-config.md # Configuration for Claude Desktop, Cursor, VS Code
Dependencies
@modelcontextprotocol/sdk— MCP server frameworkopenai— OpenAI-compatible client used with Together AI'sbaseURL
License
MIT
サーバー設定
{
"mcpServers": {
"together-ai": {
"command": "node",
"args": [
"/absolute/path/to/together-ai-mcp/index.js"
],
"env": {
"TOGETHER_API_KEY": "your_api_key_here",
"IMAGE_OUTPUT_DIR": "/path/to/save/images"
}
}
}
}