Files
sigil/test/s13-deploy-cooldown.test.ts
T
xiaoju e86bae8d4a refactor: migrate to Dynamic Workers — fix /run/{name} 404
Root cause: CF blocks Worker-to-Worker fetch on workers.dev (error 1042).
Gateway Worker could not proxy requests to child worker subdomains.

Fix: Replace CF API worker scripts with Dynamic Workers (LOADER binding).
- deploy() writes code to KV only, no CF API calls
- invoke() uses LOADER.get(id, fn) to execute code inline
- remove() clears KV only, no CF API delete
- Removed cf-api.ts, slot management, subdomain routing
- 67/67 tests passing, production verified

Reported-by: 小墨 🖊️ (KUMA)
小橘 🍊(NEKO Team)
2026-04-03 10:57:50 +00:00

86 lines
2.8 KiB
TypeScript

import { describe, it, expect, beforeEach } from 'vitest'
import { createMockKv, createMockLoader, makeRequest, MockEmbeddingService } from './setup.js'
import { WorkerPool } from '../src/backend/worker-pool.js'
import { AuthModule } from '../src/auth.js'
import { KvStore } from '../src/kv.js'
import { handleRequest } from '../src/router.js'
describe('S13: deploy_cooldown', () => {
let mockKv: KVNamespace
let mockLoader: ReturnType<typeof createMockLoader>
let mockEmbed: MockEmbeddingService
let pool: WorkerPool
let auth: AuthModule
let kv: KvStore
beforeEach(async () => {
mockKv = createMockKv()
mockLoader = createMockLoader()
mockEmbed = new MockEmbeddingService()
pool = new WorkerPool(mockKv, mockLoader.loader, mockEmbed as any)
kv = new KvStore(mockKv)
auth = new AuthModule(kv)
await auth.setToken('deploy-token')
})
it('should reject rapid second deploy with 429', async () => {
// First deploy
const req1 = makeRequest('POST', '/_api/deploy', {
token: 'deploy-token',
body: {
name: 'ping',
code: '// ping',
type: 'normal',
},
})
const resp1 = await handleRequest(req1, { SIGIL_KV: mockKv, backend: pool, auth, kv })
expect(resp1.status).toBe(201)
// Immediate second deploy (< 5s cooldown)
const req2 = makeRequest('POST', '/_api/deploy', {
token: 'deploy-token',
body: {
name: 'ping2',
code: '// ping2',
type: 'normal',
},
})
const resp2 = await handleRequest(req2, { SIGIL_KV: mockKv, backend: pool, auth, kv })
expect(resp2.status).toBe(429)
})
it('should include retry_after in 429 response', async () => {
// First deploy
const req1 = makeRequest('POST', '/_api/deploy', {
token: 'deploy-token',
body: { name: 'ping', code: '// ping', type: 'normal' },
})
await handleRequest(req1, { SIGIL_KV: mockKv, backend: pool, auth, kv })
// Immediate second
const req2 = makeRequest('POST', '/_api/deploy', {
token: 'deploy-token',
body: { name: 'ping2', code: '// ping2', type: 'normal' },
})
const resp2 = await handleRequest(req2, { SIGIL_KV: mockKv, backend: pool, auth, kv })
const body = await resp2.json() as { error: string; retry_after: number }
expect(body.retry_after).toBeGreaterThan(0)
expect(body.retry_after).toBeLessThanOrEqual(5)
})
it('should allow deploy after cooldown expires', async () => {
// Manually set last deploy time as already expired
await kv.setLastDeployTime(Date.now() - 10000) // 10s ago, past 5s cooldown
const req = makeRequest('POST', '/_api/deploy', {
token: 'deploy-token',
body: { name: 'ping', code: '// ping', type: 'normal' },
})
const resp = await handleRequest(req, { SIGIL_KV: mockKv, backend: pool, auth, kv })
expect(resp.status).toBe(201)
})
})