File size: 6,981 Bytes
55a034a 1a45d5d 55a034a 831f7e7 1a45d5d a64b653 1a45d5d 831f7e7 1a45d5d a64b653 1a45d5d 831f7e7 1a45d5d a64b653 1a45d5d 831f7e7 1a45d5d a64b653 1a45d5d 831f7e7 1a45d5d a64b653 1a45d5d 2d83648 1a45d5d 2d83648 1a45d5d 831f7e7 5bd3ab2 1a45d5d 5bd3ab2 55a034a 65676ec 5bd3ab2 65676ec 55a034a 65676ec 1a45d5d 5bd3ab2 1a45d5d 65676ec 1a45d5d 5bd3ab2 8725cc4 1a45d5d 8725cc4 1a45d5d 65676ec 1a45d5d 5bd3ab2 55a034a 5bd3ab2 65676ec 5bd3ab2 65676ec 1a45d5d 65676ec 1a45d5d 55a034a 1a45d5d 55a034a 1a45d5d 55a034a 1a45d5d a64b653 1a45d5d 55a034a 5bd3ab2 |
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 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 |
import { serve } from "https://deno.land/[email protected]/http/server.ts";
import * as Sentry from "https://deno.land/x/sentry/index.mjs";
Sentry.init({
dsn: "https://ca41c3f96489cc1b3e69c9a44704f7ee@o4508722276007936.ingest.de.sentry.io/4508772265558096",
defaultIntegrations: false,
tracesSampleRate: 1.0,
profilesSampleRate: 1.0,
});
Sentry.setTag('region', Deno.env.get('SB_REGION'));
Sentry.setTag('execution_id', Deno.env.get('SB_EXECUTION_ID'));
const corsHeaders = {
'Access-Control-Allow-Origin': '*',
'Access-Control-Allow-Headers': 'authorization, x-client-info, apikey, content-type',
};
const languagePrompts = {
en: {
systemPrompt: "You are helping in a word guessing game. Given a description, guess what single word is being described. The word must be a singular noun. The described word itself was not allowed in the description, so do not expect it to appear.",
instruction: "Based on this description",
responseInstruction: "Respond with ONLY the word you think is being described, in uppercase letters. Do not add any explanation or punctuation. ONLY respond with ONE word, nothing else."
},
fr: {
systemPrompt: "Vous aidez dans un jeu de devinettes. À partir d'une description, devinez le mot unique qui est décrit. Le mot doit être un nom commun au singulier. Le mot décrit n'était pas autorisé dans la description, ne vous attendez donc pas à le voir apparaître.",
instruction: "D'après cette description",
responseInstruction: "Répondez uniquement par le mot que vous pensez être décrit, en lettres majuscules. N'ajoutez aucune explication ni ponctuation. Répondez uniquement com um palavra, nada mais."
},
de: {
systemPrompt: "Sie helfen bei einem Worträtsel. Erraten Sie anhand einer Beschreibung, welches einzelne Wort beschrieben wird. Das Wort muss ein Substantiv im Singular sein. Das beschriebene Wort durfte nicht in der Beschreibung verwendet werden, also erwarten Sie es nicht.",
instruction: "Basierend auf dieser Beschreibung",
responseInstruction: "Antworten Sie nur mit dem Wort, das Sie für beschrieben halten, in Großbuchstaben. Fügen Sie keine Erklärungen oder Satzzeichen hinzu. Antworten Sie nur mit einem Wort, nichts anderes."
},
it: {
systemPrompt: "Stai aiutando in un gioco di indovinelli. Data una descrizione, indovina quale singola parola viene descritta. La parola deve essere un sostantivo singolare. La parola descritta non era permessa nella descrizione, quindi non aspettarti di trovarla.",
instruction: "Basandoti su questa descrizione",
responseInstruction: "Rispondi solo con la parola che pensi venga descritta, in lettere maiuscole. Non aggiungere spiegazioni o punteggiatura. Rispondi solo con una parola, nient'altro."
},
es: {
systemPrompt: "Estás ayudando en un juego de adivinanzas. Dada una descripción, adivina qué palabra única se está describiendo. La palabra debe ser un sustantivo singular. La palabra descrita no estaba permitida en la descripción, así que no esperes verla.",
instruction: "Basándote en esta descripción",
responseInstruction: "Responde únicamente con la palabra que crees que se está describiendo, en letras mayúsculas. No añadas ninguna explicación ni puntuación. Responde únicamente con una palabra, nada más."
},
pt: {
systemPrompt: "Estás ajudando em um jogo de adivinhação. Dada uma descrição, adivinha qual palavra única está sendo descrita. A palavra deve ser um substantivo singular. A palavra descrita não foi permitida na descrição, então não espere vê-la.",
instruction: "Com base nesta descrição",
responseInstruction: "Responda apenas com a palavra que você acredita estar sendo descrita, em letras maiúsculas. Não adicione nenhuma explicação nem pontuação. Responda apenas com uma palavra, nada mais."
}
};
const openRouterModels = [
'google/gemini-2.0-flash-exp:free',
'mistralai/mistral-nemo'
];
async function generateGuess(sentence: string, language: string, model?: string) {
const prompts = languagePrompts[language as keyof typeof languagePrompts] || languagePrompts.en;
const selectedModel = model || openRouterModels[Math.floor(Math.random() * openRouterModels.length)];
console.log('Using OpenRouter with model:', selectedModel);
try {
const response = await fetch("https://openrouter.ai/api/v1/chat/completions", {
method: "POST",
headers: {
"Authorization": `Bearer ${Deno.env.get('OPENROUTER_API_KEY')}`,
"HTTP-Referer": "https://think-in-sync.com",
"X-Title": "Think in Sync",
"Content-Type": "application/json"
},
body: JSON.stringify({
model: selectedModel,
messages: [
{
role: "system",
content: `${prompts.systemPrompt} ${prompts.responseInstruction}`
},
{
role: "user",
content: `${prompts.instruction} "${sentence}"`
}
]
})
});
if (!response.ok) {
const errorText = await response.text();
console.error('OpenRouter API error:', {
status: response.status,
statusText: response.statusText,
body: errorText
});
if (response.status === 402) {
throw new Error('The AI service has reached its rate limit. Please try again in a few moments.');
}
throw new Error(`OpenRouter API error (${response.status}): ${errorText}`);
}
const data = await response.json();
return {
guess: data.choices[0].message.content.trim().toUpperCase(),
model: selectedModel
};
} catch (error) {
console.error('Error in generateGuess:', error);
// Re-throw with more user-friendly message if it's not already a custom error
if (!error.message.includes('rate limit')) {
throw new Error('Failed to generate guess. Please try again.');
}
throw error;
}
}
serve(async (req) => {
if (req.method === 'OPTIONS') {
return new Response(null, { headers: corsHeaders });
}
try {
const { sentence, language = 'en', model } = await req.json();
console.log('Trying to guess word from sentence:', sentence, 'language:', language, 'model:', model);
const { guess, model: usedModel } = await generateGuess(sentence, language, model);
console.log('Successfully generated guess:', guess, 'using model:', usedModel);
return new Response(
JSON.stringify({ guess, model: usedModel }),
{ headers: { ...corsHeaders, 'Content-Type': 'application/json' } }
);
} catch (error) {
Sentry.captureException(error);
console.error('Error generating guess:', error);
return new Response(
JSON.stringify({
error: error.message || 'An unexpected error occurred',
}),
{
status: error.message.includes('rate limit') ? 429 : 500,
headers: { ...corsHeaders, 'Content-Type': 'application/json' }
}
);
}
});
|