chore: diverse Bereinigungen und Fixes
All checks were successful
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / test-go-ai-compliance (push) Successful in 35s
CI / test-python-backend-compliance (push) Successful in 30s
CI / test-python-document-crawler (push) Successful in 20s
CI / test-python-dsms-gateway (push) Successful in 28s
All checks were successful
CI / go-lint (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / test-go-ai-compliance (push) Successful in 35s
CI / test-python-backend-compliance (push) Successful in 30s
CI / test-python-document-crawler (push) Successful in 20s
CI / test-python-dsms-gateway (push) Successful in 28s
- admin-compliance: .dockerignore + Dockerfile bereinigt - dsfa-corpus/route.ts + legal-corpus/route.ts entfernt (obsolet) - webhooks/woodpecker/route.ts: minor fix - dsfa/[id]/page.tsx: Refactoring - service_modules.py + README.md: aktualisiert - Migration 028 → 032 umbenannt (legal_documents_extend) - docs: index.md + DEVELOPER.md aktualisiert Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -20,7 +20,6 @@ edu-search-service
|
|||||||
school-service
|
school-service
|
||||||
voice-service
|
voice-service
|
||||||
geo-service
|
geo-service
|
||||||
klausur-service
|
|
||||||
studio-v2
|
studio-v2
|
||||||
website
|
website
|
||||||
scripts
|
scripts
|
||||||
|
|||||||
@@ -16,13 +16,11 @@ COPY . .
|
|||||||
ARG NEXT_PUBLIC_API_URL
|
ARG NEXT_PUBLIC_API_URL
|
||||||
ARG NEXT_PUBLIC_OLD_ADMIN_URL
|
ARG NEXT_PUBLIC_OLD_ADMIN_URL
|
||||||
ARG NEXT_PUBLIC_SDK_URL
|
ARG NEXT_PUBLIC_SDK_URL
|
||||||
ARG NEXT_PUBLIC_KLAUSUR_SERVICE_URL
|
|
||||||
|
|
||||||
# Set environment variables for build
|
# Set environment variables for build
|
||||||
ENV NEXT_PUBLIC_API_URL=$NEXT_PUBLIC_API_URL
|
ENV NEXT_PUBLIC_API_URL=$NEXT_PUBLIC_API_URL
|
||||||
ENV NEXT_PUBLIC_OLD_ADMIN_URL=$NEXT_PUBLIC_OLD_ADMIN_URL
|
ENV NEXT_PUBLIC_OLD_ADMIN_URL=$NEXT_PUBLIC_OLD_ADMIN_URL
|
||||||
ENV NEXT_PUBLIC_SDK_URL=$NEXT_PUBLIC_SDK_URL
|
ENV NEXT_PUBLIC_SDK_URL=$NEXT_PUBLIC_SDK_URL
|
||||||
ENV NEXT_PUBLIC_KLAUSUR_SERVICE_URL=$NEXT_PUBLIC_KLAUSUR_SERVICE_URL
|
|
||||||
|
|
||||||
# Build the application
|
# Build the application
|
||||||
RUN npm run build
|
RUN npm run build
|
||||||
|
|||||||
@@ -1,100 +0,0 @@
|
|||||||
/**
|
|
||||||
* DSFA Corpus API Proxy
|
|
||||||
*
|
|
||||||
* Proxies requests to klausur-service for DSFA RAG operations.
|
|
||||||
* Endpoints: /api/v1/dsfa-rag/stats, /api/v1/dsfa-rag/sources
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { NextRequest, NextResponse } from 'next/server'
|
|
||||||
|
|
||||||
const KLAUSUR_SERVICE_URL = process.env.KLAUSUR_SERVICE_URL || 'http://klausur-service:8086'
|
|
||||||
|
|
||||||
export async function GET(request: NextRequest) {
|
|
||||||
const { searchParams } = new URL(request.url)
|
|
||||||
const action = searchParams.get('action')
|
|
||||||
|
|
||||||
try {
|
|
||||||
let url = `${KLAUSUR_SERVICE_URL}/api/v1/dsfa-rag`
|
|
||||||
|
|
||||||
switch (action) {
|
|
||||||
case 'status':
|
|
||||||
url += '/stats'
|
|
||||||
break
|
|
||||||
case 'sources':
|
|
||||||
url += '/sources'
|
|
||||||
break
|
|
||||||
case 'source-detail': {
|
|
||||||
const code = searchParams.get('code')
|
|
||||||
if (!code) {
|
|
||||||
return NextResponse.json({ error: 'Missing code parameter' }, { status: 400 })
|
|
||||||
}
|
|
||||||
url += `/sources/${encodeURIComponent(code)}`
|
|
||||||
break
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return NextResponse.json({ error: 'Unknown action' }, { status: 400 })
|
|
||||||
}
|
|
||||||
|
|
||||||
const res = await fetch(url, {
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
cache: 'no-store',
|
|
||||||
})
|
|
||||||
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
} catch (error) {
|
|
||||||
console.error('DSFA corpus proxy error:', error)
|
|
||||||
return NextResponse.json(
|
|
||||||
{ error: 'Failed to connect to klausur-service' },
|
|
||||||
{ status: 503 }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function POST(request: NextRequest) {
|
|
||||||
const { searchParams } = new URL(request.url)
|
|
||||||
const action = searchParams.get('action')
|
|
||||||
|
|
||||||
try {
|
|
||||||
let url = `${KLAUSUR_SERVICE_URL}/api/v1/dsfa-rag`
|
|
||||||
|
|
||||||
switch (action) {
|
|
||||||
case 'init': {
|
|
||||||
url += '/init'
|
|
||||||
const res = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
})
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
case 'ingest': {
|
|
||||||
const body = await request.json()
|
|
||||||
const sourceCode = body.source_code
|
|
||||||
if (!sourceCode) {
|
|
||||||
return NextResponse.json({ error: 'Missing source_code' }, { status: 400 })
|
|
||||||
}
|
|
||||||
url += `/sources/${encodeURIComponent(sourceCode)}/ingest`
|
|
||||||
const res = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify(body),
|
|
||||||
})
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
return NextResponse.json({ error: 'Unknown action' }, { status: 400 })
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('DSFA corpus proxy error:', error)
|
|
||||||
return NextResponse.json(
|
|
||||||
{ error: 'Failed to connect to klausur-service' },
|
|
||||||
{ status: 503 }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,180 +0,0 @@
|
|||||||
/**
|
|
||||||
* Legal Corpus API Proxy
|
|
||||||
*
|
|
||||||
* Proxies requests to klausur-service for RAG operations.
|
|
||||||
* This allows the client-side RAG page to call the API without CORS issues.
|
|
||||||
*/
|
|
||||||
|
|
||||||
import { NextRequest, NextResponse } from 'next/server'
|
|
||||||
|
|
||||||
const KLAUSUR_SERVICE_URL = process.env.KLAUSUR_SERVICE_URL || 'http://klausur-service:8086'
|
|
||||||
const QDRANT_URL = process.env.QDRANT_URL || 'http://qdrant:6333'
|
|
||||||
|
|
||||||
export async function GET(request: NextRequest) {
|
|
||||||
const { searchParams } = new URL(request.url)
|
|
||||||
const action = searchParams.get('action')
|
|
||||||
|
|
||||||
try {
|
|
||||||
let url = `${KLAUSUR_SERVICE_URL}/api/v1/admin/legal-corpus`
|
|
||||||
|
|
||||||
switch (action) {
|
|
||||||
case 'status': {
|
|
||||||
// Query Qdrant directly for collection stats
|
|
||||||
const qdrantRes = await fetch(`${QDRANT_URL}/collections/bp_legal_corpus`, {
|
|
||||||
cache: 'no-store',
|
|
||||||
})
|
|
||||||
if (!qdrantRes.ok) {
|
|
||||||
return NextResponse.json({ error: 'Qdrant not available' }, { status: 503 })
|
|
||||||
}
|
|
||||||
const qdrantData = await qdrantRes.json()
|
|
||||||
const result = qdrantData.result || {}
|
|
||||||
return NextResponse.json({
|
|
||||||
collection: 'bp_legal_corpus',
|
|
||||||
totalPoints: result.points_count || 0,
|
|
||||||
vectorSize: result.config?.params?.vectors?.size || 0,
|
|
||||||
status: result.status || 'unknown',
|
|
||||||
regulations: {},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
case 'search':
|
|
||||||
const query = searchParams.get('query')
|
|
||||||
const topK = searchParams.get('top_k') || '5'
|
|
||||||
const regulations = searchParams.get('regulations')
|
|
||||||
url += `/search?query=${encodeURIComponent(query || '')}&top_k=${topK}`
|
|
||||||
if (regulations) {
|
|
||||||
url += `®ulations=${encodeURIComponent(regulations)}`
|
|
||||||
}
|
|
||||||
break
|
|
||||||
case 'ingestion-status':
|
|
||||||
url += '/ingestion-status'
|
|
||||||
break
|
|
||||||
case 'regulations':
|
|
||||||
url += '/regulations'
|
|
||||||
break
|
|
||||||
case 'custom-documents':
|
|
||||||
url += '/custom-documents'
|
|
||||||
break
|
|
||||||
case 'pipeline-checkpoints':
|
|
||||||
url = `${KLAUSUR_SERVICE_URL}/api/v1/admin/pipeline/checkpoints`
|
|
||||||
break
|
|
||||||
case 'pipeline-status':
|
|
||||||
url = `${KLAUSUR_SERVICE_URL}/api/v1/admin/pipeline/status`
|
|
||||||
break
|
|
||||||
case 'traceability': {
|
|
||||||
const chunkId = searchParams.get('chunk_id')
|
|
||||||
const regulation = searchParams.get('regulation')
|
|
||||||
url += `/traceability?chunk_id=${encodeURIComponent(chunkId || '')}®ulation=${encodeURIComponent(regulation || '')}`
|
|
||||||
break
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return NextResponse.json({ error: 'Unknown action' }, { status: 400 })
|
|
||||||
}
|
|
||||||
|
|
||||||
const res = await fetch(url, {
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json',
|
|
||||||
},
|
|
||||||
cache: 'no-store',
|
|
||||||
})
|
|
||||||
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Legal corpus proxy error:', error)
|
|
||||||
return NextResponse.json(
|
|
||||||
{ error: 'Failed to connect to klausur-service' },
|
|
||||||
{ status: 503 }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function POST(request: NextRequest) {
|
|
||||||
const { searchParams } = new URL(request.url)
|
|
||||||
const action = searchParams.get('action')
|
|
||||||
|
|
||||||
try {
|
|
||||||
let url = `${KLAUSUR_SERVICE_URL}/api/v1/admin/legal-corpus`
|
|
||||||
|
|
||||||
switch (action) {
|
|
||||||
case 'ingest': {
|
|
||||||
url += '/ingest'
|
|
||||||
const body = await request.json()
|
|
||||||
const res = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify(body),
|
|
||||||
})
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
case 'add-link': {
|
|
||||||
url += '/add-link'
|
|
||||||
const body = await request.json()
|
|
||||||
const res = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify(body),
|
|
||||||
})
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
case 'upload': {
|
|
||||||
url += '/upload'
|
|
||||||
// Forward FormData directly
|
|
||||||
const formData = await request.formData()
|
|
||||||
const res = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
body: formData,
|
|
||||||
})
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
case 'start-pipeline': {
|
|
||||||
url = `${KLAUSUR_SERVICE_URL}/api/v1/admin/pipeline/start`
|
|
||||||
const body = await request.json()
|
|
||||||
const res = await fetch(url, {
|
|
||||||
method: 'POST',
|
|
||||||
headers: { 'Content-Type': 'application/json' },
|
|
||||||
body: JSON.stringify(body),
|
|
||||||
})
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
return NextResponse.json({ error: 'Unknown action' }, { status: 400 })
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Legal corpus proxy error:', error)
|
|
||||||
return NextResponse.json(
|
|
||||||
{ error: 'Failed to connect to klausur-service' },
|
|
||||||
{ status: 503 }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function DELETE(request: NextRequest) {
|
|
||||||
const { searchParams } = new URL(request.url)
|
|
||||||
const action = searchParams.get('action')
|
|
||||||
const docId = searchParams.get('docId')
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (action === 'delete-document' && docId) {
|
|
||||||
const url = `${KLAUSUR_SERVICE_URL}/api/v1/admin/legal-corpus/custom-documents/${docId}`
|
|
||||||
const res = await fetch(url, { method: 'DELETE' })
|
|
||||||
const data = await res.json()
|
|
||||||
return NextResponse.json(data, { status: res.status })
|
|
||||||
}
|
|
||||||
|
|
||||||
return NextResponse.json({ error: 'Unknown action' }, { status: 400 })
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Legal corpus proxy error:', error)
|
|
||||||
return NextResponse.json(
|
|
||||||
{ error: 'Failed to connect to klausur-service' },
|
|
||||||
{ status: 503 }
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -14,7 +14,7 @@ const LOG_EXTRACT_URL = process.env.NEXT_PUBLIC_APP_URL
|
|||||||
: 'http://localhost:3002/api/infrastructure/log-extract/extract'
|
: 'http://localhost:3002/api/infrastructure/log-extract/extract'
|
||||||
|
|
||||||
// Test service API URL for backlog insertion
|
// Test service API URL for backlog insertion
|
||||||
const TEST_SERVICE_URL = process.env.TEST_SERVICE_URL || 'http://localhost:8086'
|
const TEST_SERVICE_URL = process.env.TEST_SERVICE_URL || 'http://localhost:8002'
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
// Helper Functions
|
// Helper Functions
|
||||||
|
|||||||
@@ -53,8 +53,6 @@ import {
|
|||||||
ReviewScheduleSection,
|
ReviewScheduleSection,
|
||||||
AIUseCaseSection,
|
AIUseCaseSection,
|
||||||
} from '@/components/sdk/dsfa'
|
} from '@/components/sdk/dsfa'
|
||||||
import { SourceAttribution } from '@/components/sdk/dsfa/SourceAttribution'
|
|
||||||
import type { DSFALicenseCode, SourceAttributionProps } from '@/lib/sdk/types'
|
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
// SECTION EDITORS
|
// SECTION EDITORS
|
||||||
@@ -967,30 +965,21 @@ function SDMCoverageOverview({ dsfa }: { dsfa: DSFA }) {
|
|||||||
// RAG SEARCH PANEL
|
// RAG SEARCH PANEL
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
|
|
||||||
const RAG_API_BASE = process.env.NEXT_PUBLIC_KLAUSUR_SERVICE_URL || 'http://localhost:8086'
|
|
||||||
|
|
||||||
interface RAGSearchResult {
|
interface RAGSearchResult {
|
||||||
chunk_id: string
|
text: string
|
||||||
content: string
|
regulation_code: string
|
||||||
score: number
|
regulation_name: string
|
||||||
source_code: string
|
regulation_short: string
|
||||||
source_name: string
|
category: string
|
||||||
attribution_text: string
|
article?: string
|
||||||
license_code: string
|
|
||||||
license_name: string
|
|
||||||
license_url?: string
|
|
||||||
source_url?: string
|
source_url?: string
|
||||||
document_type?: string
|
score: number
|
||||||
category?: string
|
|
||||||
section_title?: string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface RAGSearchResponse {
|
interface RAGSearchResponse {
|
||||||
query: string
|
query: string
|
||||||
results: RAGSearchResult[]
|
results: RAGSearchResult[]
|
||||||
total_results: number
|
count: number
|
||||||
licenses_used: string[]
|
|
||||||
attribution_notice: string
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function RAGSearchPanel({
|
function RAGSearchPanel({
|
||||||
@@ -1023,12 +1012,15 @@ function RAGSearchPanel({
|
|||||||
setError(null)
|
setError(null)
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const params = new URLSearchParams({ query: searchQuery, limit: '5' })
|
const response = await fetch('/api/sdk/v1/rag/search', {
|
||||||
if (categories?.length) {
|
method: 'POST',
|
||||||
categories.forEach(c => params.append('categories', c))
|
headers: { 'Content-Type': 'application/json' },
|
||||||
}
|
body: JSON.stringify({
|
||||||
|
query: searchQuery,
|
||||||
const response = await fetch(`${RAG_API_BASE}/api/v1/dsfa-rag/search?${params}`)
|
collection: 'bp_dsfa_corpus',
|
||||||
|
top_k: 5,
|
||||||
|
}),
|
||||||
|
})
|
||||||
if (!response.ok) throw new Error(`Suche fehlgeschlagen (${response.status})`)
|
if (!response.ok) throw new Error(`Suche fehlgeschlagen (${response.status})`)
|
||||||
const data: RAGSearchResponse = await response.json()
|
const data: RAGSearchResponse = await response.json()
|
||||||
setResults(data)
|
setResults(data)
|
||||||
@@ -1040,25 +1032,16 @@ function RAGSearchPanel({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleInsert = (text: string, chunkId: string) => {
|
const handleInsert = (text: string, resultId: string) => {
|
||||||
if (onInsertText) {
|
if (onInsertText) {
|
||||||
onInsertText(text)
|
onInsertText(text)
|
||||||
} else {
|
} else {
|
||||||
navigator.clipboard.writeText(text)
|
navigator.clipboard.writeText(text)
|
||||||
}
|
}
|
||||||
setCopiedId(chunkId)
|
setCopiedId(resultId)
|
||||||
setTimeout(() => setCopiedId(null), 2000)
|
setTimeout(() => setCopiedId(null), 2000)
|
||||||
}
|
}
|
||||||
|
|
||||||
const sourcesForAttribution: SourceAttributionProps['sources'] = (results?.results || []).map(r => ({
|
|
||||||
sourceCode: r.source_code,
|
|
||||||
sourceName: r.source_name,
|
|
||||||
attributionText: r.attribution_text,
|
|
||||||
licenseCode: r.license_code as DSFALicenseCode,
|
|
||||||
sourceUrl: r.source_url,
|
|
||||||
score: r.score,
|
|
||||||
}))
|
|
||||||
|
|
||||||
if (!isOpen) {
|
if (!isOpen) {
|
||||||
return (
|
return (
|
||||||
<button
|
<button
|
||||||
@@ -1121,44 +1104,47 @@ function RAGSearchPanel({
|
|||||||
{/* Results */}
|
{/* Results */}
|
||||||
{results && results.results.length > 0 && (
|
{results && results.results.length > 0 && (
|
||||||
<div className="space-y-3">
|
<div className="space-y-3">
|
||||||
<p className="text-xs text-indigo-600">{results.total_results} Ergebnis(se) gefunden</p>
|
<p className="text-xs text-indigo-600">{results.count} Ergebnis(se) gefunden</p>
|
||||||
|
|
||||||
{results.results.map(r => (
|
{results.results.map((r, idx) => {
|
||||||
<div key={r.chunk_id} className="bg-white rounded-lg border border-indigo-100 p-3">
|
const resultId = `${r.regulation_code}-${idx}`
|
||||||
<div className="flex items-start justify-between gap-2">
|
return (
|
||||||
<div className="flex-1 min-w-0">
|
<div key={resultId} className="bg-white rounded-lg border border-indigo-100 p-3">
|
||||||
{r.section_title && (
|
<div className="flex items-start justify-between gap-2">
|
||||||
<div className="text-xs font-medium text-indigo-600 mb-1">{r.section_title}</div>
|
<div className="flex-1 min-w-0">
|
||||||
)}
|
<div className="text-xs font-medium text-indigo-600 mb-1">
|
||||||
<p className="text-sm text-gray-700 leading-relaxed whitespace-pre-line">
|
{r.regulation_name}{r.article ? ` — ${r.article}` : ''}
|
||||||
{r.content.length > 400 ? r.content.substring(0, 400) + '...' : r.content}
|
</div>
|
||||||
</p>
|
<p className="text-sm text-gray-700 leading-relaxed whitespace-pre-line">
|
||||||
<div className="flex items-center gap-2 mt-2">
|
{r.text.length > 400 ? r.text.substring(0, 400) + '...' : r.text}
|
||||||
<span className="text-xs text-gray-400 font-mono">
|
</p>
|
||||||
{r.source_code} ({(r.score * 100).toFixed(0)}%)
|
<div className="flex items-center gap-2 mt-2">
|
||||||
</span>
|
<span className="text-xs text-gray-400 font-mono">
|
||||||
{r.category && (
|
{r.regulation_short || r.regulation_code} ({(r.score * 100).toFixed(0)}%)
|
||||||
<span className="text-xs px-1.5 py-0.5 rounded bg-gray-100 text-gray-500">{r.category}</span>
|
</span>
|
||||||
)}
|
{r.category && (
|
||||||
|
<span className="text-xs px-1.5 py-0.5 rounded bg-gray-100 text-gray-500">{r.category}</span>
|
||||||
|
)}
|
||||||
|
{r.source_url && (
|
||||||
|
<a href={r.source_url} target="_blank" rel="noopener noreferrer" className="text-xs text-blue-500 hover:underline">Quelle</a>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<button
|
||||||
|
onClick={() => handleInsert(r.text, resultId)}
|
||||||
|
className={`flex-shrink-0 px-3 py-1.5 text-xs rounded-lg transition-colors ${
|
||||||
|
copiedId === resultId
|
||||||
|
? 'bg-green-100 text-green-700'
|
||||||
|
: 'bg-indigo-100 text-indigo-700 hover:bg-indigo-200'
|
||||||
|
}`}
|
||||||
|
title="In Beschreibung uebernehmen"
|
||||||
|
>
|
||||||
|
{copiedId === resultId ? 'Kopiert!' : 'Uebernehmen'}
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<button
|
|
||||||
onClick={() => handleInsert(r.content, r.chunk_id)}
|
|
||||||
className={`flex-shrink-0 px-3 py-1.5 text-xs rounded-lg transition-colors ${
|
|
||||||
copiedId === r.chunk_id
|
|
||||||
? 'bg-green-100 text-green-700'
|
|
||||||
: 'bg-indigo-100 text-indigo-700 hover:bg-indigo-200'
|
|
||||||
}`}
|
|
||||||
title="In Beschreibung uebernehmen"
|
|
||||||
>
|
|
||||||
{copiedId === r.chunk_id ? 'Kopiert!' : 'Uebernehmen'}
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
)
|
||||||
))}
|
})}
|
||||||
|
|
||||||
{/* Source Attribution */}
|
|
||||||
<SourceAttribution sources={sourcesForAttribution} compact showScores />
|
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ Diese Dateien enthalten die Seed-Daten für das Compliance-Modul.
|
|||||||
|-----|--------------|-----------|
|
|-----|--------------|-----------|
|
||||||
| `backend` | API/Backend Services | consent-service, python-backend |
|
| `backend` | API/Backend Services | consent-service, python-backend |
|
||||||
| `database` | Datenbanken | PostgreSQL, Qdrant, Valkey |
|
| `database` | Datenbanken | PostgreSQL, Qdrant, Valkey |
|
||||||
| `ai` | KI/ML Services | klausur-service, embedding-service |
|
| `ai` | KI/ML Services | embedding-service, ai-compliance-sdk |
|
||||||
| `communication` | Chat/Video | Matrix, Jitsi |
|
| `communication` | Chat/Video | Matrix, Jitsi |
|
||||||
| `storage` | Speichersysteme | MinIO, DSMS |
|
| `storage` | Speichersysteme | MinIO, DSMS |
|
||||||
| `infrastructure` | Infrastruktur | Vault, Mailpit, Backup |
|
| `infrastructure` | Infrastruktur | Vault, Mailpit, Backup |
|
||||||
|
|||||||
@@ -93,7 +93,7 @@ BREAKPILOT_SERVICES: List[Dict[str, Any]] = [
|
|||||||
{"code": "DSA", "relevance": LOW, "notes": "Transparenz bei Gebühren"},
|
{"code": "DSA", "relevance": LOW, "notes": "Transparenz bei Gebühren"},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{ # Lehrer-Stack (keine Compliance-Runtime-Dependency)
|
||||||
"name": "school-service",
|
"name": "school-service",
|
||||||
"display_name": "School Service",
|
"display_name": "School Service",
|
||||||
"description": "Schulverwaltung, Klassen, Noten und Zeugnisse",
|
"description": "Schulverwaltung, Klassen, Noten und Zeugnisse",
|
||||||
@@ -113,7 +113,7 @@ BREAKPILOT_SERVICES: List[Dict[str, Any]] = [
|
|||||||
{"code": "BSI-TR-03161-1", "relevance": HIGH, "notes": "Sicherheit für Bildungsanwendungen"},
|
{"code": "BSI-TR-03161-1", "relevance": HIGH, "notes": "Sicherheit für Bildungsanwendungen"},
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
{
|
{ # Lehrer-Stack (keine Compliance-Runtime-Dependency)
|
||||||
"name": "calendar-service",
|
"name": "calendar-service",
|
||||||
"display_name": "Calendar Service",
|
"display_name": "Calendar Service",
|
||||||
"description": "Kalender, Termine und Stundenplanung",
|
"description": "Kalender, Termine und Stundenplanung",
|
||||||
@@ -136,7 +136,7 @@ BREAKPILOT_SERVICES: List[Dict[str, Any]] = [
|
|||||||
# =========================================================================
|
# =========================================================================
|
||||||
# AI / ML SERVICES
|
# AI / ML SERVICES
|
||||||
# =========================================================================
|
# =========================================================================
|
||||||
{
|
{ # Lehrer-Stack (keine Compliance-Runtime-Dependency)
|
||||||
"name": "klausur-service",
|
"name": "klausur-service",
|
||||||
"display_name": "Klausur Service (AI Correction)",
|
"display_name": "Klausur Service (AI Correction)",
|
||||||
"description": "KI-gestützte Klausurbewertung, PDF-Analyse und Feedback-Generierung",
|
"description": "KI-gestützte Klausurbewertung, PDF-Analyse und Feedback-Generierung",
|
||||||
|
|||||||
@@ -10,7 +10,8 @@ Willkommen zur Dokumentation des **BreakPilot Compliance**-Stacks (Team B: DSGVO
|
|||||||
| **breakpilot-lehrer** | Bildungs-Stack | Port 8010 |
|
| **breakpilot-lehrer** | Bildungs-Stack | Port 8010 |
|
||||||
| **breakpilot-compliance** (dieses Projekt) | DSGVO/Compliance-Stack | Port 8011 |
|
| **breakpilot-compliance** (dieses Projekt) | DSGVO/Compliance-Stack | Port 8011 |
|
||||||
|
|
||||||
Compliance haengt von Core ab (PostgreSQL, Valkey, Vault, Qdrant, MinIO, Embedding, RAG).
|
Compliance haengt **ausschliesslich von Core** ab (PostgreSQL, Valkey, Vault, Qdrant, MinIO, Embedding, RAG).
|
||||||
|
Es gibt **keine Laufzeitabhaengigkeit** zu breakpilot-lehrer.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
|||||||
@@ -323,7 +323,7 @@ context_str = rag.format_for_prompt(results)
|
|||||||
```typescript
|
```typescript
|
||||||
import { queryRAG } from '@/lib/sdk/drafting-engine/rag-query'
|
import { queryRAG } from '@/lib/sdk/drafting-engine/rag-query'
|
||||||
|
|
||||||
// Sucht via klausur-service DSFA-RAG
|
// Sucht via ai-compliance-sdk RAG (Qdrant)
|
||||||
const ragContext = await queryRAG('DSFA Art. 35 DSGVO', 3)
|
const ragContext = await queryRAG('DSFA Art. 35 DSGVO', 3)
|
||||||
// → "[Quelle 1: DSGVO]\nArt. 35 regelt die DSFA..."
|
// → "[Quelle 1: DSGVO]\nArt. 35 regelt die DSFA..."
|
||||||
```
|
```
|
||||||
|
|||||||
Reference in New Issue
Block a user