forked from knative/docs
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathroute.ts
More file actions
135 lines (114 loc) · 3.55 KB
/
route.ts
File metadata and controls
135 lines (114 loc) · 3.55 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
import {WebSocket, WebSocketServer} from 'ws';
import { NextRequest, NextResponse } from 'next/server';
export async function GET(
req: NextRequest,
{ params }: { params: Promise<{ slug: string }> }
) {
const { slug } = await params // 'a', 'b', or 'c'
return proxyRequest(req, slug);
}
export async function POST(
req: NextRequest,
{ params }: { params: Promise<{ slug: string }> }
) {
const { slug } = await params // 'a', 'b', or 'c'
return proxyRequest(req, slug);
}
export async function PUT(
req: NextRequest,
{ params }: { params: Promise<{ slug: string }> }
) {
const { slug } = await params // 'a', 'b', or 'c'
return proxyRequest(req, slug);
}
export async function DELETE(
req: NextRequest,
{ params }: { params: Promise<{ slug: string }> }
) {
const { slug } = await params // 'a', 'b', or 'c'
return proxyRequest(req, slug);
}
async function proxyRequest(req: NextRequest, slug: string) {
const BACKEND_URL = (process.env.BACKEND_URL || `http://node-server-svc.${process.env.POD_NAMESPACE}.svc.cluster.local`) + '/' + slug;
const backendResponse = await fetch(BACKEND_URL, {
method: req.method,
headers: {
...Object.fromEntries(req.headers),
host: new URL(BACKEND_URL).host,
},
body: req.body ? req.body : undefined,
duplex: "half",
} as any);
// Clone the response and return it
const data = await backendResponse.arrayBuffer();
return new NextResponse(data, {
status: backendResponse.status,
headers: backendResponse.headers,
});
}
export function UPGRADE(
client: WebSocket,
server: WebSocketServer,
request: NextRequest,
context: import('next-ws/server').RouteContext<'/backend/[slug]'>,
) {
const slug = context.params?.slug
console.log('Client connected to /backend/' + slug)
// Backend server URL - replace with your actual backend server
const BACKEND_URL = (process.env.BACKEND_WS_URL || `ws://node-server-svc.${process.env.POD_NAMESPACE}.svc.cluster.local`) + '/' + slug;
console.log('Backend URL:', BACKEND_URL);
let backendConnection: WebSocket | null = null;
let isClientClosed = false;
let isBackendClosed = false;
// Connect to backend server
try {
backendConnection = new WebSocket(BACKEND_URL);
backendConnection.on('open', () => {
console.log('Connected to backend server');
});
backendConnection.on('message', (message) => {
if (!isClientClosed) {
client.send(message.toString());
}
});
backendConnection.on('close', (err) => {
console.log('Backend connection closed', err);
isBackendClosed = true;
if (!isClientClosed) {
client.close();
}
});
backendConnection.on('error', (error) => {
console.error('Backend connection error:', error);
if (!isClientClosed) {
client.close();
}
});
} catch (error) {
console.error('Failed to connect to backend:', error);
client.close();
return;
}
// Handle messages from client
client.on('message', (message) => {
if (backendConnection && !isBackendClosed) {
backendConnection.send(message.toString());
}
});
// Handle client disconnect
client.once('close', () => {
console.log('A client disconnected');
isClientClosed = true;
if (backendConnection && !isBackendClosed) {
backendConnection.close();
}
});
// Handle client errors
client.on('error', (error) => {
console.error('Client connection error:', error);
isClientClosed = true;
if (backendConnection && !isBackendClosed) {
backendConnection.close();
}
});
}