@yan162/changewayguard 6.8.26 → 6.8.27

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (45) hide show
  1. package/dist/gateway/dist/activity.d.ts +52 -0
  2. package/dist/gateway/dist/activity.d.ts.map +1 -0
  3. package/dist/gateway/dist/activity.js +111 -0
  4. package/dist/gateway/dist/activity.js.map +1 -0
  5. package/dist/gateway/dist/config.d.ts +50 -0
  6. package/dist/gateway/dist/config.d.ts.map +1 -0
  7. package/dist/gateway/dist/config.js +200 -0
  8. package/dist/gateway/dist/config.js.map +1 -0
  9. package/dist/gateway/dist/handlers/anthropic.d.ts +12 -0
  10. package/dist/gateway/dist/handlers/anthropic.d.ts.map +1 -0
  11. package/dist/gateway/dist/handlers/anthropic.js +254 -0
  12. package/dist/gateway/dist/handlers/anthropic.js.map +1 -0
  13. package/dist/gateway/dist/handlers/gemini.d.ts +12 -0
  14. package/dist/gateway/dist/handlers/gemini.d.ts.map +1 -0
  15. package/dist/gateway/dist/handlers/gemini.js +101 -0
  16. package/dist/gateway/dist/handlers/gemini.js.map +1 -0
  17. package/dist/gateway/dist/handlers/models.d.ts +4 -0
  18. package/dist/gateway/dist/handlers/models.d.ts.map +1 -0
  19. package/dist/gateway/dist/handlers/models.js +36 -0
  20. package/dist/gateway/dist/handlers/models.js.map +1 -0
  21. package/dist/gateway/dist/handlers/openai.d.ts +16 -0
  22. package/dist/gateway/dist/handlers/openai.d.ts.map +1 -0
  23. package/dist/gateway/dist/handlers/openai.js +254 -0
  24. package/dist/gateway/dist/handlers/openai.js.map +1 -0
  25. package/dist/gateway/dist/index.d.ts +27 -0
  26. package/dist/gateway/dist/index.d.ts.map +1 -0
  27. package/dist/gateway/dist/index.js +290 -0
  28. package/dist/gateway/dist/index.js.map +1 -0
  29. package/dist/gateway/dist/mapping-store.d.ts +38 -0
  30. package/dist/gateway/dist/mapping-store.d.ts.map +1 -0
  31. package/dist/gateway/dist/mapping-store.js +74 -0
  32. package/dist/gateway/dist/mapping-store.js.map +1 -0
  33. package/dist/gateway/dist/restorer.d.ts +63 -0
  34. package/dist/gateway/dist/restorer.d.ts.map +1 -0
  35. package/dist/gateway/dist/restorer.js +284 -0
  36. package/dist/gateway/dist/restorer.js.map +1 -0
  37. package/dist/gateway/dist/sanitizer.d.ts +17 -0
  38. package/dist/gateway/dist/sanitizer.d.ts.map +1 -0
  39. package/dist/gateway/dist/sanitizer.js +228 -0
  40. package/dist/gateway/dist/sanitizer.js.map +1 -0
  41. package/dist/gateway/dist/types.d.ts +53 -0
  42. package/dist/gateway/dist/types.d.ts.map +1 -0
  43. package/dist/gateway/dist/types.js +5 -0
  44. package/dist/gateway/dist/types.js.map +1 -0
  45. package/package.json +1 -1
