forked from tech-shrimp/gemini-playground
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeno_index.ts
130 lines (110 loc) · 3.46 KB
/
deno_index.ts
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
const getContentType = (path: string): string => {
const ext = path.split('.').pop()?.toLowerCase() || '';
const types: Record<string, string> = {
'js': 'application/javascript',
'css': 'text/css',
'html': 'text/html',
'json': 'application/json',
'png': 'image/png',
'jpg': 'image/jpeg',
'jpeg': 'image/jpeg',
'gif': 'image/gif'
};
return types[ext] || 'text/plain';
};
async function handleWebSocket(req: Request): Promise<Response> {
const { socket: clientWs, response } = Deno.upgradeWebSocket(req);
const url = new URL(req.url);
const targetUrl = `wss://generativelanguage.googleapis.com${url.pathname}${url.search}`;
console.log('Target URL:', targetUrl);
const pendingMessages: string[] = [];
const targetWs = new WebSocket(targetUrl);
targetWs.onopen = () => {
console.log('Connected to Gemini');
pendingMessages.forEach(msg => targetWs.send(msg));
pendingMessages.length = 0;
};
clientWs.onmessage = (event) => {
console.log('Client message received');
if (targetWs.readyState === WebSocket.OPEN) {
targetWs.send(event.data);
} else {
pendingMessages.push(event.data);
}
};
targetWs.onmessage = (event) => {
console.log('Gemini message received');
if (clientWs.readyState === WebSocket.OPEN) {
clientWs.send(event.data);
}
};
clientWs.onclose = (event) => {
console.log('Client connection closed');
if (targetWs.readyState === WebSocket.OPEN) {
targetWs.close(1000, event.reason);
}
};
targetWs.onclose = (event) => {
console.log('Gemini connection closed');
if (clientWs.readyState === WebSocket.OPEN) {
clientWs.close(event.code, event.reason);
}
};
targetWs.onerror = (error) => {
console.error('Gemini WebSocket error:', error);
};
return response;
}
async function handleAPIRequest(req: Request): Promise<Response> {
try {
const worker = await import('./api_proxy/worker.mjs');
return await worker.default.fetch(req);
} catch (error) {
console.error('API request error:', error);
const errorMessage = error instanceof Error ? error.message : 'Unknown error occurred';
const errorStatus = (error as { status?: number }).status || 500;
return new Response(errorMessage, {
status: errorStatus,
headers: {
'content-type': 'text/plain;charset=UTF-8',
}
});
}
}
async function handleRequest(req: Request): Promise<Response> {
const url = new URL(req.url);
console.log('Request URL:', req.url);
// WebSocket 处理
if (req.headers.get("Upgrade")?.toLowerCase() === "websocket") {
return handleWebSocket(req);
}
if (url.pathname.endsWith("/chat/completions") ||
url.pathname.endsWith("/embeddings") ||
url.pathname.endsWith("/models")) {
return handleAPIRequest(req);
}
// 静态文件处理
try {
let filePath = url.pathname;
if (filePath === '/' || filePath === '/index.html') {
filePath = '/index.html';
}
const fullPath = `${Deno.cwd()}/src/static${filePath}`;
const file = await Deno.readFile(fullPath);
const contentType = getContentType(filePath);
return new Response(file, {
headers: {
'content-type': `${contentType};charset=UTF-8`,
},
});
} catch (e) {
console.error('Error details:', e);
return new Response('Not Found', {
status: 404,
headers: {
'content-type': 'text/plain;charset=UTF-8',
}
});
}
}
Deno.serve(handleRequest);