-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathapi-service.js
More file actions
154 lines (130 loc) · 6.08 KB
/
api-service.js
File metadata and controls
154 lines (130 loc) · 6.08 KB
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
149
150
151
152
153
154
// API Service for Gemini API calls
class ApiService {
constructor(config) {
this.config = config;
// Use the correct Gemini model - gemini-1.5-flash is the current stable model
this.baseUrl = 'https://generativelanguage.googleapis.com/v1beta/models/gemini-1.5-flash:generateContent';
}
async generateHints(problemStatement) {
if (!this.config.hasValidApiKey()) {
throw new Error('No valid API key found. Please set your Gemini API key.');
}
const prompt = formatPrompt(PROMPT_TEMPLATES.DSA_HINTS, problemStatement);
try {
console.log('Making API request to Gemini...');
const response = await fetch(
`${this.baseUrl}?key=${this.config.getApiKey()}`,
{
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
contents: [{
parts: [{
text: prompt
}]
}],
generationConfig: {
temperature: 0.7,
topK: 40,
topP: 0.95,
maxOutputTokens: 800,
}
})
}
);
console.log('API Response status:', response.status);
if (!response.ok) {
const errorData = await response.json().catch(() => ({}));
const errorMessage = errorData.error?.message || `HTTP ${response.status}: ${response.statusText}`;
console.error('API Error:', errorData);
// Handle specific API errors
if (response.status === 400) {
throw new Error(`Invalid request: ${errorMessage}`);
} else if (response.status === 401) {
throw new Error('Invalid API key. Please check your Gemini API key.');
} else if (response.status === 403) {
throw new Error('API key not authorized. Please check your API key permissions.');
} else if (response.status === 429) {
throw new Error('Rate limit exceeded. Please try again later.');
} else if (response.status >= 500) {
throw new Error('Server error. Please try again later.');
} else {
throw new Error(`API Error: ${errorMessage}`);
}
}
const data = await response.json();
console.log('API Response data:', data);
// Validate response structure
if (!data.candidates || !data.candidates[0] || !data.candidates[0].content || !data.candidates[0].content.parts || !data.candidates[0].content.parts[0]) {
throw new Error('Invalid response format from API');
}
const responseText = data.candidates[0].content.parts[0].text;
if (!responseText || responseText.trim().length === 0) {
throw new Error('Empty response from API');
}
return extractHints(responseText);
} catch (error) {
console.error('Error in generateHints:', error);
// Re-throw API errors
if (error.message.includes('API Error') || error.message.includes('Invalid') || error.message.includes('Rate limit')) {
throw error;
}
// Handle network errors
if (error.name === 'TypeError' && error.message.includes('fetch')) {
throw new Error('Network error. Please check your internet connection.');
}
// Handle JSON parsing errors
if (error.name === 'SyntaxError') {
throw new Error('Invalid response from API. Please try again.');
}
// Generic error
throw new Error(`Failed to generate hints: ${error.message}`);
}
}
async testApiKey(apiKey) {
const testPrompt = "Hello, this is a test message.";
try {
console.log('Testing API key...');
const response = await fetch(
`${this.baseUrl}?key=${apiKey}`,
{
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
contents: [{
parts: [{
text: testPrompt
}]
}],
generationConfig: {
maxOutputTokens: 10,
}
})
}
);
console.log('Test API Response status:', response.status);
if (!response.ok) {
const errorData = await response.json().catch(() => ({}));
console.error('Test API Error:', errorData);
if (response.status === 401 || response.status === 403) {
return { valid: false, error: 'Invalid API key' };
}
return { valid: false, error: errorData.error?.message || 'API test failed' };
}
const data = await response.json();
console.log('Test API Response data:', data);
return { valid: true };
} catch (error) {
console.error('Error testing API key:', error);
return { valid: false, error: error.message };
}
}
}
// Export for use in other files
if (typeof module !== 'undefined' && module.exports) {
module.exports = ApiService;
}