@@ -0,0 +1,254 @@
1
+ /**
2
+ * AI Security Gateway - OpenAI Chat Completions API handler
3
+ *
4
+ * Handles POST /v1/chat/completions requests in OpenAI's format.
5
+ * Also compatible with OpenAI-compatible APIs (Kimi, DeepSeek, etc.)
6
+ */
7
+ import { sanitize } from "../sanitizer.js";
8
+ import { restore, createStreamRestorer } from "../restorer.js";
9
+ import { generateRequestId, logSanitizeEvent, logRestoreEvent } from "../activity.js";
10
+ /**
11
+ * Handle OpenAI API request
12
+ *
13
+ * @param backend - Config for OpenAI-compatible backend
14
+ * @param extraHeaders - Optional additional headers (e.g., OpenRouter attribution)
15
+ */
16
+ export async function handleOpenAIRequest(req, res, backend, extraHeaders) {
17
+ try {
18
+ const requestId = generateRequestId();
19
+ const sanitizeStart = Date.now();
20
+ // 1. Parse request body
21
+ const body = await readBody(req);
22
+ const requestData = JSON.parse(body);
23
+ const { model, messages, tools, tool_choice, temperature, max_tokens, stream = false, ...rest } = requestData;
24
+ // 2. Sanitize messages
25
+ const { sanitized: sanitizedMessages, mappingTable, redactionCount } = sanitize(messages);
26
+ // Debug: log what was sanitized
27
+ console.log(`[ai-security-gateway] Sanitized ${redactionCount} items`);
28
+ if (mappingTable.size > 0) {
29
+ for (const [placeholder, original] of mappingTable.entries()) {
30
+ console.log(`[ai-security-gateway] ${placeholder} <- (${original.length} chars)`);
31
+ }
32
+ }
33
+ // Log sanitization event
34
+ if (redactionCount > 0) {
35
+ logSanitizeEvent({
36
+ requestId,
37
+ backend: "openai",
38
+ endpoint: "/v1/chat/completions",
39
+ model,
40
+ mappingTable,
41
+ redactionCount,
42
+ durationMs: Date.now() - sanitizeStart,
43
+ });
44
+ }
45
+ // 3. Build sanitized request
46
+ const sanitizedRequest = {
47
+ model,
48
+ messages: sanitizedMessages,
49
+ ...(tools && { tools }),
50
+ ...(tool_choice && { tool_choice }),
51
+ ...(temperature !== undefined && { temperature }),
52
+ ...(max_tokens && { max_tokens }),
53
+ stream,
54
+ ...rest,
55
+ };
56
+ // 4. Use provided backend config
57
+ // Note: baseUrl already includes the full path prefix (e.g., /v1 or /v1/coding)
58
+ const apiUrl = `${backend.baseUrl}/chat/completions`;
59
+ const headers = {
60
+ "Content-Type": "application/json",
61
+ "Authorization": `Bearer ${backend.apiKey}`,
62
+ };
63
+ // Merge extra headers (e.g., OpenRouter attribution headers)
64
+ if (extraHeaders) {
65
+ Object.assign(headers, extraHeaders);
66
+ }
67
+ const response = await fetch(apiUrl, {
68
+ method: "POST",
69
+ headers,
70
+ body: JSON.stringify(sanitizedRequest),
71
+ });
72
+ if (!response.ok) {
73
+ // Forward error response
74
+ res.writeHead(response.status, { "Content-Type": "application/json" });
75
+ const errorBody = await response.text();
76
+ res.end(errorBody);
77
+ return;
78
+ }
79
+ // 6. Handle streaming or non-streaming response
80
+ if (stream) {
81
+ await handleOpenAIStream(response, res, mappingTable, requestId, model);
82
+ }
83
+ else {
84
+ await handleOpenAINonStream(response, res, mappingTable, requestId, model);
85
+ }
86
+ }
87
+ catch (error) {
88
+ console.error("[ai-security-gateway] OpenAI handler error:", error);
89
+ res.writeHead(500, { "Content-Type": "application/json" });
90
+ res.end(JSON.stringify({
91
+ error: "Internal gateway error",
92
+ message: error instanceof Error ? error.message : String(error),
93
+ }));
94
+ }
95
+ }
96
+ /**
97
+ * Handle streaming response (SSE) with smart placeholder restoration
98
+ *
99
+ * Uses StreamRestorer to detect `__` and buffer potential placeholders.
100
+ * Only buffers when necessary, maintaining streaming UX.
101
+ */
102
+ async function handleOpenAIStream(response, res, mappingTable, requestId, model) {
103
+ const restoreStart = Date.now();
104
+ // Debug: log mapping table
105
+ if (mappingTable.size > 0) {
106
+ console.log(`[ai-security-gateway] Streaming with ${mappingTable.size} placeholders to restore`);
107
+ }
108
+ // Set SSE headers
109
+ res.writeHead(200, {
110
+ "Content-Type": "text/event-stream",
111
+ "Cache-Control": "no-cache",
112
+ "Connection": "keep-alive",
113
+ });
114
+ const reader = response.body?.getReader();
115
+ if (!reader) {
116
+ res.end();
117
+ return;
118
+ }
119
+ const decoder = new TextDecoder();
120
+ let lineBuffer = "";
121
+ // Create stream restorer for text content
122
+ const streamRestorer = createStreamRestorer(mappingTable);
123
+ // Buffer for SSE chunks waiting for restoration
124
+ const pendingChunks = [];
125
+ try {
126
+ while (true) {
127
+ const { done, value } = await reader.read();
128
+ if (done)
129
+ break;
130
+ // Decode chunk
131
+ lineBuffer += decoder.decode(value, { stream: true });
132
+ // Process complete lines
133
+ const lines = lineBuffer.split("\n");
134
+ lineBuffer = lines.pop() || ""; // Keep incomplete line in buffer
135
+ for (const line of lines) {
136
+ if (!line.trim()) {
137
+ // Flush pending chunks before empty line
138
+ flushPendingChunks(pendingChunks, streamRestorer, res);
139
+ res.write("\n");
140
+ continue;
141
+ }
142
+ if (!line.startsWith("data: ")) {
143
+ res.write(line + "\n");
144
+ continue;
145
+ }
146
+ const dataContent = line.slice(6);
147
+ if (dataContent === "[DONE]") {
148
+ // Finalize any pending content
149
+ flushPendingChunks(pendingChunks, streamRestorer, res);
150
+ res.write(line + "\n");
151
+ continue;
152
+ }
153
+ try {
154
+ const parsed = JSON.parse(dataContent);
155
+ const textContent = parsed.choices?.[0]?.delta?.content;
156
+ if (textContent !== undefined && mappingTable.size > 0) {
157
+ // Process text through stream restorer
158
+ const restored = streamRestorer.process(textContent);
159
+ if (restored.length > 0) {
160
+ // We have restorable content - flush it
161
+ const restoredChunk = { ...parsed };
162
+ restoredChunk.choices = parsed.choices.map((c, i) => i === 0 ? { ...c, delta: { ...c.delta, content: restored } } : c);
163
+ res.write(`data: ${JSON.stringify(restoredChunk)}\n`);
164
+ }
165
+ // If restorer is buffering, we don't output anything yet
166
+ // Content will be output when buffer is flushed
167
+ }
168
+ else {
169
+ // No text content or no mappings - pass through
170
+ res.write(line + "\n");
171
+ }
172
+ }
173
+ catch {
174
+ // Not valid JSON, pass through
175
+ res.write(line + "\n");
176
+ }
177
+ }
178
+ }
179
+ // Write any remaining line buffer
180
+ if (lineBuffer.trim()) {
181
+ res.write(lineBuffer + "\n");
182
+ }
183
+ // Finalize stream restorer - flush any remaining buffered content
184
+ const finalContent = streamRestorer.finalize();
185
+ if (finalContent.length > 0) {
186
+ // Create a final chunk with remaining content
187
+ const finalChunk = {
188
+ choices: [{ delta: { content: finalContent }, index: 0, finish_reason: null }],
189
+ };
190
+ res.write(`data: ${JSON.stringify(finalChunk)}\n`);
191
+ }
192
+ // Log restoration event
193
+ if (mappingTable.size > 0) {
194
+ logRestoreEvent({
195
+ requestId,
196
+ backend: "openai",
197
+ endpoint: "/v1/chat/completions",
198
+ model,
199
+ mappingTable,
200
+ restorationCount: mappingTable.size,
201
+ durationMs: Date.now() - restoreStart,
202
+ });
203
+ }
204
+ res.end();
205
+ }
206
+ catch (error) {
207
+ console.error("[ai-security-gateway] Stream error:", error);
208
+ res.end();
209
+ }
210
+ }
211
+ /**
212
+ * Flush pending chunks with restored content
213
+ */
214
+ function flushPendingChunks(_pendingChunks, _streamRestorer, _res) {
215
+ // Currently unused - StreamRestorer handles buffering internally
216
+ }
217
+ /**
218
+ * Handle non-streaming response
219
+ */
220
+ async function handleOpenAINonStream(response, res, mappingTable, requestId, model) {
221
+ const restoreStart = Date.now();
222
+ const responseBody = await response.text();
223
+ const responseData = JSON.parse(responseBody);
224
+ // Restore placeholders in response
225
+ const restoredData = restore(responseData, mappingTable);
226
+ // Log restoration event
227
+ if (mappingTable.size > 0) {
228
+ logRestoreEvent({
229
+ requestId,
230
+ backend: "openai",
231
+ endpoint: "/v1/chat/completions",
232
+ model,
233
+ mappingTable,
234
+ restorationCount: mappingTable.size,
235
+ durationMs: Date.now() - restoreStart,
236
+ });
237
+ }
238
+ res.writeHead(200, { "Content-Type": "application/json" });
239
+ res.end(JSON.stringify(restoredData));
240
+ }
241
+ /**
242
+ * Read request body as string
243
+ */
244
+ function readBody(req) {
245
+ return new Promise((resolve, reject) => {
246
+ let body = "";
247
+ req.on("data", (chunk) => {
248
+ body += chunk.toString();
249
+ });
250
+ req.on("end", () => resolve(body));
251
+ req.on("error", reject);
252
+ });
253
+ }
254
+ //# sourceMappingURL=openai.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"openai.js","sourceRoot":"","sources":["../../src/handlers/openai.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAIH,OAAO,EAAE,QAAQ,EAAE,MAAM,iBAAiB,CAAC;AAC3C,OAAO,EAAE,OAAO,EAAE,oBAAoB,EAAE,MAAM,gBAAgB,CAAC;AAC/D,OAAO,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAEtF;;;;;GAKG;AACH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACvC,GAAoB,EACpB,GAAmB,EACnB,OAAsB,EACtB,YAAqC;IAErC,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,iBAAiB,EAAE,CAAC;QACtC,MAAM,aAAa,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAEjC,wBAAwB;QACxB,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,GAAG,CAAC,CAAC;QACjC,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAErC,MAAM,EACJ,KAAK,EACL,QAAQ,EACR,KAAK,EACL,WAAW,EACX,WAAW,EACX,UAAU,EACV,MAAM,GAAG,KAAK,EACd,GAAG,IAAI,EACR,GAAG,WAAW,CAAC;QAEhB,uBAAuB;QACvB,MAAM,EAAE,SAAS,EAAE,iBAAiB,EAAE,YAAY,EAAE,cAAc,EAAE,GAAG,QAAQ,CAAC,QAAQ,CAAC,CAAC;QAE1F,gCAAgC;QAChC,OAAO,CAAC,GAAG,CAAC,mCAAmC,cAAc,QAAQ,CAAC,CAAC;QACvE,IAAI,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;YAC1B,KAAK,MAAM,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,YAAY,CAAC,OAAO,EAAE,EAAE,CAAC;gBAC7D,OAAO,CAAC,GAAG,CAAC,2BAA2B,WAAW,QAAQ,QAAQ,CAAC,MAAM,SAAS,CAAC,CAAC;YACtF,CAAC;QACH,CAAC;QAED,yBAAyB;QACzB,IAAI,cAAc,GAAG,CAAC,EAAE,CAAC;YACvB,gBAAgB,CAAC;gBACf,SAAS;gBACT,OAAO,EAAE,QAAQ;gBACjB,QAAQ,EAAE,sBAAsB;gBAChC,KAAK;gBACL,YAAY;gBACZ,cAAc;gBACd,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,aAAa;aACvC,CAAC,CAAC;QACL,CAAC;QAED,6BAA6B;QAC7B,MAAM,gBAAgB,GAAG;YACvB,KAAK;YACL,QAAQ,EAAE,iBAAiB;YAC3B,GAAG,CAAC,KAAK,IAAI,EAAE,KAAK,EAAE,CAAC;YACvB,GAAG,CAAC,WAAW,IAAI,EAAE,WAAW,EAAE,CAAC;YACnC,GAAG,CAAC,WAAW,KAAK,SAAS,IAAI,EAAE,WAAW,EAAE,CAAC;YACjD,GAAG,CAAC,UAAU,IAAI,EAAE,UAAU,EAAE,CAAC;YACjC,MAAM;YACN,GAAG,IAAI;SACR,CAAC;QAEF,iCAAiC;QACjC,gFAAgF;QAChF,MAAM,MAAM,GAAG,GAAG,OAAO,CAAC,OAAO,mBAAmB,CAAC;QACrD,MAAM,OAAO,GAA2B;YACtC,cAAc,EAAE,kBAAkB;YAClC,eAAe,EAAE,UAAU,OAAO,CAAC,MAAM,EAAE;SAC5C,CAAC;QACF,6DAA6D;QAC7D,IAAI,YAAY,EAAE,CAAC;YACjB,MAAM,CAAC,MAAM,CAAC,OAAO,EAAE,YAAY,CAAC,CAAC;QACvC,CAAC;QACD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,MAAM,EAAE;YACnC,MAAM,EAAE,MAAM;YACd,OAAO;YACP,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,gBAAgB,CAAC;SACvC,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,yBAAyB;YACzB,GAAG,CAAC,SAAS,CAAC,QAAQ,CAAC,MAAM,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;YACvE,MAAM,SAAS,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACxC,GAAG,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YACnB,OAAO;QACT,CAAC;QAED,gDAAgD;QAChD,IAAI,MAAM,EAAE,CAAC;YACX,MAAM,kBAAkB,CAAC,QAAQ,EAAE,GAAG,EAAE,YAAY,EAAE,SAAS,EAAE,KAAK,CAAC,CAAC;QAC1E,CAAC;aAAM,CAAC;YACN,MAAM,qBAAqB,CAAC,QAAQ,EAAE,GAAG,EAAE,YAAY,EAAE,SAAS,EAAE,KAAK,CAAC,CAAC;QAC7E,CAAC;IACH,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,6CAA6C,EAAE,KAAK,CAAC,CAAC;QACpE,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;QAC3D,GAAG,CAAC,GAAG,CACL,IAAI,CAAC,SAAS,CAAC;YACb,KAAK,EAAE,wBAAwB;YAC/B,OAAO,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;SAChE,CAAC,CACH,CAAC;IACJ,CAAC;AACH,CAAC;AAED;;;;;GAKG;AACH,KAAK,UAAU,kBAAkB,CAC/B,QAAkB,EAClB,GAAmB,EACnB,YAA0B,EAC1B,SAAiB,EACjB,KAAc;IAEd,MAAM,YAAY,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAEhC,2BAA2B;IAC3B,IAAI,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;QAC1B,OAAO,CAAC,GAAG,CAAC,wCAAwC,YAAY,CAAC,IAAI,0BAA0B,CAAC,CAAC;IACnG,CAAC;IAED,kBAAkB;IAClB,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE;QACjB,cAAc,EAAE,mBAAmB;QACnC,eAAe,EAAE,UAAU;QAC3B,YAAY,EAAE,YAAY;KAC3B,CAAC,CAAC;IAEH,MAAM,MAAM,GAAG,QAAQ,CAAC,IAAI,EAAE,SAAS,EAAE,CAAC;IAC1C,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,GAAG,CAAC,GAAG,EAAE,CAAC;QACV,OAAO;IACT,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,WAAW,EAAE,CAAC;IAClC,IAAI,UAAU,GAAG,EAAE,CAAC;IAEpB,0CAA0C;IAC1C,MAAM,cAAc,GAAG,oBAAoB,CAAC,YAAY,CAAC,CAAC;IAE1D,gDAAgD;IAChD,MAAM,aAAa,GAA4D,EAAE,CAAC;IAElF,IAAI,CAAC;QACH,OAAO,IAAI,EAAE,CAAC;YACZ,MAAM,EAAE,IAAI,EAAE,KAAK,EAAE,GAAG,MAAM,MAAM,CAAC,IAAI,EAAE,CAAC;YAC5C,IAAI,IAAI;gBAAE,MAAM;YAEhB,eAAe;YACf,UAAU,IAAI,OAAO,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;YAEtD,yBAAyB;YACzB,MAAM,KAAK,GAAG,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;YACrC,UAAU,GAAG,KAAK,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,CAAC,iCAAiC;YAEjE,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;gBACzB,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC;oBACjB,yCAAyC;oBACzC,kBAAkB,CAAC,aAAa,EAAE,cAAc,EAAE,GAAG,CAAC,CAAC;oBACvD,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;oBAChB,SAAS;gBACX,CAAC;gBAED,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;oBAC/B,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;oBACvB,SAAS;gBACX,CAAC;gBAED,MAAM,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;gBAClC,IAAI,WAAW,KAAK,QAAQ,EAAE,CAAC;oBAC7B,+BAA+B;oBAC/B,kBAAkB,CAAC,aAAa,EAAE,cAAc,EAAE,GAAG,CAAC,CAAC;oBACvD,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;oBACvB,SAAS;gBACX,CAAC;gBAED,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAmB,CAAC;oBACzD,MAAM,WAAW,GAAG,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,EAAE,OAAO,CAAC;oBAExD,IAAI,WAAW,KAAK,SAAS,IAAI,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;wBACvD,uCAAuC;wBACvC,MAAM,QAAQ,GAAG,cAAc,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;wBAErD,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;4BACxB,wCAAwC;4BACxC,MAAM,aAAa,GAAG,EAAE,GAAG,MAAM,EAAE,CAAC;4BACpC,aAAa,CAAC,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAClD,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,KAAK,EAAE,EAAE,GAAG,CAAC,CAAC,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CACjE,CAAC;4BACF,GAAG,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;wBACxD,CAAC;wBAED,yDAAyD;wBACzD,gDAAgD;oBAClD,CAAC;yBAAM,CAAC;wBACN,gDAAgD;wBAChD,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;oBACzB,CAAC;gBACH,CAAC;gBAAC,MAAM,CAAC;oBACP,+BAA+B;oBAC/B,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,CAAC;gBACzB,CAAC;YACH,CAAC;QACH,CAAC;QAED,kCAAkC;QAClC,IAAI,UAAU,CAAC,IAAI,EAAE,EAAE,CAAC;YACtB,GAAG,CAAC,KAAK,CAAC,UAAU,GAAG,IAAI,CAAC,CAAC;QAC/B,CAAC;QAED,kEAAkE;QAClE,MAAM,YAAY,GAAG,cAAc,CAAC,QAAQ,EAAE,CAAC;QAC/C,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5B,8CAA8C;YAC9C,MAAM,UAAU,GAAmB;gBACjC,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,EAAE,OAAO,EAAE,YAAY,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC;aAC/E,CAAC;YACF,GAAG,CAAC,KAAK,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;QACrD,CAAC;QAED,wBAAwB;QACxB,IAAI,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;YAC1B,eAAe,CAAC;gBACd,SAAS;gBACT,OAAO,EAAE,QAAQ;gBACjB,QAAQ,EAAE,sBAAsB;gBAChC,KAAK;gBACL,YAAY;gBACZ,gBAAgB,EAAE,YAAY,CAAC,IAAI;gBACnC,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,YAAY;aACtC,CAAC,CAAC;QACL,CAAC;QAED,GAAG,CAAC,GAAG,EAAE,CAAC;IACZ,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,OAAO,CAAC,KAAK,CAAC,qCAAqC,EAAE,KAAK,CAAC,CAAC;QAC5D,GAAG,CAAC,GAAG,EAAE,CAAC;IACZ,CAAC;AACH,CAAC;AAcD;;GAEG;AACH,SAAS,kBAAkB,CACzB,cAAuE,EACvE,eAAwD,EACxD,IAAoB;IAEpB,iEAAiE;AACnE,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,qBAAqB,CAClC,QAAkB,EAClB,GAAmB,EACnB,YAA0B,EAC1B,SAAiB,EACjB,KAAc;IAEd,MAAM,YAAY,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;IAChC,MAAM,YAAY,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;IAC3C,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;IAE9C,mCAAmC;IACnC,MAAM,YAAY,GAAG,OAAO,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;IAEzD,wBAAwB;IACxB,IAAI,YAAY,CAAC,IAAI,GAAG,CAAC,EAAE,CAAC;QAC1B,eAAe,CAAC;YACd,SAAS;YACT,OAAO,EAAE,QAAQ;YACjB,QAAQ,EAAE,sBAAsB;YAChC,KAAK;YACL,YAAY;YACZ,gBAAgB,EAAE,YAAY,CAAC,IAAI;YACnC,UAAU,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,YAAY;SACtC,CAAC,CAAC;IACL,CAAC;IAED,GAAG,CAAC,SAAS,CAAC,GAAG,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE,CAAC,CAAC;IAC3D,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC,CAAC;AACxC,CAAC;AAED;;GAEG;AACH,SAAS,QAAQ,CAAC,GAAoB;IACpC,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QACrC,IAAI,IAAI,GAAG,EAAE,CAAC;QACd,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,KAAK,EAAE,EAAE;YACvB,IAAI,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;QAC3B,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,EAAE,CAAC,KAAK,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC;QACnC,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAC1B,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,27 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * OpenGuardrails AI Security Gateway
4
+ *
5
+ * Local HTTP proxy that intercepts LLM API calls, sanitizes sensitive data
6
+ * before sending to providers, and restores it in responses.
7
+ * Supports Anthropic, OpenAI, and Gemini protocols.
8
+ */
9
+ /**
10
+ * Stop the gateway server
11
+ */
12
+ export declare function stopGateway(): Promise<void>;
13
+ /**
14
+ * Check if gateway is running
15
+ */
16
+ export declare function isGatewayServerRunning(): boolean;
17
+ /**
18
+ * Start gateway server
19
+ * @param configPath - Path to config file
20
+ * @param embedded - If true, don't call process.exit on errors (for in-process use)
21
+ */
22
+ export declare function startGateway(configPath?: string, embedded?: boolean): void;
23
+ export { sanitize, sanitizeMessages } from "./sanitizer.js";
24
+ export { restore, restoreJSON, restoreSSELine } from "./restorer.js";
25
+ export { addActivityListener, removeActivityListener, clearActivityListeners, } from "./activity.js";
26
+ export type { GatewayConfig, MappingTable, SanitizeResult, EntityMatch, GatewayActivityEvent, ActivityListener, } from "./types.js";
27
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA;;;;;;GAMG;AAwMH;;GAEG;AACH,wBAAgB,WAAW,IAAI,OAAO,CAAC,IAAI,CAAC,CAY3C;AAED;;GAEG;AACH,wBAAgB,sBAAsB,IAAI,OAAO,CAEhD;AAED;;;;GAIG;AACH,wBAAgB,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,QAAQ,UAAQ,GAAG,IAAI,CAmFxE;AAGD,OAAO,EAAE,QAAQ,EAAE,gBAAgB,EAAE,MAAM,gBAAgB,CAAC;AAC5D,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;AACrE,OAAO,EACL,mBAAmB,EACnB,sBAAsB,EACtB,sBAAsB,GACvB,MAAM,eAAe,CAAC;AAEvB,YAAY,EACV,aAAa,EACb,YAAY,EACZ,cAAc,EACd,WAAW,EACX,oBAAoB,EACpB,gBAAgB,GACjB,MAAM,YAAY,CAAC"}
@@ -0,0 +1,290 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * OpenGuardrails AI Security Gateway
4
+ *
5
+ * Local HTTP proxy that intercepts LLM API calls, sanitizes sensitive data
6
+ * before sending to providers, and restores it in responses.
7
+ * Supports Anthropic, OpenAI, and Gemini protocols.
8
+ */
9
+ import { createServer } from "node:http";
10
+ import { loadConfig, validateConfig, findBackendByApiKey, findDefaultBackend, findBackendByPathPrefix } from "./config.js";
11
+ import { handleAnthropicRequest } from "./handlers/anthropic.js";
12
+ import { handleOpenAIRequest } from "./handlers/openai.js";
13
+ import { handleGeminiRequest } from "./handlers/gemini.js";
14
+ import { handleModelsRequest } from "./handlers/models.js";
15
+ let config;
16
+ let currentServer = null;
17
+ /**
18
+ * Extract API key from request headers
19
+ */
20
+ function extractApiKey(req) {
21
+ // Try x-api-key header (Anthropic style)
22
+ const xApiKey = req.headers["x-api-key"];
23
+ if (xApiKey && typeof xApiKey === "string") {
24
+ return xApiKey;
25
+ }
26
+ // Try Authorization: Bearer (OpenAI style)
27
+ const auth = req.headers["authorization"];
28
+ if (auth && typeof auth === "string" && auth.startsWith("Bearer ")) {
29
+ return auth.slice(7);
30
+ }
31
+ // Try x-goog-api-key (Gemini style)
32
+ const googKey = req.headers["x-goog-api-key"];
33
+ if (googKey && typeof googKey === "string") {
34
+ return googKey;
35
+ }
36
+ return null;
37
+ }
38
+ /**
39
+ * Resolve backend for a request based on path prefix, API key, or defaults
40
+ * Priority: pathPrefix > apiKey > defaultBackend
41
+ */
42
+ function resolveBackend(req, apiType) {
43
+ const url = req.url || "";
44
+ // 1. Try to find backend by path prefix (most specific)
45
+ const byPath = findBackendByPathPrefix(url, config);
46
+ if (byPath) {
47
+ return byPath;
48
+ }
49
+ // 2. Try to find backend by API key
50
+ const apiKey = extractApiKey(req);
51
+ if (apiKey) {
52
+ const byKey = findBackendByApiKey(apiKey, config);
53
+ if (byKey) {
54
+ return byKey;
55
+ }
56
+ }
57
+ // 3. Fall back to default backend for the API type
58
+ return findDefaultBackend(apiType, config);
59
+ }
60
+ /**
61
+ * Main request handler
62
+ */
63
+ async function handleRequest(req, res) {
64
+ const { method, url } = req;
65
+ // Log request (skip health checks to reduce noise)
66
+ if (url !== "/health") {
67
+ console.log(`[ai-security-gateway] ${method} ${url}`);
68
+ }
69
+ // CORS headers (for browser-based clients)
70
+ res.setHeader("Access-Control-Allow-Origin", "*");
71
+ res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
72
+ res.setHeader("Access-Control-Allow-Headers", "Content-Type, Authorization, x-api-key, anthropic-version");
73
+ // Handle OPTIONS for CORS preflight
74
+ if (method === "OPTIONS") {
75
+ res.writeHead(204);
76
+ res.end();
77
+ return;
78
+ }
79
+ // Health check (allow GET)
80
+ if (url === "/health") {
81
+ res.writeHead(200, { "Content-Type": "application/json" });
82
+ res.end(JSON.stringify({ status: "ok", version: "1.0.0" }));
83
+ return;
84
+ }
85
+ // Handle GET /v1/models — proxy to configured backend's models endpoint
86
+ if (method === "GET" && url === "/v1/models") {
87
+ await handleModelsRequest(res, config);
88
+ return;
89
+ }
90
+ // Only allow POST for API endpoints
91
+ if (method !== "POST") {
92
+ res.writeHead(405, { "Content-Type": "application/json" });
93
+ res.end(JSON.stringify({ error: "Method not allowed" }));
94
+ return;
95
+ }
96
+ // Route to appropriate handler based on path suffix
97
+ // This allows flexible path prefixes (e.g., /v1/coding/chat/completions)
98
+ try {
99
+ if (url?.endsWith("/messages")) {
100
+ // Anthropic Messages API (matches /v1/messages, /v1/xxx/messages, etc.)
101
+ const resolved = resolveBackend(req, "anthropic");
102
+ if (!resolved) {
103
+ res.writeHead(500, { "Content-Type": "application/json" });
104
+ res.end(JSON.stringify({ error: "No Anthropic-compatible backend configured" }));
105
+ return;
106
+ }
107
+ await handleAnthropicRequest(req, res, resolved.backend);
108
+ }
109
+ else if (url?.endsWith("/chat/completions")) {
110
+ // OpenAI/OpenRouter Chat Completions API
111
+ // Try to extract backend name from URL: /backend/{name}/chat/completions
112
+ const backendMatch = url.match(/^\/backend\/([^/]+)\//);
113
+ let resolved = null;
114
+ if (backendMatch) {
115
+ const backendName = backendMatch[1];
116
+ const backend = config.backends[backendName];
117
+ if (backend) {
118
+ resolved = { name: backendName, backend };
119
+ console.log(`[ai-security-gateway] Backend from URL: ${backendName}`);
120
+ }
121
+ }
122
+ // Fallback to path prefix or default
123
+ if (!resolved) {
124
+ resolved = resolveBackend(req, "openai");
125
+ console.log(`[ai-security-gateway] Resolved backend: ${resolved?.name}`);
126
+ }
127
+ // Check explicit routing config
128
+ const explicitBackendName = config.routing?.["/v1/chat/completions"];
129
+ const backend = explicitBackendName
130
+ ? config.backends[explicitBackendName]
131
+ : resolved?.backend;
132
+ if (!backend) {
133
+ res.writeHead(500, { "Content-Type": "application/json" });
134
+ res.end(JSON.stringify({ error: "No OpenAI-compatible backend configured" }));
135
+ return;
136
+ }
137
+ const extraHeaders = {};
138
+ if (backend.referer) {
139
+ extraHeaders["HTTP-Referer"] = backend.referer;
140
+ }
141
+ if (backend.title) {
142
+ extraHeaders["X-Title"] = backend.title;
143
+ }
144
+ await handleOpenAIRequest(req, res, backend, extraHeaders);
145
+ }
146
+ else if (url?.match(/\/models\/(.+):generateContent$/)) {
147
+ // Gemini API (matches any path ending with /models/{model}:generateContent)
148
+ const match = url.match(/\/models\/(.+):generateContent$/);
149
+ const modelName = match?.[1];
150
+ if (modelName) {
151
+ const resolved = resolveBackend(req, "gemini");
152
+ if (!resolved) {
153
+ res.writeHead(500, { "Content-Type": "application/json" });
154
+ res.end(JSON.stringify({ error: "No Gemini backend configured" }));
155
+ return;
156
+ }
157
+ await handleGeminiRequest(req, res, resolved.backend, modelName);
158
+ }
159
+ else {
160
+ res.writeHead(404, { "Content-Type": "application/json" });
161
+ res.end(JSON.stringify({ error: "Model name required" }));
162
+ }
163
+ }
164
+ else {
165
+ // Unknown endpoint
166
+ res.writeHead(404, { "Content-Type": "application/json" });
167
+ res.end(JSON.stringify({ error: "Not found", url }));
168
+ }
169
+ }
170
+ catch (error) {
171
+ console.error("[ai-security-gateway] Request handler error:", error);
172
+ res.writeHead(500, { "Content-Type": "application/json" });
173
+ res.end(JSON.stringify({
174
+ error: "Internal server error",
175
+ message: error instanceof Error ? error.message : String(error),
176
+ }));
177
+ }
178
+ }
179
+ /**
180
+ * Stop the gateway server
181
+ */
182
+ export function stopGateway() {
183
+ return new Promise((resolve) => {
184
+ if (currentServer) {
185
+ currentServer.close(() => {
186
+ currentServer = null;
187
+ console.log("[ai-security-gateway] Server stopped");
188
+ resolve();
189
+ });
190
+ }
191
+ else {
192
+ resolve();
193
+ }
194
+ });
195
+ }
196
+ /**
197
+ * Check if gateway is running
198
+ */
199
+ export function isGatewayServerRunning() {
200
+ return currentServer !== null;
201
+ }
202
+ /**
203
+ * Start gateway server
204
+ * @param configPath - Path to config file
205
+ * @param embedded - If true, don't call process.exit on errors (for in-process use)
206
+ */
207
+ export function startGateway(configPath, embedded = false) {
208
+ // Stop existing server if running (same process)
209
+ if (currentServer) {
210
+ if (!embedded)
211
+ console.log("[ai-security-gateway] Stopping existing server for restart...");
212
+ currentServer.close();
213
+ currentServer = null;
214
+ }
215
+ try {
216
+ // Load and validate configuration
217
+ config = loadConfig(configPath);
218
+ validateConfig(config);
219
+ if (!embedded) {
220
+ console.log("[ai-security-gateway] Configuration loaded:");
221
+ console.log(` Port: ${config.port}`);
222
+ console.log(` Backends: ${Object.keys(config.backends).join(", ") || "(none)"}`);
223
+ }
224
+ // Create HTTP server
225
+ const server = createServer(handleRequest);
226
+ currentServer = server;
227
+ // Handle server errors
228
+ server.on("error", (err) => {
229
+ console.error("[ai-security-gateway] Server error:", err);
230
+ currentServer = null;
231
+ if (!embedded) {
232
+ process.exit(1);
233
+ }
234
+ throw err;
235
+ });
236
+ // Start listening
237
+ server.listen(config.port, "127.0.0.1", () => {
238
+ if (!embedded) {
239
+ console.log(`[ai-security-gateway] Server listening on http://127.0.0.1:${config.port}`);
240
+ console.log("[ai-security-gateway] Ready to proxy requests");
241
+ console.log("");
242
+ console.log("Endpoints:");
243
+ console.log(` POST http://127.0.0.1:${config.port}/v1/messages - Anthropic`);
244
+ console.log(` POST http://127.0.0.1:${config.port}/v1/chat/completions - OpenAI / OpenRouter`);
245
+ console.log(` POST http://127.0.0.1:${config.port}/v1/models/:model:generateContent - Gemini`);
246
+ console.log(` GET http://127.0.0.1:${config.port}/v1/models - List models (OpenAI / OpenRouter)`);
247
+ console.log(` GET http://127.0.0.1:${config.port}/health - Health check`);
248
+ }
249
+ });
250
+ // In embedded mode, don't let the server prevent process exit
251
+ if (embedded) {
252
+ server.unref();
253
+ }
254
+ // Only register shutdown handlers if not embedded
255
+ if (!embedded) {
256
+ process.on("SIGINT", () => {
257
+ console.log("\n[ai-security-gateway] Shutting down...");
258
+ server.close(() => {
259
+ console.log("[ai-security-gateway] Server stopped");
260
+ process.exit(0);
261
+ });
262
+ });
263
+ process.on("SIGTERM", () => {
264
+ console.log("\n[ai-security-gateway] Shutting down...");
265
+ server.close(() => {
266
+ console.log("[ai-security-gateway] Server stopped");
267
+ process.exit(0);
268
+ });
269
+ });
270
+ }
271
+ }
272
+ catch (error) {
273
+ console.error("[ai-security-gateway] Failed to start:", error);
274
+ currentServer = null;
275
+ if (!embedded) {
276
+ process.exit(1);
277
+ }
278
+ throw error;
279
+ }
280
+ }
281
+ // Re-export for programmatic use
282
+ export { sanitize, sanitizeMessages } from "./sanitizer.js";
283
+ export { restore, restoreJSON, restoreSSELine } from "./restorer.js";
284
+ export { addActivityListener, removeActivityListener, clearActivityListeners, } from "./activity.js";
285
+ // Start if run directly
286
+ if (import.meta.url === `file://${process.argv[1]}`) {
287
+ const configPath = process.argv[2];
288
+ startGateway(configPath);
289
+ }
290
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,38 @@
1
+ /**
2
+ * Persistent Mapping Store
3
+ *
4
+ * Maintains a global mapping between original values and placeholders.
5
+ * This ensures the same sensitive data always gets the same placeholder,
6
+ * allowing restoration to work across multiple requests in a conversation.
7
+ */
8
+ import type { MappingTable } from "./types.js";
9
+ /**
10
+ * Get or create a placeholder for an original value
11
+ * If the value was seen before, returns the same placeholder
12
+ */
13
+ export declare function getOrCreatePlaceholder(originalValue: string, entityType: string): string;
14
+ /**
15
+ * Get the original value for a placeholder
16
+ */
17
+ export declare function getOriginalValue(placeholder: string): string | undefined;
18
+ /**
19
+ * Check if a string is a known placeholder
20
+ */
21
+ export declare function isKnownPlaceholder(text: string): boolean;
22
+ /**
23
+ * Get all placeholders and their original values as a MappingTable
24
+ * This is used for restoration
25
+ */
26
+ export declare function getGlobalMappingTable(): MappingTable;
27
+ /**
28
+ * Get current statistics
29
+ */
30
+ export declare function getMappingStats(): {
31
+ totalMappings: number;
32
+ byType: Record<string, number>;
33
+ };
34
+ /**
35
+ * Clear all mappings (for testing or reset)
36
+ */
37
+ export declare function clearMappings(): void;
38
+ //# sourceMappingURL=mapping-store.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mapping-store.d.ts","sourceRoot":"","sources":["../src/mapping-store.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,YAAY,CAAC;AAW/C;;;GAGG;AACH,wBAAgB,sBAAsB,CAAC,aAAa,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,GAAG,MAAM,CAmBxF;AAED;;GAEG;AACH,wBAAgB,gBAAgB,CAAC,WAAW,EAAE,MAAM,GAAG,MAAM,GAAG,SAAS,CAExE;AAED;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAExD;AAED;;;GAGG;AACH,wBAAgB,qBAAqB,IAAI,YAAY,CAEpD;AAED;;GAEG;AACH,wBAAgB,eAAe,IAAI;IAAE,aAAa,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAA;CAAE,CAS3F;AAED;;GAEG;AACH,wBAAgB,aAAa,IAAI,IAAI,CAIpC"}