diff --git a/docs/providers/index.md b/docs/providers/index.md index 21aaff7ed33..fdf67c9ec53 100644 --- a/docs/providers/index.md +++ b/docs/providers/index.md @@ -40,6 +40,7 @@ See [Venice AI](/providers/venice). - [Qwen (OAuth)](/providers/qwen) - [OpenRouter](/providers/openrouter) - [Vercel AI Gateway](/providers/vercel-ai-gateway) +- [Together AI](/providers/together) - [Cloudflare AI Gateway](/providers/cloudflare-ai-gateway) - [Moonshot AI (Kimi + Kimi Coding)](/providers/moonshot) - [OpenCode Zen](/providers/opencode) diff --git a/docs/providers/together.md b/docs/providers/together.md new file mode 100644 index 00000000000..a81c75e334f --- /dev/null +++ b/docs/providers/together.md @@ -0,0 +1,65 @@ +--- +summary: "Together AI setup (auth + model selection)" +read_when: + - You want to use Together AI with OpenClaw + - You need the API key env var or CLI auth choice +--- + +# Together AI + +The [Together AI](https://together.ai) provides access to leading open-source models including Llama, DeepSeek, Kimi, and more through a unified API. + +- Provider: `together` +- Auth: `TOGETHER_API_KEY` +- API: OpenAI-compatible + +## Quick start + +1. Set the API key (recommended: store it for the Gateway): + +```bash +openclaw onboard --auth-choice together-api-key +``` + +2. Set a default model: + +```json5 +{ + agents: { + defaults: { + model: { primary: "together/zai-org/GLM-4.7" }, + }, + }, +} +``` + +## Non-interactive example + +```bash +openclaw onboard --non-interactive \ + --mode local \ + --auth-choice together-api-key \ + --together-api-key "$TOGETHER_API_KEY" +``` + +This will set `together/zai-org/GLM-4.7` as the default model. + +## Environment note + +If the Gateway runs as a daemon (launchd/systemd), make sure `TOGETHER_API_KEY` +is available to that process (for example, in `~/.clawdbot/.env` or via +`env.shellEnv`). + +## Available models + +Together AI provides access to many popular open-source models: + +- **GLM 4.7 Fp8** - Default model with 200K context window +- **Llama 3.3 70B Instruct Turbo** - Fast, efficient instruction following +- **Llama 4 Scout** - Vision model with image understanding +- **Llama 4 Maverick** - Advanced vision and reasoning +- **DeepSeek V3.1** - Powerful coding and reasoning model +- **DeepSeek R1** - Advanced reasoning model +- **Kimi K2 Instruct** - High-performance model with 262K context window + +All models support standard chat completions and are OpenAI API compatible. diff --git a/src/agents/model-auth.ts b/src/agents/model-auth.ts index d363ce96267..08fbefb682a 100644 --- a/src/agents/model-auth.ts +++ b/src/agents/model-auth.ts @@ -305,6 +305,7 @@ export function resolveEnvApiKey(provider: string): EnvApiKeyResult | null { venice: "VENICE_API_KEY", mistral: "MISTRAL_API_KEY", opencode: "OPENCODE_API_KEY", + together: "TOGETHER_API_KEY", qianfan: "QIANFAN_API_KEY", ollama: "OLLAMA_API_KEY", }; diff --git a/src/agents/models-config.providers.ts b/src/agents/models-config.providers.ts index d4ae66cc038..f5723c53b0c 100644 --- a/src/agents/models-config.providers.ts +++ b/src/agents/models-config.providers.ts @@ -16,6 +16,11 @@ import { SYNTHETIC_BASE_URL, SYNTHETIC_MODEL_CATALOG, } from "./synthetic-models.js"; +import { + TOGETHER_BASE_URL, + TOGETHER_MODEL_CATALOG, + buildTogetherModelDefinition, +} from "./together-models.js"; import { discoverVeniceModels, VENICE_BASE_URL } from "./venice-models.js"; type ModelsConfig = NonNullable; @@ -414,6 +419,14 @@ async function buildOllamaProvider(): Promise { }; } +function buildTogetherProvider(): ProviderConfig { + return { + baseUrl: TOGETHER_BASE_URL, + api: "openai-completions", + models: TOGETHER_MODEL_CATALOG.map(buildTogetherModelDefinition), + }; +} + export function buildQianfanProvider(): ProviderConfig { return { baseUrl: QIANFAN_BASE_URL, @@ -536,6 +549,16 @@ export async function resolveImplicitProviders(params: { providers.ollama = { ...(await buildOllamaProvider()), apiKey: ollamaKey }; } + const togetherKey = + resolveEnvApiKeyVarName("together") ?? + resolveApiKeyFromProfiles({ provider: "together", store: authStore }); + if (togetherKey) { + providers.together = { + ...buildTogetherProvider(), + apiKey: togetherKey, + }; + } + const qianfanKey = resolveEnvApiKeyVarName("qianfan") ?? resolveApiKeyFromProfiles({ provider: "qianfan", store: authStore }); @@ -551,7 +574,9 @@ export async function resolveImplicitCopilotProvider(params: { env?: NodeJS.ProcessEnv; }): Promise { const env = params.env ?? process.env; - const authStore = ensureAuthProfileStore(params.agentDir, { allowKeychainPrompt: false }); + const authStore = ensureAuthProfileStore(params.agentDir, { + allowKeychainPrompt: false, + }); const hasProfile = listProfilesForProvider(authStore, "github-copilot").length > 0; const envToken = env.COPILOT_GITHUB_TOKEN ?? env.GH_TOKEN ?? env.GITHUB_TOKEN; const githubToken = (envToken ?? "").trim(); @@ -622,7 +647,10 @@ export async function resolveImplicitBedrockProvider(params: { } const region = discoveryConfig?.region ?? env.AWS_REGION ?? env.AWS_DEFAULT_REGION ?? "us-east-1"; - const models = await discoverBedrockModels({ region, config: discoveryConfig }); + const models = await discoverBedrockModels({ + region, + config: discoveryConfig, + }); if (models.length === 0) { return null; } diff --git a/src/agents/together-models.ts b/src/agents/together-models.ts new file mode 100644 index 00000000000..41608a9c86e --- /dev/null +++ b/src/agents/together-models.ts @@ -0,0 +1,133 @@ +import type { ModelDefinitionConfig } from "../config/types.models.js"; + +export const TOGETHER_BASE_URL = "https://api.together.xyz/v1"; + +export const TOGETHER_MODEL_CATALOG: ModelDefinitionConfig[] = [ + { + id: "zai-org/GLM-4.7", + name: "GLM 4.7 Fp8", + reasoning: false, + input: ["text"], + contextWindow: 202752, + maxTokens: 8192, + cost: { + input: 0.45, + output: 2.0, + cacheRead: 0.45, + cacheWrite: 2.0, + }, + }, + { + id: "moonshotai/Kimi-K2.5", + name: "Kimi K2.5", + reasoning: true, + input: ["text", "image"], + cost: { + input: 0.5, + output: 2.8, + cacheRead: 0.5, + cacheWrite: 2.8, + }, + contextWindow: 262144, + maxTokens: 32768, + }, + { + id: "meta-llama/Llama-3.3-70B-Instruct-Turbo", + name: "Llama 3.3 70B Instruct Turbo", + reasoning: false, + input: ["text"], + contextWindow: 131072, + maxTokens: 8192, + cost: { + input: 0.88, + output: 0.88, + cacheRead: 0.88, + cacheWrite: 0.88, + }, + }, + { + id: "meta-llama/Llama-4-Scout-17B-16E-Instruct", + name: "Llama 4 Scout 17B 16E Instruct", + reasoning: false, + input: ["text", "image"], + contextWindow: 10000000, + maxTokens: 32768, + cost: { + input: 0.18, + output: 0.59, + cacheRead: 0.18, + cacheWrite: 0.18, + }, + }, + { + id: "meta-llama/Llama-4-Maverick-17B-128E-Instruct-FP8", + name: "Llama 4 Maverick 17B 128E Instruct FP8", + reasoning: false, + input: ["text", "image"], + contextWindow: 20000000, + maxTokens: 32768, + cost: { + input: 0.27, + output: 0.85, + cacheRead: 0.27, + cacheWrite: 0.27, + }, + }, + { + id: "deepseek-ai/DeepSeek-V3.1", + name: "DeepSeek V3.1", + reasoning: false, + input: ["text"], + contextWindow: 131072, + maxTokens: 8192, + cost: { + input: 0.6, + output: 1.25, + cacheRead: 0.6, + cacheWrite: 0.6, + }, + }, + { + id: "deepseek-ai/DeepSeek-R1", + name: "DeepSeek R1", + reasoning: true, + input: ["text"], + contextWindow: 131072, + maxTokens: 8192, + cost: { + input: 3.0, + output: 7.0, + cacheRead: 3.0, + cacheWrite: 3.0, + }, + }, + { + id: "moonshotai/Kimi-K2-Instruct-0905", + name: "Kimi K2-Instruct 0905", + reasoning: false, + input: ["text"], + contextWindow: 262144, + maxTokens: 8192, + cost: { + input: 1.0, + output: 3.0, + cacheRead: 1.0, + cacheWrite: 3.0, + }, + }, +]; + +export function buildTogetherModelDefinition( + model: (typeof TOGETHER_MODEL_CATALOG)[number], +): ModelDefinitionConfig { + return { + id: model.id, + name: model.name, + api: "openai-completions", + reasoning: model.reasoning, + input: model.input, + cost: model.cost, + contextWindow: model.contextWindow, + maxTokens: model.maxTokens, + }; +} diff --git a/src/cli/program.smoke.test.ts b/src/cli/program.smoke.test.ts index 28e100e1e20..66fefef84c6 100644 --- a/src/cli/program.smoke.test.ts +++ b/src/cli/program.smoke.test.ts @@ -22,7 +22,9 @@ const runtime = { }), }; -vi.mock("./plugin-registry.js", () => ({ ensurePluginRegistryLoaded: () => undefined })); +vi.mock("./plugin-registry.js", () => ({ + ensurePluginRegistryLoaded: () => undefined, +})); vi.mock("../commands/message.js", () => ({ messageCommand })); vi.mock("../commands/status.js", () => ({ statusCommand })); @@ -42,7 +44,9 @@ vi.mock("../commands/configure.js", () => ({ })); vi.mock("../commands/setup.js", () => ({ setupCommand })); vi.mock("../commands/onboard.js", () => ({ onboardCommand })); -vi.mock("../commands/doctor-config-flow.js", () => ({ loadAndMaybeMigrateDoctorConfig })); +vi.mock("../commands/doctor-config-flow.js", () => ({ + loadAndMaybeMigrateDoctorConfig, +})); vi.mock("../runtime.js", () => ({ defaultRuntime: runtime })); vi.mock("./channel-auth.js", () => ({ runChannelLogin, runChannelLogout })); vi.mock("../tui/tui.js", () => ({ runTui })); @@ -174,6 +178,12 @@ describe("cli program (smoke)", () => { key: "sk-moonshot-test", field: "moonshotApiKey", }, + { + authChoice: "together-api-key", + flag: "--together-api-key", + key: "sk-together-test", + field: "togetherApiKey", + }, { authChoice: "moonshot-api-key-cn", flag: "--moonshot-api-key", diff --git a/src/cli/program/register.onboard.ts b/src/cli/program/register.onboard.ts index 0ddaeb55e70..3c2e842fa2d 100644 --- a/src/cli/program/register.onboard.ts +++ b/src/cli/program/register.onboard.ts @@ -58,7 +58,7 @@ export function registerOnboardCommand(program: Command) { .option("--mode ", "Wizard mode: local|remote") .option( "--auth-choice ", - "Auth: setup-token|token|chutes|openai-codex|openai-api-key|xai-api-key|qianfan-api-key|openrouter-api-key|ai-gateway-api-key|cloudflare-ai-gateway-api-key|moonshot-api-key|moonshot-api-key-cn|kimi-code-api-key|synthetic-api-key|venice-api-key|gemini-api-key|zai-api-key|xiaomi-api-key|apiKey|minimax-api|minimax-api-lightning|opencode-zen|skip", + "Auth: setup-token|token|chutes|openai-codex|openai-api-key|xai-api-key|qianfan-api-key|openrouter-api-key|ai-gateway-api-key|cloudflare-ai-gateway-api-key|moonshot-api-key|moonshot-api-key-cn|kimi-code-api-key|synthetic-api-key|venice-api-key|gemini-api-key|zai-api-key|xiaomi-api-key|apiKey|minimax-api|minimax-api-lightning|opencode-zen|skip|together-api-key", ) .option( "--token-provider ", @@ -85,6 +85,7 @@ export function registerOnboardCommand(program: Command) { .option("--minimax-api-key ", "MiniMax API key") .option("--synthetic-api-key ", "Synthetic API key") .option("--venice-api-key ", "Venice API key") + .option("--together-api-key ", "Together AI API key") .option("--opencode-zen-api-key ", "OpenCode Zen API key") .option("--xai-api-key ", "xAI API key") .option("--qianfan-api-key ", "QIANFAN API key") @@ -142,6 +143,7 @@ export function registerOnboardCommand(program: Command) { minimaxApiKey: opts.minimaxApiKey as string | undefined, syntheticApiKey: opts.syntheticApiKey as string | undefined, veniceApiKey: opts.veniceApiKey as string | undefined, + togetherApiKey: opts.togetherApiKey as string | undefined, opencodeZenApiKey: opts.opencodeZenApiKey as string | undefined, xaiApiKey: opts.xaiApiKey as string | undefined, gatewayPort: diff --git a/src/commands/auth-choice-options.test.ts b/src/commands/auth-choice-options.test.ts index c0608f1ec53..2e593a09739 100644 --- a/src/commands/auth-choice-options.test.ts +++ b/src/commands/auth-choice-options.test.ts @@ -63,6 +63,7 @@ describe("buildAuthChoiceOptions", () => { expect(options.some((opt) => opt.value === "moonshot-api-key")).toBe(true); expect(options.some((opt) => opt.value === "moonshot-api-key-cn")).toBe(true); expect(options.some((opt) => opt.value === "kimi-code-api-key")).toBe(true); + expect(options.some((opt) => opt.value === "together-api-key")).toBe(true); }); it("includes Vercel AI Gateway auth choice", () => { @@ -81,10 +82,19 @@ describe("buildAuthChoiceOptions", () => { store, includeSkip: false, }); - expect(options.some((opt) => opt.value === "cloudflare-ai-gateway-api-key")).toBe(true); }); + it("includes Together AI auth choice", () => { + const store: AuthProfileStore = { version: 1, profiles: {} }; + const options = buildAuthChoiceOptions({ + store, + includeSkip: false, + }); + + expect(options.some((opt) => opt.value === "together-api-key")).toBe(true); + }); + it("includes Synthetic auth choice", () => { const store: AuthProfileStore = { version: 1, profiles: {} }; const options = buildAuthChoiceOptions({ diff --git a/src/commands/auth-choice-options.ts b/src/commands/auth-choice-options.ts index 7208febc83f..3840aecc312 100644 --- a/src/commands/auth-choice-options.ts +++ b/src/commands/auth-choice-options.ts @@ -23,6 +23,7 @@ export type AuthChoiceGroupId = | "synthetic" | "venice" | "qwen" + | "together" | "qianfan" | "xai"; @@ -129,6 +130,12 @@ const AUTH_CHOICE_GROUP_DEFS: { hint: "Anthropic-compatible (multi-model)", choices: ["synthetic-api-key"], }, + { + value: "together", + label: "Together AI", + hint: "API key", + choices: ["together-api-key"], + }, { value: "venice", label: "Venice AI", @@ -185,13 +192,21 @@ export function buildAuthChoiceOptions(params: { value: "moonshot-api-key-cn", label: "Kimi API key (.cn)", }); - options.push({ value: "kimi-code-api-key", label: "Kimi Code API key (subscription)" }); + options.push({ + value: "kimi-code-api-key", + label: "Kimi Code API key (subscription)", + }); options.push({ value: "synthetic-api-key", label: "Synthetic API key" }); options.push({ value: "venice-api-key", label: "Venice AI API key", hint: "Privacy-focused inference (uncensored models)", }); + options.push({ + value: "together-api-key", + label: "Together AI API key", + hint: "Access to Llama, DeepSeek, Qwen, and more open models", + }); options.push({ value: "github-copilot", label: "GitHub Copilot (GitHub device login)", diff --git a/src/commands/auth-choice.apply.api-providers.ts b/src/commands/auth-choice.apply.api-providers.ts index 574128d6ace..cb506ee5dc6 100644 --- a/src/commands/auth-choice.apply.api-providers.ts +++ b/src/commands/auth-choice.apply.api-providers.ts @@ -29,6 +29,8 @@ import { applyOpenrouterProviderConfig, applySyntheticConfig, applySyntheticProviderConfig, + applyTogetherConfig, + applyTogetherProviderConfig, applyVeniceConfig, applyVeniceProviderConfig, applyVercelAiGatewayConfig, @@ -42,6 +44,7 @@ import { MOONSHOT_DEFAULT_MODEL_REF, OPENROUTER_DEFAULT_MODEL_REF, SYNTHETIC_DEFAULT_MODEL_REF, + TOGETHER_DEFAULT_MODEL_REF, VENICE_DEFAULT_MODEL_REF, VERCEL_AI_GATEWAY_DEFAULT_MODEL_REF, XIAOMI_DEFAULT_MODEL_REF, @@ -53,6 +56,7 @@ import { setOpencodeZenApiKey, setOpenrouterApiKey, setSyntheticApiKey, + setTogetherApiKey, setVeniceApiKey, setVercelAiGatewayApiKey, setXiaomiApiKey, @@ -106,6 +110,8 @@ export async function applyAuthChoiceApiProviders( authChoice = "synthetic-api-key"; } else if (params.opts.tokenProvider === "venice") { authChoice = "venice-api-key"; + } else if (params.opts.tokenProvider === "together") { + authChoice = "together-api-key"; } else if (params.opts.tokenProvider === "opencode") { authChoice = "opencode-zen"; } else if (params.opts.tokenProvider === "qianfan") { @@ -803,6 +809,64 @@ export async function applyAuthChoiceApiProviders( return { config: nextConfig, agentModelOverride }; } + if (authChoice === "together-api-key") { + let hasCredential = false; + + if (!hasCredential && params.opts?.token && params.opts?.tokenProvider === "together") { + await setTogetherApiKey(normalizeApiKeyInput(params.opts.token), params.agentDir); + hasCredential = true; + } + + if (!hasCredential) { + await params.prompter.note( + [ + "Together AI provides access to leading open-source models including Llama, DeepSeek, Qwen, and more.", + "Get your API key at: https://api.together.xyz/settings/api-keys", + ].join("\n"), + "Together AI", + ); + } + + const envKey = resolveEnvApiKey("together"); + if (envKey) { + const useExisting = await params.prompter.confirm({ + message: `Use existing TOGETHER_API_KEY (${envKey.source}, ${formatApiKeyPreview(envKey.apiKey)})?`, + initialValue: true, + }); + if (useExisting) { + await setTogetherApiKey(envKey.apiKey, params.agentDir); + hasCredential = true; + } + } + if (!hasCredential) { + const key = await params.prompter.text({ + message: "Enter Together AI API key", + validate: validateApiKeyInput, + }); + await setTogetherApiKey(normalizeApiKeyInput(String(key)), params.agentDir); + } + nextConfig = applyAuthProfileConfig(nextConfig, { + profileId: "together:default", + provider: "together", + mode: "api_key", + }); + { + const applied = await applyDefaultModelChoice({ + config: nextConfig, + setDefaultModel: params.setDefaultModel, + defaultModel: TOGETHER_DEFAULT_MODEL_REF, + applyDefaultConfig: applyTogetherConfig, + applyProviderConfig: applyTogetherProviderConfig, + noteDefault: TOGETHER_DEFAULT_MODEL_REF, + noteAgentModel, + prompter: params.prompter, + }); + nextConfig = applied.config; + agentModelOverride = applied.agentModelOverride ?? agentModelOverride; + } + return { config: nextConfig, agentModelOverride }; + } + if (authChoice === "qianfan-api-key") { let hasCredential = false; if (!hasCredential && params.opts?.token && params.opts?.tokenProvider === "qianfan") { diff --git a/src/commands/auth-choice.preferred-provider.ts b/src/commands/auth-choice.preferred-provider.ts index c77283b5072..c9820c46f80 100644 --- a/src/commands/auth-choice.preferred-provider.ts +++ b/src/commands/auth-choice.preferred-provider.ts @@ -23,6 +23,7 @@ const PREFERRED_PROVIDER_BY_AUTH_CHOICE: Partial> = { "xiaomi-api-key": "xiaomi", "synthetic-api-key": "synthetic", "venice-api-key": "venice", + "together-api-key": "together", "github-copilot": "github-copilot", "copilot-proxy": "copilot-proxy", "minimax-cloud": "minimax", diff --git a/src/commands/onboard-auth.config-core.ts b/src/commands/onboard-auth.config-core.ts index 131067fc95a..eafd295a621 100644 --- a/src/commands/onboard-auth.config-core.ts +++ b/src/commands/onboard-auth.config-core.ts @@ -16,6 +16,11 @@ import { SYNTHETIC_DEFAULT_MODEL_REF, SYNTHETIC_MODEL_CATALOG, } from "../agents/synthetic-models.js"; +import { + buildTogetherModelDefinition, + TOGETHER_BASE_URL, + TOGETHER_MODEL_CATALOG, +} from "../agents/together-models.js"; import { buildVeniceModelDefinition, VENICE_BASE_URL, @@ -25,6 +30,7 @@ import { import { CLOUDFLARE_AI_GATEWAY_DEFAULT_MODEL_REF, OPENROUTER_DEFAULT_MODEL_REF, + TOGETHER_DEFAULT_MODEL_REF, VERCEL_AI_GATEWAY_DEFAULT_MODEL_REF, XIAOMI_DEFAULT_MODEL_REF, ZAI_DEFAULT_MODEL_REF, @@ -600,6 +606,83 @@ export function applyVeniceConfig(cfg: OpenClawConfig): OpenClawConfig { }; } +/** + * Apply Together provider configuration without changing the default model. + * Registers Together models and sets up the provider, but preserves existing model selection. + */ +export function applyTogetherProviderConfig(cfg: OpenClawConfig): OpenClawConfig { + const models = { ...cfg.agents?.defaults?.models }; + models[TOGETHER_DEFAULT_MODEL_REF] = { + ...models[TOGETHER_DEFAULT_MODEL_REF], + alias: models[TOGETHER_DEFAULT_MODEL_REF]?.alias ?? "Together AI", + }; + + const providers = { ...cfg.models?.providers }; + const existingProvider = providers.together; + const existingModels = Array.isArray(existingProvider?.models) ? existingProvider.models : []; + const togetherModels = TOGETHER_MODEL_CATALOG.map(buildTogetherModelDefinition); + const mergedModels = [ + ...existingModels, + ...togetherModels.filter( + (model) => !existingModels.some((existing) => existing.id === model.id), + ), + ]; + const { apiKey: existingApiKey, ...existingProviderRest } = (existingProvider ?? {}) as Record< + string, + unknown + > as { apiKey?: string }; + const resolvedApiKey = typeof existingApiKey === "string" ? existingApiKey : undefined; + const normalizedApiKey = resolvedApiKey?.trim(); + providers.together = { + ...existingProviderRest, + baseUrl: TOGETHER_BASE_URL, + api: "openai-completions", + ...(normalizedApiKey ? { apiKey: normalizedApiKey } : {}), + models: mergedModels.length > 0 ? mergedModels : togetherModels, + }; + + return { + ...cfg, + agents: { + ...cfg.agents, + defaults: { + ...cfg.agents?.defaults, + models, + }, + }, + models: { + mode: cfg.models?.mode ?? "merge", + providers, + }, + }; +} + +/** + * Apply Together provider configuration AND set Together as the default model. + * Use this when Together is the primary provider choice during onboarding. + */ +export function applyTogetherConfig(cfg: OpenClawConfig): OpenClawConfig { + const next = applyTogetherProviderConfig(cfg); + const existingModel = next.agents?.defaults?.model; + return { + ...next, + agents: { + ...next.agents, + defaults: { + ...next.agents?.defaults, + model: { + ...(existingModel && "fallbacks" in (existingModel as Record) + ? { + fallbacks: (existingModel as { fallbacks?: string[] }).fallbacks, + } + : undefined), + primary: TOGETHER_DEFAULT_MODEL_REF, + }, + }, + }, + }; +} + export function applyXaiProviderConfig(cfg: OpenClawConfig): OpenClawConfig { const models = { ...cfg.agents?.defaults?.models }; models[XAI_DEFAULT_MODEL_REF] = { diff --git a/src/commands/onboard-auth.credentials.ts b/src/commands/onboard-auth.credentials.ts index 20784d34d0b..a9ddbe890af 100644 --- a/src/commands/onboard-auth.credentials.ts +++ b/src/commands/onboard-auth.credentials.ts @@ -118,6 +118,7 @@ export async function setVeniceApiKey(key: string, agentDir?: string) { export const ZAI_DEFAULT_MODEL_REF = "zai/glm-4.7"; export const XIAOMI_DEFAULT_MODEL_REF = "xiaomi/mimo-v2-flash"; export const OPENROUTER_DEFAULT_MODEL_REF = "openrouter/auto"; +export const TOGETHER_DEFAULT_MODEL_REF = "together/zai-org/GLM-4.7"; export const VERCEL_AI_GATEWAY_DEFAULT_MODEL_REF = "vercel-ai-gateway/anthropic/claude-opus-4.6"; export async function setZaiApiKey(key: string, agentDir?: string) { @@ -205,6 +206,18 @@ export async function setOpencodeZenApiKey(key: string, agentDir?: string) { }); } +export async function setTogetherApiKey(key: string, agentDir?: string) { + upsertAuthProfile({ + profileId: "together:default", + credential: { + type: "api_key", + provider: "together", + key, + }, + agentDir: resolveAuthAgentDir(agentDir), + }); +} + export function setQianfanApiKey(key: string, agentDir?: string) { upsertAuthProfile({ profileId: "qianfan:default", diff --git a/src/commands/onboard-auth.ts b/src/commands/onboard-auth.ts index a2732b7bfad..e89d9451ce9 100644 --- a/src/commands/onboard-auth.ts +++ b/src/commands/onboard-auth.ts @@ -19,15 +19,17 @@ export { applyOpenrouterProviderConfig, applySyntheticConfig, applySyntheticProviderConfig, + applyTogetherConfig, + applyTogetherProviderConfig, applyVeniceConfig, applyVeniceProviderConfig, applyVercelAiGatewayConfig, applyVercelAiGatewayProviderConfig, + applyXaiConfig, + applyXaiProviderConfig, applyXiaomiConfig, applyXiaomiProviderConfig, applyZaiConfig, - applyXaiConfig, - applyXaiProviderConfig, } from "./onboard-auth.config-core.js"; export { applyMinimaxApiConfig, @@ -55,6 +57,7 @@ export { setOpencodeZenApiKey, setOpenrouterApiKey, setSyntheticApiKey, + setTogetherApiKey, setVeniceApiKey, setVercelAiGatewayApiKey, setXiaomiApiKey, @@ -64,6 +67,7 @@ export { VERCEL_AI_GATEWAY_DEFAULT_MODEL_REF, XIAOMI_DEFAULT_MODEL_REF, ZAI_DEFAULT_MODEL_REF, + TOGETHER_DEFAULT_MODEL_REF, XAI_DEFAULT_MODEL_REF, } from "./onboard-auth.credentials.js"; export { diff --git a/src/commands/onboard-non-interactive/local/auth-choice.ts b/src/commands/onboard-non-interactive/local/auth-choice.ts index 4d757e01790..d29afab423e 100644 --- a/src/commands/onboard-non-interactive/local/auth-choice.ts +++ b/src/commands/onboard-non-interactive/local/auth-choice.ts @@ -22,6 +22,7 @@ import { applyOpenrouterConfig, applySyntheticConfig, applyVeniceConfig, + applyTogetherConfig, applyVercelAiGatewayConfig, applyXaiConfig, applyXiaomiConfig, @@ -38,6 +39,7 @@ import { setSyntheticApiKey, setXaiApiKey, setVeniceApiKey, + setTogetherApiKey, setVercelAiGatewayApiKey, setXiaomiApiKey, setZaiApiKey, @@ -544,6 +546,29 @@ export async function applyNonInteractiveAuthChoice(params: { return applyOpencodeZenConfig(nextConfig); } + if (authChoice === "together-api-key") { + const resolved = await resolveNonInteractiveApiKey({ + provider: "together", + cfg: baseConfig, + flagValue: opts.togetherApiKey, + flagName: "--together-api-key", + envVar: "TOGETHER_API_KEY", + runtime, + }); + if (!resolved) { + return null; + } + if (resolved.source !== "profile") { + await setTogetherApiKey(resolved.key); + } + nextConfig = applyAuthProfileConfig(nextConfig, { + profileId: "together:default", + provider: "together", + mode: "api_key", + }); + return applyTogetherConfig(nextConfig); + } + if ( authChoice === "oauth" || authChoice === "chutes" || diff --git a/src/commands/onboard-types.ts b/src/commands/onboard-types.ts index 488a4e9f5a8..9405795837e 100644 --- a/src/commands/onboard-types.ts +++ b/src/commands/onboard-types.ts @@ -19,6 +19,7 @@ export type AuthChoice = | "kimi-code-api-key" | "synthetic-api-key" | "venice-api-key" + | "together-api-key" | "codex-cli" | "apiKey" | "gemini-api-key" @@ -80,6 +81,7 @@ export type OnboardOptions = { minimaxApiKey?: string; syntheticApiKey?: string; veniceApiKey?: string; + togetherApiKey?: string; opencodeZenApiKey?: string; xaiApiKey?: string; qianfanApiKey?: string;