Files
breakpilot-compliance/admin-compliance/app/sdk/control-library/page.tsx
T
Benjamin Admin 372e1fe9e9
CI / detect-changes (push) Successful in 14s
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) Failing after 7s
CI / validate-canonical-controls (push) Successful in 13s
CI / loc-budget (push) Failing after 15s
CI / go-lint (push) Has been skipped
CI / test-go (push) Has been skipped
CI / python-lint (push) Has been skipped
CI / nodejs-lint (push) Has been skipped
CI / nodejs-build (push) Successful in 2m23s
CI / iace-gt-coverage (push) Has been skipped
CI / test-python-backend (push) Successful in 34s
CI / test-python-document-crawler (push) Has been skipped
CI / test-python-dsms-gateway (push) Has been skipped
Use-Case-Mapping-Filter für Master Controls + Mapper-Präzisionsfix
Phase 2: Live-Filter an /sdk/master-controls (Use Case, Quell-Regulierung,
Verifikations-Methode, Coverage, Primärzweck-Toggle, category via Member-EXISTS).
API mit EXISTS-Filtern + gecachten Meta-Counts in master-controls/route.ts.

Phase A: neue UseCase telekommunikation + Fix der Impressum-Fehlrouten im
Register (TKG/AT-TKG->telekommunikation, telemedien->dse, GewO->handelsrecht);
echte Impressum-Quellen (TMG/Mediengesetz) bleiben impressum. Deterministischer
Seed aus source_regulation; Tests grün.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-09 23:19:56 +02:00

274 lines
11 KiB
TypeScript

