|
1 | 1 | import { describe, expect, test } from '@jest/globals'; |
2 | | -import { addCacheBreakpoints } from '@/lib/ai-gateway/providers/openrouter/request-helpers'; |
| 2 | +import { |
| 3 | + addCacheBreakpoints, |
| 4 | + removeCacheBreakpoints, |
| 5 | +} from '@/lib/ai-gateway/providers/openrouter/request-helpers'; |
3 | 6 | import type { GatewayRequest } from '@/lib/ai-gateway/providers/openrouter/types'; |
4 | 7 | import type OpenAI from 'openai'; |
5 | 8 |
|
@@ -251,3 +254,109 @@ describe('addCacheBreakpoints', () => { |
251 | 254 | expect(request.body.cache_control).toBeUndefined(); |
252 | 255 | }); |
253 | 256 | }); |
| 257 | + |
| 258 | +describe('removeCacheBreakpoints', () => { |
| 259 | + test('removes all cache breakpoints added to a chat completions request', () => { |
| 260 | + const request: GatewayRequest = { |
| 261 | + kind: 'chat_completions', |
| 262 | + body: { |
| 263 | + model: 'test-model', |
| 264 | + messages: [ |
| 265 | + { role: 'system', content: 'You are helpful.' }, |
| 266 | + { role: 'user', content: 'First prompt' }, |
| 267 | + { role: 'assistant', content: 'First response' }, |
| 268 | + { |
| 269 | + role: 'user', |
| 270 | + content: [ |
| 271 | + { type: 'text', text: 'Latest prompt' }, |
| 272 | + { type: 'text', text: 'Latest detail' }, |
| 273 | + ], |
| 274 | + }, |
| 275 | + ], |
| 276 | + }, |
| 277 | + }; |
| 278 | + |
| 279 | + addCacheBreakpoints(request); |
| 280 | + expect(containsCacheControlDeep(request.body.messages)).toBe(true); |
| 281 | + |
| 282 | + removeCacheBreakpoints(request); |
| 283 | + |
| 284 | + expect(containsCacheControlDeep(request.body.messages)).toBe(false); |
| 285 | + }); |
| 286 | + |
| 287 | + test('removes all cache breakpoints added to a responses request', () => { |
| 288 | + const request: GatewayRequest = { |
| 289 | + kind: 'responses', |
| 290 | + body: { |
| 291 | + model: 'test-model', |
| 292 | + input: [ |
| 293 | + { type: 'message', role: 'system', content: 'You are helpful.' }, |
| 294 | + { |
| 295 | + type: 'message', |
| 296 | + role: 'user', |
| 297 | + content: [{ type: 'input_text', text: 'First prompt' }], |
| 298 | + }, |
| 299 | + { |
| 300 | + type: 'function_call_output', |
| 301 | + call_id: 'call_123', |
| 302 | + output: [ |
| 303 | + { type: 'input_text', text: 'Tool output' }, |
| 304 | + { type: 'input_text', text: 'Tool detail' }, |
| 305 | + ], |
| 306 | + }, |
| 307 | + ], |
| 308 | + }, |
| 309 | + }; |
| 310 | + |
| 311 | + addCacheBreakpoints(request); |
| 312 | + if (request.kind !== 'responses' || !Array.isArray(request.body.input)) return; |
| 313 | + expect(containsCacheControlDeep(request.body.input)).toBe(true); |
| 314 | + |
| 315 | + removeCacheBreakpoints(request); |
| 316 | + |
| 317 | + expect(containsCacheControlDeep(request.body.input)).toBe(false); |
| 318 | + }); |
| 319 | + |
| 320 | + test('removes top-level and nested cache_control from a messages request', () => { |
| 321 | + const request: GatewayRequest = { |
| 322 | + kind: 'messages', |
| 323 | + body: { |
| 324 | + model: 'anthropic/claude-sonnet-4-5', |
| 325 | + max_tokens: 1024, |
| 326 | + cache_control: { type: 'ephemeral' }, |
| 327 | + messages: [ |
| 328 | + { |
| 329 | + role: 'user', |
| 330 | + content: [ |
| 331 | + { |
| 332 | + type: 'text', |
| 333 | + text: 'First prompt', |
| 334 | + cache_control: { type: 'ephemeral' }, |
| 335 | + }, |
| 336 | + ], |
| 337 | + }, |
| 338 | + { role: 'assistant', content: 'First response' }, |
| 339 | + { role: 'user', content: 'Latest prompt' }, |
| 340 | + ], |
| 341 | + }, |
| 342 | + }; |
| 343 | + |
| 344 | + removeCacheBreakpoints(request); |
| 345 | + |
| 346 | + expect(request.body.cache_control).toBeUndefined(); |
| 347 | + expect(containsCacheControlDeep(request.body.messages)).toBe(false); |
| 348 | + }); |
| 349 | +}); |
| 350 | + |
| 351 | +function containsCacheControlDeep(value: unknown): boolean { |
| 352 | + if (Array.isArray(value)) { |
| 353 | + return value.some(containsCacheControlDeep); |
| 354 | + } |
| 355 | + if (typeof value !== 'object' || value === null) { |
| 356 | + return false; |
| 357 | + } |
| 358 | + if (Object.hasOwn(value, 'cache_control')) { |
| 359 | + return true; |
| 360 | + } |
| 361 | + return Object.values(value).some(containsCacheControlDeep); |
| 362 | +} |
0 commit comments