1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
| // lib/ai/client.ts
import Anthropic from "@anthropic-ai/sdk";
import OpenAI from "openai";
const anthropic = new Anthropic();
const openai = new OpenAI();
type AIProvider = "anthropic" | "openai";
interface GenerateOptions {
prompt: string;
maxTokens?: number;
temperature?: number;
}
export async function generate(
options: GenerateOptions,
preferredProvider: AIProvider = "anthropic"
): Promise<string> {
const { prompt, maxTokens = 1000, temperature = 0.7 } = options;
try {
if (preferredProvider === "anthropic") {
return await generateWithAnthropic(prompt, maxTokens, temperature);
}
return await generateWithOpenAI(prompt, maxTokens, temperature);
} catch (error) {
console.error(`${preferredProvider} failed, trying fallback`);
// Fallback to other provider
const fallback = preferredProvider === "anthropic" ? "openai" : "anthropic";
try {
if (fallback === "anthropic") {
return await generateWithAnthropic(prompt, maxTokens, temperature);
}
return await generateWithOpenAI(prompt, maxTokens, temperature);
} catch (fallbackError) {
throw new Error("All AI providers failed");
}
}
}
async function generateWithAnthropic(
prompt: string,
maxTokens: number,
temperature: number
): Promise<string> {
const response = await anthropic.messages.create({
model: "claude-sonnet-4-20250514",
max_tokens: maxTokens,
temperature,
messages: [{ role: "user", content: prompt }]
});
return response.content[0].type === "text"
? response.content[0].text
: "";
}
async function generateWithOpenAI(
prompt: string,
maxTokens: number,
temperature: number
): Promise<string> {
const response = await openai.chat.completions.create({
model: "gpt-4",
max_tokens: maxTokens,
temperature,
messages: [{ role: "user", content: prompt }]
});
return response.choices[0].message.content || "";
}
|