'use client'
import { useEffect } from 'react'
import { useSearchParams } from 'next/navigation'
import { EMPTY_CONTROL } from './components/helpers'
import { ControlForm } from './components/ControlForm'
import { ControlDetail } from './components/ControlDetail'
import { ReviewCompare } from './components/ReviewCompare'
import { V1CompareView } from './components/V1CompareView'
import { ControlListView } from './components/ControlListView'
import { useControlLibraryState } from './components/useControlLibraryState'
import { createCRUDHandlers } from './components/useControlCRUD'
import { BACKEND_URL } from './components/helpers'
export default function ControlLibraryPage() {
const state = useControlLibraryState()
const searchParams = useSearchParams()
// Deep-link via /sdk/control-library?control=<id>
// — e.g. from /sdk/master-controls member list.
useEffect(() => {
const cid = searchParams?.get('control')
if (!cid || state.selectedControl?.control_id === cid) return
fetch(`${BACKEND_URL}?endpoint=control&id=${encodeURIComponent(cid)}`)
.then(r => r.ok ? r.json() : null)
.then(ctrl => {
if (ctrl?.control_id) {
state.setSelectedControl(ctrl)
state.setMode('detail')
}
})
.catch(() => { /* user just sees the list */ })
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [searchParams])
const {
handleCreate, handleUpdate, handleDelete, handleReview, handleBulkReject,
} = createCRUDHandlers({
selectedControl: state.selectedControl,
fullReload: state.fullReload,
reviewMode: state.reviewMode,
reviewIndex: state.reviewIndex,
reviewItems: state.reviewItems,
setMode: state.setMode,
setSelectedControl: state.setSelectedControl,
setReviewMode: state.setReviewMode,
setReviewItems: state.setReviewItems,
setReviewIndex: state.setReviewIndex,
setSaving: state.setSaving,
setBulkProcessing: state.setBulkProcessing,
reviewCount: state.reviewCount,
totalCount: state.totalCount,
stateFilter: state.stateFilter,
})
// Loading / error screens
if (state.loading && state.controls.length === 0) {
return (
<div className="flex items-center justify-center h-96">
<div className="animate-spin rounded-full h-8 w-8 border-2 border-purple-600 border-t-transparent" />
</div>
)
}
if (state.error) {
return (
<div className="flex items-center justify-center h-96">
<p className="text-red-600">{state.error}</p>
</div>
)
}
// CREATE mode
if (state.mode === 'create') {
return <ControlForm initial={EMPTY_CONTROL} onSave={handleCreate} onCancel={() => state.setMode('list')} saving={state.saving} />
}
// EDIT mode
if (state.mode === 'edit' && state.selectedControl) {
return (
<ControlForm
initial={{
...EMPTY_CONTROL,
...state.selectedControl,
scope: {
platforms: state.selectedControl.scope?.platforms ?? [],
components: state.selectedControl.scope?.components ?? [],
data_classes: state.selectedControl.scope?.data_classes ?? [],
},
target_audience: Array.isArray(state.selectedControl.target_audience)
? state.selectedControl.target_audience.join(', ')
: state.selectedControl.target_audience,
risk_score: state.selectedControl.risk_score,
implementation_effort: state.selectedControl.implementation_effort,
open_anchors: state.selectedControl.open_anchors.length > 0
? state.selectedControl.open_anchors
: [{ framework: '', ref: '', url: '' }],
requirements: state.selectedControl.requirements.length > 0 ? state.selectedControl.requirements : [''],
test_procedure: state.selectedControl.test_procedure.length > 0 ? state.selectedControl.test_procedure : [''],
evidence: state.selectedControl.evidence.length > 0
? state.selectedControl.evidence.map(e => typeof e === 'string' ? { type: '', description: e } : e)
: [{ type: '', description: '' }],
}}
onSave={handleUpdate}
onCancel={() => state.setMode('detail')}
saving={state.saving}
/>
)
}
// V1 COMPARE mode
if (state.compareMode && state.compareV1Control) {
return (
<V1CompareView
v1Control={state.compareV1Control}
matches={state.compareMatches}
onBack={() => state.setCompareMode(false)}
onNavigateToControl={async (controlId: string) => {
try {
const res = await fetch(`${BACKEND_URL}?endpoint=control&id=${controlId}`)
if (res.ok) { state.setCompareMode(false); state.setSelectedControl(await res.json()); state.setMode('detail') }
} catch { /* ignore */ }
}}
/>
)
}
// DETAIL mode
if (state.mode === 'detail' && state.selectedControl) {
const isDuplicateReview = state.reviewMode && state.reviewTab === 'duplicates'
const reviewTabBar = state.reviewMode ? (
<div className="border-b border-gray-200 bg-white px-6 py-2 flex items-center gap-4">
<button
onClick={() => state.switchReviewTab('duplicates')}
className={`px-3 py-1.5 text-sm rounded-lg font-medium ${state.reviewTab === 'duplicates' ? 'bg-amber-100 text-amber-800 border border-amber-300' : 'text-gray-500 hover:text-gray-700 hover:bg-gray-100'}`}
>
Duplikat-Verdacht ({state.reviewDuplicates.length})
</button>
<button
onClick={() => state.switchReviewTab('rule3')}
className={`px-3 py-1.5 text-sm rounded-lg font-medium ${state.reviewTab === 'rule3' ? 'bg-purple-100 text-purple-800 border border-purple-300' : 'text-gray-500 hover:text-gray-700 hover:bg-gray-100'}`}
>
Rule 3 ohne Anchor ({state.reviewRule3.length})
</button>
</div>
) : null
const reviewNavProps = {
reviewMode: state.reviewMode,
reviewIndex: state.reviewIndex,
reviewTotal: state.reviewItems.length,
onReviewPrev: () => {
const idx = Math.max(0, state.reviewIndex - 1)
state.setReviewIndex(idx)
state.setSelectedControl(state.reviewItems[idx])
},
onReviewNext: () => {
const idx = Math.min(state.reviewItems.length - 1, state.reviewIndex + 1)
state.setReviewIndex(idx)
state.setSelectedControl(state.reviewItems[idx])
},
}
if (isDuplicateReview) {
return (
<div className="flex flex-col h-full">
{reviewTabBar}
<div className="flex-1 overflow-hidden">
<ReviewCompare
ctrl={state.selectedControl}
onBack={() => { state.setMode('list'); state.setSelectedControl(null); state.setReviewMode(false) }}
onReview={handleReview}
onEdit={() => state.setMode('edit')}
reviewIndex={state.reviewIndex}
reviewTotal={state.reviewItems.length}
onReviewPrev={reviewNavProps.onReviewPrev}
onReviewNext={reviewNavProps.onReviewNext}
/>
</div>
</div>
)
}
return (
<div className="flex flex-col h-full">
{reviewTabBar}
<div className="flex-1 overflow-hidden">
<ControlDetail
ctrl={state.selectedControl}
onBack={() => { state.setMode('list'); state.setSelectedControl(null); state.setReviewMode(false) }}
onEdit={() => state.setMode('edit')}
onDelete={handleDelete}
onReview={handleReview}
onRefresh={state.fullReload}
onCompare={(ctrl, matches) => { state.setCompareV1Control(ctrl); state.setCompareMatches(matches); state.setCompareMode(true) }}
onNavigateToControl={async (controlId: string) => {
try {
const res = await fetch(`${BACKEND_URL}?endpoint=control&id=${controlId}`)
if (res.ok) { state.setSelectedControl(await res.json()); state.setMode('detail') }
} catch { /* ignore */ }
}}
reviewMode={state.reviewMode}
reviewIndex={state.reviewIndex}
reviewTotal={state.reviewItems.length}
onReviewPrev={reviewNavProps.onReviewPrev}
onReviewNext={reviewNavProps.onReviewNext}
/>
</div>
</div>
)
}
// LIST mode
return (
<ControlListView
frameworks={state.frameworks}
controls={state.controls}
totalCount={state.totalCount}
meta={state.meta}
loading={state.loading}
reviewCount={state.reviewCount}
bulkProcessing={state.bulkProcessing}
showStats={state.showStats}
processedStats={state.processedStats}
showGenerator={state.showGenerator}
currentPage={state.currentPage}
totalPages={state.totalPages}
sortBy={state.sortBy}
searchQuery={state.searchQuery}
severityFilter={state.severityFilter}
domainFilter={state.domainFilter}
stateFilter={state.stateFilter}
verificationFilter={state.verificationFilter}
useCaseFilter={state.useCaseFilter}
primaryOnly={state.primaryOnly}
regulationFilter={state.regulationFilter}
mappedFilter={state.mappedFilter}
categoryFilter={state.categoryFilter}
evidenceTypeFilter={state.evidenceTypeFilter}
audienceFilter={state.audienceFilter}
sourceFilter={state.sourceFilter}
typeFilter={state.typeFilter}
hideDuplicates={state.hideDuplicates}
setSearchQuery={state.setSearchQuery}
setSeverityFilter={state.setSeverityFilter}
setDomainFilter={state.setDomainFilter}
setStateFilter={state.setStateFilter}
setVerificationFilter={state.setVerificationFilter}
setUseCaseFilter={state.setUseCaseFilter}
setPrimaryOnly={state.setPrimaryOnly}
setRegulationFilter={state.setRegulationFilter}
setMappedFilter={state.setMappedFilter}
setCategoryFilter={state.setCategoryFilter}
setEvidenceTypeFilter={state.setEvidenceTypeFilter}
setAudienceFilter={state.setAudienceFilter}
setSourceFilter={state.setSourceFilter}
setTypeFilter={state.setTypeFilter}
setHideDuplicates={state.setHideDuplicates}
setSortBy={state.setSortBy}
setShowStats={state.setShowStats}
setShowGenerator={state.setShowGenerator}
setCurrentPage={state.setCurrentPage}
onSelectControl={(ctrl) => { state.setSelectedControl(ctrl); state.setMode('detail') }}
onCreateMode={() => state.setMode('create')}
onEnterReview={state.enterReviewMode}
onBulkReject={handleBulkReject}
onRefresh={() => { state.loadControls(); state.loadMeta(); state.loadFrameworks(); state.loadReviewCount() }}
onLoadStats={state.loadProcessedStats}
onFullReload={state.fullReload}
/>
)
}