90a70c8404
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / iace-gt-coverage (push) Has been skipped
CI / test-python-backend (push) Has been skipped
CI / test-python-document-crawler (push) Has been skipped
CI / test-python-dsms-gateway (push) Has been skipped
CI / detect-changes (push) Successful in 7s
CI / branch-name (push) Has been skipped
CI / guardrail-integrity (push) Has been skipped
CI / secret-scan (push) Has been skipped
CI / dep-audit (push) Has been skipped
CI / sbom-scan (push) Has been skipped
CI / build-sha-integrity (push) Successful in 5s
CI / validate-canonical-controls (push) Successful in 4s
CI / loc-budget (push) Successful in 17s
CI / go-lint (push) Has been skipped
CI / nodejs-build (push) Successful in 3m2s
CI / test-go (push) Has been skipped
Die Drafting-Engine (Dokument-Entwurf, v2-Pipeline, Validierung, Drafting-Chat, Vendor-Vertragspruefung) war auf prod doppelt tot: - RAG ueber bp-core-rag-service:8097 (existiert auf prod nicht) - LLM ueber OLLAMA_URL/api/chat mit qwen2.5vl (prod = ollama-embed, kein Chat-Modell) Fix (analog zum Compliance-Advisor): - rag-query.ts -> ai-compliance-sdk /sdk/v1/rag/search (bge-m3, prod-erreichbar). - Neue lib/sdk/drafting-engine/llm-cascade.ts: OVH/LiteLLM (gpt-oss-120b) zuerst, Ollama als Dev-Fallback; cascadeComplete (JSON) + cascadeStream. Das Backend nutzt OVH+JSON bereits erfolgreich auf prod (extract-datasheet). - 5 Aufrufstellen (draft-helpers, draft-helpers-v2, validate, chat, vendor-review) auf die Kaskade umgestellt; keine direkten Ollama-Calls mehr. - Tests: llm-cascade + rag-query aktualisiert. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
128 lines
3.8 KiB
TypeScript
128 lines
3.8 KiB
TypeScript
/**
|
|
* Tests for the shared queryRAG utility (ai-sdk /sdk/v1/rag/search, bge-m3).
|
|
*/
|
|
|
|
import { describe, it, expect, beforeEach, vi } from 'vitest'
|
|
|
|
// Mock fetch globally
|
|
const mockFetch = vi.fn()
|
|
vi.stubGlobal('fetch', mockFetch)
|
|
|
|
describe('queryRAG', () => {
|
|
let queryRAG: (query: string, topK?: number, collection?: string) => Promise<string>
|
|
|
|
beforeEach(async () => {
|
|
vi.resetModules()
|
|
mockFetch.mockReset()
|
|
// Dynamic import to pick up fresh env
|
|
const mod = await import('../rag-query')
|
|
queryRAG = mod.queryRAG
|
|
})
|
|
|
|
it('should return formatted results on success (ai-sdk shape)', async () => {
|
|
mockFetch.mockResolvedValueOnce({
|
|
ok: true,
|
|
json: async () => ({
|
|
results: [
|
|
{ text: 'Art. 35 regelt die DSFA...', regulation_short: 'DSGVO' },
|
|
{ text: 'Risikobewertung erforderlich', regulation_code: 'EU_2016_679' },
|
|
],
|
|
}),
|
|
})
|
|
|
|
const result = await queryRAG('DSFA Art. 35')
|
|
|
|
expect(result).toContain('[Quelle 1: DSGVO]')
|
|
expect(result).toContain('Art. 35 regelt die DSFA...')
|
|
expect(result).toContain('[Quelle 2: EU_2016_679]')
|
|
expect(mockFetch).toHaveBeenCalledTimes(1)
|
|
})
|
|
|
|
it('should POST to the ai-sdk /sdk/v1/rag/search endpoint', async () => {
|
|
mockFetch.mockResolvedValueOnce({
|
|
ok: true,
|
|
json: async () => ({ results: [] }),
|
|
})
|
|
|
|
await queryRAG('test query')
|
|
|
|
expect(mockFetch).toHaveBeenCalledWith(
|
|
expect.stringContaining('/sdk/v1/rag/search'),
|
|
expect.objectContaining({
|
|
method: 'POST',
|
|
headers: expect.objectContaining({ 'Content-Type': 'application/json' }),
|
|
})
|
|
)
|
|
})
|
|
|
|
it('should include collection in request body when provided', async () => {
|
|
mockFetch.mockResolvedValueOnce({
|
|
ok: true,
|
|
json: async () => ({ results: [] }),
|
|
})
|
|
|
|
await queryRAG('test query', 3, 'bp_dsfa_corpus')
|
|
|
|
const callArgs = mockFetch.mock.calls[0]
|
|
const body = JSON.parse(callArgs[1].body)
|
|
expect(body.collection).toBe('bp_dsfa_corpus')
|
|
expect(body.query).toBe('test query')
|
|
expect(body.top_k).toBe(3)
|
|
})
|
|
|
|
it('should omit collection from body when not provided', async () => {
|
|
mockFetch.mockResolvedValueOnce({
|
|
ok: true,
|
|
json: async () => ({ results: [] }),
|
|
})
|
|
|
|
await queryRAG('test query')
|
|
|
|
const callArgs = mockFetch.mock.calls[0]
|
|
const body = JSON.parse(callArgs[1].body)
|
|
expect(body.collection).toBeUndefined()
|
|
expect(body.query).toBe('test query')
|
|
expect(body.top_k).toBe(3)
|
|
})
|
|
|
|
it('should pass custom topK in request body', async () => {
|
|
mockFetch.mockResolvedValueOnce({
|
|
ok: true,
|
|
json: async () => ({ results: [] }),
|
|
})
|
|
|
|
await queryRAG('test query', 7)
|
|
|
|
const callArgs = mockFetch.mock.calls[0]
|
|
const body = JSON.parse(callArgs[1].body)
|
|
expect(body.top_k).toBe(7)
|
|
})
|
|
|
|
it('should return empty string on HTTP error', async () => {
|
|
mockFetch.mockResolvedValueOnce({ ok: false, status: 500 })
|
|
expect(await queryRAG('test query')).toBe('')
|
|
})
|
|
|
|
it('should return empty string on network error', async () => {
|
|
mockFetch.mockRejectedValueOnce(new Error('Connection refused'))
|
|
expect(await queryRAG('test query')).toBe('')
|
|
})
|
|
|
|
it('should return empty string when no results', async () => {
|
|
mockFetch.mockResolvedValueOnce({ ok: true, json: async () => ({ results: [] }) })
|
|
expect(await queryRAG('test query')).toBe('')
|
|
})
|
|
|
|
it('should handle results with missing source fields gracefully', async () => {
|
|
mockFetch.mockResolvedValueOnce({
|
|
ok: true,
|
|
json: async () => ({ results: [{ text: 'Some content without source' }] }),
|
|
})
|
|
|
|
const result = await queryRAG('test')
|
|
|
|
expect(result).toContain('[Quelle 1: Unbekannt]')
|
|
expect(result).toContain('Some content without source')
|
|
})
|
|
})
|