principles-disciple 1.72.0 → 1.74.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/INSTALL.md +1 -3
- package/openclaw.plugin.json +10 -5
- package/package.json +17 -19
- package/scripts/acceptance-test.mjs +16 -73
- package/scripts/sync-plugin.mjs +382 -77
- package/src/commands/archive-impl.ts +2 -1
- package/src/commands/capabilities.ts +2 -2
- package/src/commands/context.ts +2 -2
- package/src/commands/disable-impl.ts +2 -1
- package/src/commands/evolution-status.ts +16 -16
- package/src/commands/export.ts +12 -67
- package/src/commands/pain.ts +91 -1
- package/src/commands/principle-rollback.ts +2 -1
- package/src/commands/promote-impl.ts +7 -43
- package/src/commands/rollback-impl.ts +2 -1
- package/src/commands/rollback.ts +2 -1
- package/src/commands/samples.ts +2 -1
- package/src/commands/thinking-os.ts +2 -1
- package/src/config/errors.ts +18 -2
- package/src/constants/diagnostician.ts +2 -2
- package/src/constants/tools.ts +2 -1
- package/src/core/__tests__/focus-history.test.ts +210 -0
- package/src/core/config.ts +1 -1
- package/src/core/correction-cue-learner.ts +2 -136
- package/src/core/correction-types.ts +16 -88
- package/src/core/dictionary.ts +19 -20
- package/src/core/empathy-keyword-matcher.ts +17 -289
- package/src/core/empathy-types.ts +18 -229
- package/src/core/event-log.ts +29 -132
- package/src/core/evolution-reducer.ts +21 -2
- package/src/core/evolution-types.ts +76 -464
- package/src/core/file-store.ts +80 -0
- package/src/core/focus-history.ts +228 -955
- package/src/core/local-worker-routing.ts +34 -314
- package/src/core/merge-gate-audit.ts +0 -195
- package/src/core/migration.ts +0 -1
- package/src/core/pain-diagnostic-gate.ts +154 -0
- package/src/core/pain-signal.ts +21 -138
- package/src/core/pain.ts +15 -88
- package/src/core/path-resolver.ts +0 -1
- package/src/core/paths.ts +0 -1
- package/src/core/pd-task-reconciler.ts +26 -115
- package/src/core/pd-task-service.ts +9 -9
- package/src/core/pd-task-types.ts +23 -127
- package/src/core/principle-compiler/__tests__/compiler-replay-gate.test.ts +174 -0
- package/src/core/principle-compiler/code-validator.ts +15 -42
- package/src/core/principle-compiler/compiler.ts +100 -15
- package/src/core/principle-compiler/index.ts +5 -2
- package/src/core/principle-compiler/template-generator.ts +4 -104
- package/src/core/principle-injection.ts +10 -202
- package/src/core/principle-internalization/filesystem-lifecycle-datasource.ts +42 -0
- package/src/core/principle-internalization/lifecycle-read-model.ts +39 -242
- package/src/core/principle-internalization/principle-lifecycle-service.ts +12 -10
- package/src/core/principle-tree-ledger-adapter.ts +145 -0
- package/src/core/principle-tree-ledger.ts +8 -6
- package/src/core/reflection/reflection-context.ts +14 -109
- package/src/core/replay-engine.ts +8 -500
- package/src/core/rule-host-helpers.ts +5 -35
- package/src/core/rule-host-types.ts +10 -82
- package/src/core/rule-host.ts +6 -63
- package/src/core/runtime-v2-prompt-activation-reader.ts +231 -0
- package/src/core/session-tracker.ts +87 -101
- package/src/core/shadow-observation-registry.ts +19 -48
- package/src/core/trajectory.ts +3 -1
- package/src/core/workflow-funnel-loader.ts +62 -68
- package/src/core/workspace-context.ts +46 -0
- package/src/core/workspace-dir-service.ts +1 -1
- package/src/core/workspace-dir-validation.ts +18 -9
- package/src/hooks/AGENTS.md +1 -1
- package/src/hooks/gate-block-helper.ts +71 -64
- package/src/hooks/gate.ts +183 -31
- package/src/hooks/lifecycle.ts +30 -32
- package/src/hooks/llm.ts +60 -32
- package/src/hooks/pain.ts +297 -103
- package/src/hooks/prompt.ts +400 -440
- package/src/hooks/subagent.ts +2 -29
- package/src/i18n/commands.ts +2 -10
- package/src/index.ts +95 -85
- package/src/openclaw-sdk.ts +311 -0
- package/src/service/central-database.ts +8 -4
- package/src/service/evolution-queue-migration.ts +2 -1
- package/src/service/evolution-worker.ts +163 -1786
- package/src/service/internalization-trigger-adapter.ts +302 -0
- package/src/service/keyword-optimization-service.ts +4 -4
- package/src/service/monitoring-query-service.ts +1 -215
- package/src/service/queue-io.ts +60 -331
- package/src/service/runtime-summary-service.ts +59 -16
- package/src/service/subagent-workflow/index.ts +0 -41
- package/src/service/subagent-workflow/types.ts +9 -120
- package/src/service/subagent-workflow/workflow-store.ts +2 -119
- package/src/service/workflow-watchdog.ts +0 -43
- package/src/types/event-payload.ts +16 -74
- package/src/types/event-types.ts +38 -547
- package/src/types/hygiene-types.ts +7 -30
- package/src/types/principle-tree-schema.ts +20 -222
- package/src/types/queue.ts +15 -70
- package/src/types/runtime-summary.ts +5 -49
- package/src/utils/io.ts +8 -20
- package/src/utils/retry.ts +1 -1
- package/src/utils/shadow-fingerprint.ts +2 -2
- package/src/utils/workspace-resolver.ts +50 -0
- package/templates/langs/en/core/AGENTS.md +7 -7
- package/templates/langs/en/core/BOOT.md +1 -1
- package/templates/langs/en/core/HEARTBEAT.md +2 -2
- package/templates/langs/en/principles/THINKING_OS.md +3 -2
- package/templates/langs/en/skills/ai-sprint-orchestration/references/agent-registry.json +1 -72
- package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/bugfix-complex-template.json +6 -6
- package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/feature-complex-template.json +6 -6
- package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal-verify.json +2 -12
- package/templates/langs/en/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal.json +2 -12
- package/templates/langs/en/skills/ai-sprint-orchestration/scripts/run.mjs +51 -15
- package/templates/langs/en/skills/evolve-task/SKILL.md +3 -3
- package/templates/langs/en/skills/pd-cli-operator/SKILL.md +67 -0
- package/templates/langs/en/skills/pd-diagnostician/SKILL.md +1 -1
- package/templates/langs/en/skills/pd-mentor/SKILL.md +2 -3
- package/templates/langs/en/skills/pd-pain-signal/SKILL.md +17 -39
- package/templates/langs/en/skills/pd-runtime-v2/SKILL.md +61 -0
- package/templates/langs/zh/core/AGENTS.md +7 -7
- package/templates/langs/zh/core/BOOT.md +1 -1
- package/templates/langs/zh/core/HEARTBEAT.md +2 -2
- package/templates/langs/zh/principles/THINKING_OS.md +3 -2
- package/templates/langs/zh/skills/ai-sprint-orchestration/references/agent-registry.json +1 -72
- package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/bugfix-complex-template.json +6 -6
- package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/feature-complex-template.json +6 -6
- package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/nocturnal-trinity-quality-enhancement.json +8 -8
- package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal-verify.json +2 -12
- package/templates/langs/zh/skills/ai-sprint-orchestration/references/specs/workflow-validation-minimal.json +2 -12
- package/templates/langs/zh/skills/ai-sprint-orchestration/scripts/run.mjs +51 -15
- package/templates/langs/zh/skills/ai-sprint-orchestration/test/run.test.mjs +21 -5
- package/templates/langs/zh/skills/evolve-task/SKILL.md +4 -4
- package/templates/langs/zh/skills/pd-cli-operator/SKILL.md +67 -0
- package/templates/langs/zh/skills/pd-diagnostician/SKILL.md +1 -1
- package/templates/langs/zh/skills/pd-mentor/SKILL.md +2 -3
- package/templates/langs/zh/skills/pd-pain-signal/SKILL.md +17 -38
- package/templates/langs/zh/skills/pd-runtime-v2/SKILL.md +61 -0
- package/tests/build-artifacts.test.ts +1 -3
- package/tests/commands/evolution-status.test.ts +0 -118
- package/tests/core/bootstrap-rules.test.ts +1 -1
- package/tests/core/config.test.ts +1 -1
- package/tests/core/event-log.test.ts +35 -0
- package/tests/core/evolution-engine.test.ts +610 -0
- package/tests/core/file-store.test.ts +102 -0
- package/tests/core/focus-history.test.ts +203 -11
- package/tests/core/merge-gate-audit.test.ts +2 -169
- package/tests/core/migration.test.ts +7 -7
- package/tests/core/model-deployment-registry.test.ts +7 -1
- package/tests/core/model-training-registry.test.ts +19 -0
- package/tests/core/observability.test.ts +0 -1
- package/tests/core/pain-diagnostic-gate.test.ts +498 -0
- package/tests/core/pain.test.ts +0 -1
- package/tests/core/path-resolver.test.ts +1 -1
- package/tests/core/paths-refactor.test.ts +0 -22
- package/tests/core/principle-internalization/deprecated-readiness.test.ts +2 -2
- package/tests/core/principle-internalization/lifecycle-metrics.test.ts +2 -2
- package/tests/core/principle-internalization/{internalization-routing-policy.test.ts → lifecycle-routing-policy.test.ts} +6 -6
- package/tests/core/principle-internalization/lineage-source-retired.test.ts +56 -0
- package/tests/core/principle-internalization/principle-lifecycle-service.test.ts +1 -23
- package/tests/core/principle-tree-ledger-adapter.test.ts +253 -0
- package/tests/core/reflection-context.test.ts +0 -14
- package/tests/core/replay-engine.test.ts +127 -215
- package/tests/core/rule-host-helpers.test.ts +2 -2
- package/tests/core/rule-implementation-runtime.test.ts +0 -27
- package/tests/core/workflow-funnel-loader.test.ts +162 -0
- package/tests/core/workspace-context.test.ts +2 -2
- package/tests/core/workspace-dir-validation.test.ts +8 -1
- package/tests/core-anti-growth.test.ts +191 -0
- package/tests/hook-workspace-nextaction-contract.test.ts +42 -0
- package/tests/hooks/confirm-first-removal.test.ts +188 -0
- package/tests/hooks/gate-auto-correct-shadow.test.ts +310 -0
- package/tests/hooks/gate-auto-correct.test.ts +665 -0
- package/tests/hooks/gate-no-path-write-tool.test.ts +172 -0
- package/tests/hooks/gate-rule-host-pipeline.test.ts +2 -1
- package/tests/hooks/pain.test.ts +269 -12
- package/tests/hooks/prompt-characterization.test.ts +500 -0
- package/tests/hooks/prompt-size-guard.test.ts +32 -17
- package/tests/hooks/runtime-v2-prompt-activation.test.ts +869 -0
- package/tests/index.test.ts +94 -1
- package/tests/integration/auto-entry-gate.test.ts +248 -0
- package/tests/integration/internalization-trigger-guard.test.ts +69 -0
- package/tests/integration/m8-legacy-paths.test.ts +63 -0
- package/tests/integration/runtime-v2-pain-guard.test.ts +125 -0
- package/tests/plugin-config-resolution-cutover.test.ts +359 -0
- package/tests/runtime-v2-discovery-guard.test.ts +154 -0
- package/tests/service/central-database.test.ts +457 -0
- package/tests/service/evolution-worker.correction-observer.test.ts +173 -0
- package/tests/service/evolution-worker.timeout.test.ts +11 -129
- package/tests/service/internalization-trigger-adapter.test.ts +251 -0
- package/tests/service/monitoring-query-service.test.ts +1 -47
- package/tests/service/queue-io.test.ts +1 -62
- package/tests/service/runtime-summary-service.test.ts +3 -1
- package/tests/service/workflow-watchdog.test.ts +0 -91
- package/tests/utils/file-lock.test.ts +5 -3
- package/tests/utils/session-key.test.ts +52 -0
- package/tests/utils/subagent-probe.test.ts +48 -1
- package/vitest.config.ts +4 -11
- package/.planning/codebase/ARCHITECTURE.md +0 -157
- package/.planning/codebase/CONCERNS.md +0 -145
- package/.planning/codebase/CONVENTIONS.md +0 -148
- package/.planning/codebase/INTEGRATIONS.md +0 -81
- package/.planning/codebase/STACK.md +0 -87
- package/.planning/codebase/STRUCTURE.md +0 -193
- package/.planning/codebase/TESTING.md +0 -243
- package/.planning/phases/01-basic-visualization/01-GAP-CLOSURE-VERIFICATION.md +0 -113
- package/docs/COMMAND_REFERENCE.md +0 -76
- package/docs/COMMAND_REFERENCE_EN.md +0 -79
- package/scripts/build-web.mjs +0 -46
- package/scripts/diagnose-nocturnal.mjs +0 -537
- package/scripts/seed-nocturnal-scenarios.mjs +0 -384
- package/src/commands/nocturnal-review.ts +0 -322
- package/src/commands/nocturnal-rollout.ts +0 -790
- package/src/commands/nocturnal-train.ts +0 -986
- package/src/commands/pd-reflect.ts +0 -88
- package/src/core/adaptive-thresholds.ts +0 -478
- package/src/core/diagnostician-task-store.ts +0 -192
- package/src/core/nocturnal-arbiter.ts +0 -715
- package/src/core/nocturnal-artifact-lineage.ts +0 -116
- package/src/core/nocturnal-artificer.ts +0 -257
- package/src/core/nocturnal-candidate-scoring.ts +0 -530
- package/src/core/nocturnal-compliance.ts +0 -1146
- package/src/core/nocturnal-dataset.ts +0 -763
- package/src/core/nocturnal-executability.ts +0 -428
- package/src/core/nocturnal-export.ts +0 -499
- package/src/core/nocturnal-paths.ts +0 -240
- package/src/core/nocturnal-reasoning-deriver.ts +0 -343
- package/src/core/nocturnal-rule-implementation-validator.ts +0 -246
- package/src/core/nocturnal-snapshot-contract.ts +0 -99
- package/src/core/nocturnal-trajectory-extractor.ts +0 -512
- package/src/core/nocturnal-trinity-types.ts +0 -218
- package/src/core/nocturnal-trinity.ts +0 -2680
- package/src/core/principle-internalization/deprecated-readiness.ts +0 -93
- package/src/core/principle-internalization/internalization-routing-policy.ts +0 -208
- package/src/core/principle-internalization/lifecycle-metrics.ts +0 -152
- package/src/http/principles-console-route.ts +0 -709
- package/src/service/central-health-service.ts +0 -49
- package/src/service/central-overview-service.ts +0 -138
- package/src/service/control-ui-query-service.ts +0 -900
- package/src/service/cooldown-strategy.ts +0 -97
- package/src/service/evolution-pain-context.ts +0 -79
- package/src/service/evolution-query-service.ts +0 -407
- package/src/service/health-query-service.ts +0 -1038
- package/src/service/nocturnal-config.ts +0 -214
- package/src/service/nocturnal-runtime.ts +0 -734
- package/src/service/nocturnal-service.ts +0 -1605
- package/src/service/nocturnal-target-selector.ts +0 -545
- package/src/service/sleep-cycle.ts +0 -157
- package/src/service/startup-reconciler.ts +0 -112
- package/src/service/subagent-workflow/correction-observer-types.ts +0 -82
- package/src/service/subagent-workflow/correction-observer-workflow-manager.ts +0 -250
- package/src/service/subagent-workflow/deep-reflect-workflow-manager.ts +0 -1
- package/src/service/subagent-workflow/dynamic-timeout.ts +0 -30
- package/src/service/subagent-workflow/empathy-observer-workflow-manager.ts +0 -268
- package/src/service/subagent-workflow/nocturnal-workflow-manager.ts +0 -795
- package/src/service/subagent-workflow/runtime-direct-driver.ts +0 -268
- package/src/service/subagent-workflow/workflow-manager-base.ts +0 -580
- package/src/tools/write-pain-flag.ts +0 -215
- package/templates/langs/en/skills/plan-script/SKILL.md +0 -32
- package/templates/langs/zh/skills/plan-script/SKILL.md +0 -32
- package/tests/commands/nocturnal-review.test.ts +0 -448
- package/tests/commands/nocturnal-train.test.ts +0 -97
- package/tests/commands/pd-reflect.test.ts +0 -49
- package/tests/core/adaptive-thresholds.test.ts +0 -261
- package/tests/core/nocturnal-arbiter.test.ts +0 -559
- package/tests/core/nocturnal-artifact-lineage.test.ts +0 -53
- package/tests/core/nocturnal-artificer.test.ts +0 -241
- package/tests/core/nocturnal-candidate-scoring.test.ts +0 -532
- package/tests/core/nocturnal-compliance-p-principles.test.ts +0 -133
- package/tests/core/nocturnal-compliance.test.ts +0 -646
- package/tests/core/nocturnal-dataset.test.ts +0 -892
- package/tests/core/nocturnal-e2e.test.ts +0 -234
- package/tests/core/nocturnal-executability.test.ts +0 -357
- package/tests/core/nocturnal-export.test.ts +0 -517
- package/tests/core/nocturnal-reasoning-deriver.test.ts +0 -372
- package/tests/core/nocturnal-reviewed-subset-comparison.test.ts +0 -428
- package/tests/core/nocturnal-rule-implementation-validator.test.ts +0 -127
- package/tests/core/nocturnal-snapshot-contract.test.ts +0 -121
- package/tests/core/nocturnal-trajectory-extractor.test.ts +0 -634
- package/tests/core/nocturnal-trinity.test.ts +0 -2053
- package/tests/core/pain-auto-repair.test.ts +0 -96
- package/tests/core/pain-integration.test.ts +0 -510
- package/tests/fixtures/nocturnal-reviewed-subset.json +0 -183
- package/tests/http/principles-console-route.test.ts +0 -162
- package/tests/integration/chaos-resilience.test.ts +0 -348
- package/tests/integration/empathy-workflow-integration.test.ts +0 -626
- package/tests/integration/pain-diagnostician-loop.e2e.test.ts +0 -380
- package/tests/service/control-ui-query-service.test.ts +0 -121
- package/tests/service/cooldown-strategy.test.ts +0 -164
- package/tests/service/data-endpoints-regression.test.ts +0 -834
- package/tests/service/empathy-observer-workflow-manager.test.ts +0 -175
- package/tests/service/evolution-worker.nocturnal.test.ts +0 -601
- package/tests/service/nocturnal-runtime-hardening.test.ts +0 -118
- package/tests/service/nocturnal-runtime.test.ts +0 -473
- package/tests/service/nocturnal-service-code-candidate.test.ts +0 -330
- package/tests/service/nocturnal-target-selector.test.ts +0 -615
- package/tests/service/startup-reconciler.test.ts +0 -148
- package/tests/tools/write-pain-flag.test.ts +0 -358
- package/ui/src/App.tsx +0 -45
- package/ui/src/api.ts +0 -220
- package/ui/src/charts.tsx +0 -955
- package/ui/src/components/ErrorState.tsx +0 -6
- package/ui/src/components/Loading.tsx +0 -13
- package/ui/src/components/ProtectedRoute.tsx +0 -12
- package/ui/src/components/Shell.tsx +0 -91
- package/ui/src/components/WorkspaceConfig.tsx +0 -178
- package/ui/src/components/index.ts +0 -5
- package/ui/src/context/auth.tsx +0 -80
- package/ui/src/context/theme.tsx +0 -66
- package/ui/src/hooks/useAutoRefresh.ts +0 -39
- package/ui/src/i18n/ui.ts +0 -473
- package/ui/src/main.tsx +0 -16
- package/ui/src/pages/EvolutionPage.tsx +0 -333
- package/ui/src/pages/FeedbackPage.tsx +0 -138
- package/ui/src/pages/GateMonitorPage.tsx +0 -136
- package/ui/src/pages/LoginPage.tsx +0 -89
- package/ui/src/pages/OverviewPage.tsx +0 -599
- package/ui/src/pages/SamplesPage.tsx +0 -174
- package/ui/src/pages/ThinkingModelsPage.tsx +0 -702
- package/ui/src/styles.css +0 -2020
- package/ui/src/types.ts +0 -384
- package/ui/src/utils/format.ts +0 -15
|
@@ -1,333 +0,0 @@
|
|
|
1
|
-
import React, { useEffect, useMemo, useState } from 'react';
|
|
2
|
-
import { ChevronLeft, Clock, Activity, Shield, Zap, BookOpen } from 'lucide-react';
|
|
3
|
-
import { api } from '../api';
|
|
4
|
-
import type {
|
|
5
|
-
EvolutionTasksResponse,
|
|
6
|
-
EvolutionTraceResponse,
|
|
7
|
-
EvolutionStatsResponse,
|
|
8
|
-
EvolutionPrinciplesResponse,
|
|
9
|
-
} from '../types';
|
|
10
|
-
import { DonutChart, GroupedBarChart, TimeRangeSelector, StatusBadge, EmptyState } from '../charts';
|
|
11
|
-
import { useI18n } from '../i18n/ui';
|
|
12
|
-
import { formatPercent, formatDate, formatDuration } from '../utils/format';
|
|
13
|
-
import { Loading, ErrorState } from '../components';
|
|
14
|
-
|
|
15
|
-
const STAGE_COLORS: Record<string, string> = {
|
|
16
|
-
pain_detected: 'var(--error)',
|
|
17
|
-
queued: 'var(--warning)',
|
|
18
|
-
started: 'var(--info)',
|
|
19
|
-
analyzing: '#8b5cf6',
|
|
20
|
-
principle_generated: 'var(--success)',
|
|
21
|
-
completed: 'var(--success)',
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
const STAGE_LABEL_KEYS: string[] = [
|
|
25
|
-
'pain_detected',
|
|
26
|
-
'queued',
|
|
27
|
-
'started',
|
|
28
|
-
'analyzing',
|
|
29
|
-
'principle_generated',
|
|
30
|
-
'completed',
|
|
31
|
-
];
|
|
32
|
-
|
|
33
|
-
export function EvolutionPage() {
|
|
34
|
-
const { t } = useI18n();
|
|
35
|
-
const [tasks, setTasks] = useState<EvolutionTasksResponse | null>(null);
|
|
36
|
-
const [stats, setStats] = useState<EvolutionStatsResponse | null>(null);
|
|
37
|
-
const [trace, setTrace] = useState<EvolutionTraceResponse | null>(null);
|
|
38
|
-
const [evoPrinciples, setEvoPrinciples] = useState<EvolutionPrinciplesResponse | null>(null);
|
|
39
|
-
const [selectedId, setSelectedId] = useState('');
|
|
40
|
-
const [statusFilter, setStatusFilter] = useState('all');
|
|
41
|
-
const [error, setError] = useState('');
|
|
42
|
-
const [days, setDays] = useState(30);
|
|
43
|
-
|
|
44
|
-
const search = useMemo(() => {
|
|
45
|
-
const next = new URLSearchParams();
|
|
46
|
-
if (statusFilter !== 'all') next.set('status', statusFilter);
|
|
47
|
-
next.set('page', '1');
|
|
48
|
-
next.set('pageSize', '20');
|
|
49
|
-
return next;
|
|
50
|
-
}, [statusFilter]);
|
|
51
|
-
|
|
52
|
-
useEffect(() => {
|
|
53
|
-
Promise.all([
|
|
54
|
-
api.getEvolutionTasks(search),
|
|
55
|
-
api.getEvolutionStats(days),
|
|
56
|
-
api.getEvolutionPrinciples(),
|
|
57
|
-
]).then(([tasksData, statsData, principlesData]) => {
|
|
58
|
-
setTasks(tasksData);
|
|
59
|
-
setStats(statsData);
|
|
60
|
-
setEvoPrinciples(principlesData);
|
|
61
|
-
setError('');
|
|
62
|
-
}).catch((err) => setError(String(err)));
|
|
63
|
-
}, [search, days]);
|
|
64
|
-
|
|
65
|
-
useEffect(() => {
|
|
66
|
-
if (!selectedId) {
|
|
67
|
-
setTrace(null);
|
|
68
|
-
return;
|
|
69
|
-
}
|
|
70
|
-
api.getEvolutionTrace(selectedId).then(setTrace).catch((err) => setError(String(err)));
|
|
71
|
-
}, [selectedId]);
|
|
72
|
-
|
|
73
|
-
if (error) return <ErrorState error={error} />;
|
|
74
|
-
if (!tasks || !stats) return <Loading />;
|
|
75
|
-
|
|
76
|
-
// Prepare donut chart data
|
|
77
|
-
const statusSegments = [
|
|
78
|
-
{ label: t('evolution.pending'), value: stats.pending, color: 'var(--warning)' },
|
|
79
|
-
{ label: t('evolution.inProgress'), value: stats.inProgress, color: 'var(--info)' },
|
|
80
|
-
{ label: t('evolution.completed'), value: stats.completed, color: 'var(--success)' },
|
|
81
|
-
{ label: t('evolution.failed'), value: stats.failed, color: 'var(--error)' },
|
|
82
|
-
].filter(s => s.value > 0);
|
|
83
|
-
|
|
84
|
-
return (
|
|
85
|
-
<div className="page">
|
|
86
|
-
<header className="page-header">
|
|
87
|
-
<div>
|
|
88
|
-
<h2>{t('evolution.pageTitle')}</h2>
|
|
89
|
-
</div>
|
|
90
|
-
<div className="meta">
|
|
91
|
-
<TimeRangeSelector value={days} onChange={setDays} />
|
|
92
|
-
<div className="pill-row">
|
|
93
|
-
<StatusBadge variant="warning">{t('evolution.pending')} {stats.pending}</StatusBadge>
|
|
94
|
-
<StatusBadge variant="info">{t('evolution.inProgress')} {stats.inProgress}</StatusBadge>
|
|
95
|
-
<StatusBadge variant="success">{t('evolution.completed')} {stats.completed}</StatusBadge>
|
|
96
|
-
<StatusBadge variant="error">{t('evolution.failed')} {stats.failed}</StatusBadge>
|
|
97
|
-
</div>
|
|
98
|
-
</div>
|
|
99
|
-
</header>
|
|
100
|
-
|
|
101
|
-
{/* Current Stage Indicator */}
|
|
102
|
-
{evoPrinciples && (
|
|
103
|
-
<section className="panel" style={{ marginBottom: 'var(--space-5)' }}>
|
|
104
|
-
<h3>{t('evolution.currentStage')}</h3>
|
|
105
|
-
<div style={{ display: 'flex', alignItems: 'center', gap: 'var(--space-3)', padding: 'var(--space-3) 0' }}>
|
|
106
|
-
<span className="stage-badge">
|
|
107
|
-
{evoPrinciples.activeStage === 'pending' ? <><Clock size={16} /><span>{t('evolution.activeStage.pending')}</span></> :
|
|
108
|
-
evoPrinciples.activeStage === 'in_progress' ? <><Activity size={16} /><span>{t('evolution.activeStage.in_progress')}</span></> :
|
|
109
|
-
evoPrinciples.activeStage === 'completed' ? <><Shield size={16} /><span>{t('evolution.activeStage.completed')}</span></> :
|
|
110
|
-
evoPrinciples.activeStage === 'idle' ? <><Zap size={16} /><span>{t('evolution.activeStage.idle')}</span></> : evoPrinciples.activeStage}
|
|
111
|
-
</span>
|
|
112
|
-
<span className="text-sm" style={{ color: 'var(--text-secondary)' }}>
|
|
113
|
-
{t('evolution.enhancementLoopStatus')}
|
|
114
|
-
</span>
|
|
115
|
-
</div>
|
|
116
|
-
</section>
|
|
117
|
-
)}
|
|
118
|
-
|
|
119
|
-
{/* Principle Lifecycle & Nocturnal Training (Phase 5) */}
|
|
120
|
-
{evoPrinciples && (
|
|
121
|
-
<div className="grid two-columns" style={{ marginBottom: 'var(--space-5)' }}>
|
|
122
|
-
<section className="panel">
|
|
123
|
-
<h3><BookOpen size={16} />{t('evolution.principleLifecycle')}</h3>
|
|
124
|
-
<div className="pill-row" style={{ marginBottom: 'var(--space-3)' }}>
|
|
125
|
-
<StatusBadge variant="warning">{t('evolution.stageLabels.candidate')}: {evoPrinciples.principles.summary.candidate}</StatusBadge>
|
|
126
|
-
<StatusBadge variant="info">{t('evolution.stageLabels.probation')}: {evoPrinciples.principles.summary.probation}</StatusBadge>
|
|
127
|
-
<StatusBadge variant="success">{t('evolution.stageLabels.active')}: {evoPrinciples.principles.summary.active}</StatusBadge>
|
|
128
|
-
<StatusBadge variant="error">{t('evolution.stageLabels.deprecated')}: {evoPrinciples.principles.summary.deprecated}</StatusBadge>
|
|
129
|
-
</div>
|
|
130
|
-
{evoPrinciples.principles.recent.length > 0 && (
|
|
131
|
-
<div className="stack">
|
|
132
|
-
{evoPrinciples.principles.recent.slice(0, 5).map((item, i) => (
|
|
133
|
-
<div className="row-card" key={`${item.principleId}-${i}`}>
|
|
134
|
-
<div>
|
|
135
|
-
<strong>{item.principleId}</strong>
|
|
136
|
-
<span>{item.fromStatus} → {item.toStatus}</span>
|
|
137
|
-
</div>
|
|
138
|
-
<span className="badge">{new Date(item.timestamp).toLocaleString()}</span>
|
|
139
|
-
</div>
|
|
140
|
-
))}
|
|
141
|
-
</div>
|
|
142
|
-
)}
|
|
143
|
-
</section>
|
|
144
|
-
<section className="panel">
|
|
145
|
-
<h3>💤 {t('evolution.nocturnalTrainingStatus')}</h3>
|
|
146
|
-
<div className="stack">
|
|
147
|
-
<div className="row-card">
|
|
148
|
-
<strong>{t('evolution.trainingQueue')}</strong>
|
|
149
|
-
<span>{t('evolution.pendingShort')}: {evoPrinciples.nocturnalTraining.queue.pending} | {t('evolution.inProgressShort')}: {evoPrinciples.nocturnalTraining.queue.inProgress} | {t('evolution.completedShort')}: {evoPrinciples.nocturnalTraining.queue.completed}</span>
|
|
150
|
-
</div>
|
|
151
|
-
<div className="row-card">
|
|
152
|
-
<strong>{t('evolution.arbiterPassRate')}</strong>
|
|
153
|
-
<span>{(evoPrinciples.nocturnalTraining.arbiterPassRate * 100).toFixed(1)}%</span>
|
|
154
|
-
</div>
|
|
155
|
-
<div className="row-card">
|
|
156
|
-
<strong>{t('evolution.orpoSampleCount')}</strong>
|
|
157
|
-
<span>{evoPrinciples.nocturnalTraining.orpoSampleCount}</span>
|
|
158
|
-
</div>
|
|
159
|
-
<div className="row-card">
|
|
160
|
-
<strong>{t('evolution.modelDeployments')}</strong>
|
|
161
|
-
<span>{evoPrinciples.nocturnalTraining.deployments.length} {t('evolution.deploymentCount')}</span>
|
|
162
|
-
</div>
|
|
163
|
-
</div>
|
|
164
|
-
</section>
|
|
165
|
-
</div>
|
|
166
|
-
)}
|
|
167
|
-
|
|
168
|
-
{/* Status Distribution & Recent Activity */}
|
|
169
|
-
<div className="grid two-columns" style={{ marginBottom: 'var(--space-5)' }}>
|
|
170
|
-
<section className="panel">
|
|
171
|
-
<h3>{t('evolution.statusDistribution')}</h3>
|
|
172
|
-
<div style={{ display: 'flex', justifyContent: 'center', padding: 'var(--space-4) 0' }}>
|
|
173
|
-
<DonutChart segments={statusSegments} size={100} strokeWidth={10} />
|
|
174
|
-
</div>
|
|
175
|
-
</section>
|
|
176
|
-
<section className="panel">
|
|
177
|
-
<h3>{t('evolution.recentActivity')}</h3>
|
|
178
|
-
{stats.recentActivity && stats.recentActivity.length > 0 && (
|
|
179
|
-
<>
|
|
180
|
-
<div style={{ marginBottom: 'var(--space-3)' }}>
|
|
181
|
-
<GroupedBarChart
|
|
182
|
-
data={stats.recentActivity.slice(-14).map((item) => ({
|
|
183
|
-
label: item.day.slice(5),
|
|
184
|
-
values: [item.created, item.completed],
|
|
185
|
-
}))}
|
|
186
|
-
colors={['var(--accent)', 'var(--success)']}
|
|
187
|
-
width={280}
|
|
188
|
-
height={60}
|
|
189
|
-
/>
|
|
190
|
-
<div style={{ display: 'flex', justifyContent: 'center', gap: 'var(--space-4)', marginTop: 'var(--space-2)', fontSize: '11px', color: 'var(--text-tertiary)' }}>
|
|
191
|
-
<span><span style={{ display: 'inline-block', width: '10px', height: '10px', background: 'var(--accent)', borderRadius: '2px', marginRight: '4px' }}></span>{t('evolution.created')}</span>
|
|
192
|
-
<span><span style={{ display: 'inline-block', width: '10px', height: '10px', background: 'var(--success)', borderRadius: '2px', marginRight: '4px' }}></span>{t('evolution.finished')}</span>
|
|
193
|
-
</div>
|
|
194
|
-
</div>
|
|
195
|
-
<div className="stack">
|
|
196
|
-
{stats.recentActivity.slice(-7).reverse().map((item) => (
|
|
197
|
-
<div className="row-card" key={item.day}>
|
|
198
|
-
<strong>{item.day}</strong>
|
|
199
|
-
<span>+{item.created} {t('evolution.created')} {item.completed} {t('evolution.finished')}</span>
|
|
200
|
-
</div>
|
|
201
|
-
))}
|
|
202
|
-
</div>
|
|
203
|
-
</>
|
|
204
|
-
)}
|
|
205
|
-
</section>
|
|
206
|
-
</div>
|
|
207
|
-
|
|
208
|
-
{/* Stage Distribution */}
|
|
209
|
-
{stats.stageDistribution && stats.stageDistribution.length > 0 && (
|
|
210
|
-
<section className="panel" style={{ marginBottom: 'var(--space-5)' }}>
|
|
211
|
-
<h3>{t('evolution.stageDistribution')}</h3>
|
|
212
|
-
<div className="stack" style={{ flexDirection: 'row', flexWrap: 'wrap', gap: 'var(--space-2)' }}>
|
|
213
|
-
{stats.stageDistribution.map((stage) => (
|
|
214
|
-
<StatusBadge key={stage.stage} variant="neutral">
|
|
215
|
-
{stage.stageLabel}: {stage.count}
|
|
216
|
-
</StatusBadge>
|
|
217
|
-
))}
|
|
218
|
-
</div>
|
|
219
|
-
</section>
|
|
220
|
-
)}
|
|
221
|
-
|
|
222
|
-
<div className="grid two-columns wide-right">
|
|
223
|
-
<section className="panel">
|
|
224
|
-
<div className="filters">
|
|
225
|
-
<label>
|
|
226
|
-
{t('evolution.statusFilter')}
|
|
227
|
-
<select value={statusFilter} onChange={(e) => setStatusFilter(e.target.value)}>
|
|
228
|
-
<option value="all">{t('evolution.filterAll')}</option>
|
|
229
|
-
<option value="pending">{t('evolution.pending')}</option>
|
|
230
|
-
<option value="in_progress">{t('evolution.inProgress')}</option>
|
|
231
|
-
<option value="completed">{t('evolution.completed')}</option>
|
|
232
|
-
</select>
|
|
233
|
-
</label>
|
|
234
|
-
</div>
|
|
235
|
-
|
|
236
|
-
<div className="list-table">
|
|
237
|
-
{tasks.items.map((task) => (
|
|
238
|
-
<button
|
|
239
|
-
className={`table-row ${selectedId === task.traceId ? 'active' : ''}`}
|
|
240
|
-
key={task.taskId}
|
|
241
|
-
onClick={() => setSelectedId(task.traceId)}
|
|
242
|
-
>
|
|
243
|
-
<div>
|
|
244
|
-
<strong style={{ color: STAGE_COLORS[task.status] || '#6b7280' }}>{task.taskId}</strong>
|
|
245
|
-
<span>{task.source}</span>
|
|
246
|
-
</div>
|
|
247
|
-
<div>
|
|
248
|
-
<StatusBadge variant="neutral">{task.status}</StatusBadge>
|
|
249
|
-
<span>{t('evolution.score')}: {task.score}</span>
|
|
250
|
-
</div>
|
|
251
|
-
<div className="align-right">
|
|
252
|
-
<strong>{formatDuration(task.duration)}</strong>
|
|
253
|
-
<span>{task.eventCount} {t('evolution.events')}</span>
|
|
254
|
-
</div>
|
|
255
|
-
</button>
|
|
256
|
-
))}
|
|
257
|
-
</div>
|
|
258
|
-
|
|
259
|
-
<div className="pagination">
|
|
260
|
-
{t('common.total')} {tasks.pagination.total} {t('common.items')}
|
|
261
|
-
</div>
|
|
262
|
-
</section>
|
|
263
|
-
|
|
264
|
-
<section className="panel">
|
|
265
|
-
{!selectedId && (
|
|
266
|
-
<EmptyState title={t('evolution.emptyTitle')} description={t('evolution.emptyDesc')} />
|
|
267
|
-
)}
|
|
268
|
-
{trace && (
|
|
269
|
-
<div className="detail-stack">
|
|
270
|
-
<div className="detail-header">
|
|
271
|
-
<button
|
|
272
|
-
className="back-button"
|
|
273
|
-
onClick={() => setSelectedId('')}
|
|
274
|
-
title={t('common.back') || 'Back'}
|
|
275
|
-
>
|
|
276
|
-
<ChevronLeft strokeWidth={1.75} size={18} />
|
|
277
|
-
</button>
|
|
278
|
-
<div>
|
|
279
|
-
<h3>{t('evolution.taskLabel')} {trace.task.taskId}</h3>
|
|
280
|
-
<p>{t('evolution.source')}: {trace.task.source} | {t('evolution.score')}: {trace.task.score}</p>
|
|
281
|
-
<p style={{ fontSize: '0.85em', color: '#6b7280' }}>{trace.task.reason}</p>
|
|
282
|
-
</div>
|
|
283
|
-
<StatusBadge variant="neutral">{trace.task.status}</StatusBadge>
|
|
284
|
-
</div>
|
|
285
|
-
|
|
286
|
-
<article>
|
|
287
|
-
<h4>{t('evolution.evolutionTimeline')}</h4>
|
|
288
|
-
<div className="timeline">
|
|
289
|
-
{trace.timeline.map((item, index) => (
|
|
290
|
-
<div className="timeline-item" key={`${item.stage}-${index}`}>
|
|
291
|
-
<div
|
|
292
|
-
className="timeline-marker"
|
|
293
|
-
style={{ background: item.stageColor }}
|
|
294
|
-
/>
|
|
295
|
-
<div className="timeline-content">
|
|
296
|
-
<div className="timeline-time">{formatDate(item.timestamp)}</div>
|
|
297
|
-
<div className="timeline-stage" style={{ color: item.stageColor }}>
|
|
298
|
-
{item.stageLabel}
|
|
299
|
-
</div>
|
|
300
|
-
<div className="timeline-summary">{item.summary || item.message}</div>
|
|
301
|
-
</div>
|
|
302
|
-
</div>
|
|
303
|
-
))}
|
|
304
|
-
</div>
|
|
305
|
-
</article>
|
|
306
|
-
|
|
307
|
-
{trace.events.length > 0 && (
|
|
308
|
-
<article>
|
|
309
|
-
<h4>{t('evolution.detailedEvents')} ({trace.events.length})</h4>
|
|
310
|
-
<div className="stack">
|
|
311
|
-
{trace.events.slice(0, 10).map((event) => (
|
|
312
|
-
<div className="row-card vertical" key={event.id}>
|
|
313
|
-
<div>
|
|
314
|
-
<strong style={{ color: event.stageColor }}>{event.stageLabel}</strong>
|
|
315
|
-
<span>{formatDate(event.createdAt)}</span>
|
|
316
|
-
</div>
|
|
317
|
-
<div style={{ fontSize: '0.9em', color: '#374151' }}>
|
|
318
|
-
{event.summary || event.message}
|
|
319
|
-
</div>
|
|
320
|
-
</div>
|
|
321
|
-
))}
|
|
322
|
-
</div>
|
|
323
|
-
</article>
|
|
324
|
-
)}
|
|
325
|
-
</div>
|
|
326
|
-
)}
|
|
327
|
-
</section>
|
|
328
|
-
</div>
|
|
329
|
-
</div>
|
|
330
|
-
);
|
|
331
|
-
}
|
|
332
|
-
|
|
333
|
-
// ===== Phase 6: Feedback Loop Page =====
|
|
@@ -1,138 +0,0 @@
|
|
|
1
|
-
import React, { useCallback, useEffect, useState } from 'react';
|
|
2
|
-
import { api } from '../api';
|
|
3
|
-
import type { FeedbackGfiResponse, EmpathyEvent, FeedbackGateBlock } from '../types';
|
|
4
|
-
import { GroupedBarChart, EmptyState } from '../charts';
|
|
5
|
-
import { useI18n } from '../i18n/ui';
|
|
6
|
-
import { Loading, ErrorState } from '../components';
|
|
7
|
-
import { useAutoRefresh } from '../hooks/useAutoRefresh';
|
|
8
|
-
|
|
9
|
-
export function FeedbackPage() {
|
|
10
|
-
const { t } = useI18n();
|
|
11
|
-
const [gfi, setGfi] = useState<FeedbackGfiResponse | null>(null);
|
|
12
|
-
const [empathyEvents, setEmpathyEvents] = useState<EmpathyEvent[]>([]);
|
|
13
|
-
const [gateBlocks, setGateBlocks] = useState<FeedbackGateBlock[]>([]);
|
|
14
|
-
const [error, setError] = useState('');
|
|
15
|
-
|
|
16
|
-
const loadAll = useCallback(async () => {
|
|
17
|
-
try {
|
|
18
|
-
const [gfiData, events, blocks] = await Promise.all([
|
|
19
|
-
api.getFeedbackGfi(),
|
|
20
|
-
api.getEmpathyEvents(20),
|
|
21
|
-
api.getFeedbackGateBlocks(20),
|
|
22
|
-
]);
|
|
23
|
-
setGfi(gfiData);
|
|
24
|
-
setEmpathyEvents(events);
|
|
25
|
-
setGateBlocks(blocks);
|
|
26
|
-
setError('');
|
|
27
|
-
} catch (err) {
|
|
28
|
-
setError(String(err));
|
|
29
|
-
}
|
|
30
|
-
}, []);
|
|
31
|
-
|
|
32
|
-
const { isRefreshing } = useAutoRefresh(loadAll, {
|
|
33
|
-
intervalMs: 15000,
|
|
34
|
-
enabled: !!gfi,
|
|
35
|
-
});
|
|
36
|
-
|
|
37
|
-
useEffect(() => {
|
|
38
|
-
loadAll();
|
|
39
|
-
}, [loadAll]);
|
|
40
|
-
|
|
41
|
-
if (error) return <ErrorState error={error} />;
|
|
42
|
-
if (!gfi) return <Loading />;
|
|
43
|
-
|
|
44
|
-
const gfiPercent = Math.min(100, (gfi.current / gfi.threshold) * 100);
|
|
45
|
-
const gfiColor = gfi.current >= gfi.threshold ? 'var(--error)' : gfi.current >= gfi.threshold * 0.8 ? 'var(--warning)' : 'var(--success)';
|
|
46
|
-
|
|
47
|
-
return (
|
|
48
|
-
<div className="page">
|
|
49
|
-
<header className="page-header">
|
|
50
|
-
<div>
|
|
51
|
-
<span className="eyebrow">{t('feedback.pageTitle')}</span>
|
|
52
|
-
<h2>{t('feedback.pageSubtitle')}</h2>
|
|
53
|
-
</div>
|
|
54
|
-
</header>
|
|
55
|
-
|
|
56
|
-
{/* GFI Dashboard */}
|
|
57
|
-
<section className="panel" style={{ marginBottom: 'var(--space-5)' }}>
|
|
58
|
-
<h3>{t('feedback.gfiDashboard')}</h3>
|
|
59
|
-
<div style={{ display: 'flex', alignItems: 'center', gap: 'var(--space-5)', padding: 'var(--space-4) 0' }}>
|
|
60
|
-
<div style={{ flex: 1 }}>
|
|
61
|
-
<div style={{ fontSize: '48px', fontWeight: 700, color: gfiColor }}>{gfi.current}</div>
|
|
62
|
-
<div style={{ fontSize: '14px', color: 'var(--text-secondary)' }}>
|
|
63
|
-
{t('feedback.threshold')}: {gfi.threshold} | {t('feedback.peakToday')}: {gfi.peakToday}
|
|
64
|
-
</div>
|
|
65
|
-
<div style={{ marginTop: 'var(--space-2)', width: '100%', height: '8px', background: 'var(--bg-sunken)', borderRadius: '4px' }}>
|
|
66
|
-
<div style={{ width: `${gfiPercent}%`, height: '100%', background: gfiColor, borderRadius: '4px', transition: 'width 0.3s' }} />
|
|
67
|
-
</div>
|
|
68
|
-
</div>
|
|
69
|
-
<div style={{ flex: 2 }}>
|
|
70
|
-
<h4 style={{ marginBottom: 'var(--space-2)' }}>{t('feedback.hourlyTrend')}</h4>
|
|
71
|
-
{gfi.trend.length > 0 && (
|
|
72
|
-
<GroupedBarChart
|
|
73
|
-
data={gfi.trend.slice(-12).map((item) => ({
|
|
74
|
-
label: item.hour.slice(-5),
|
|
75
|
-
values: [item.value],
|
|
76
|
-
}))}
|
|
77
|
-
colors={['var(--warning)']}
|
|
78
|
-
width={400}
|
|
79
|
-
height={80}
|
|
80
|
-
/>
|
|
81
|
-
)}
|
|
82
|
-
</div>
|
|
83
|
-
</div>
|
|
84
|
-
</section>
|
|
85
|
-
|
|
86
|
-
<div className="grid two-columns">
|
|
87
|
-
{/* Empathy Events */}
|
|
88
|
-
<section className="panel">
|
|
89
|
-
<h3>{t('feedback.empathyEvents')}</h3>
|
|
90
|
-
{empathyEvents.length === 0 ? (
|
|
91
|
-
<EmptyState title={t('feedback.noEmpathyEvents')} description={t('feedback.noEmpathyEventsDesc')} />
|
|
92
|
-
) : (
|
|
93
|
-
<div className="stack">
|
|
94
|
-
{empathyEvents.map((event, i) => (
|
|
95
|
-
<div className="row-card vertical" key={i}>
|
|
96
|
-
<div>
|
|
97
|
-
<strong style={{ color: event.severity === 'high' || event.severity === 'severe' || event.severity === 'critical' ? 'var(--error)' : 'var(--warning)' }}>
|
|
98
|
-
[{event.severity}] {new Date(event.timestamp).toLocaleTimeString()}
|
|
99
|
-
</strong>
|
|
100
|
-
<span>{event.origin}</span>
|
|
101
|
-
</div>
|
|
102
|
-
<div style={{ fontSize: '0.9em', color: 'var(--text-secondary)' }}>{event.reason}</div>
|
|
103
|
-
<div className="pill-row">
|
|
104
|
-
<span className="badge">+{event.score}</span>
|
|
105
|
-
<span className="badge">GFI: {event.gfiAfter}</span>
|
|
106
|
-
</div>
|
|
107
|
-
</div>
|
|
108
|
-
))}
|
|
109
|
-
</div>
|
|
110
|
-
)}
|
|
111
|
-
</section>
|
|
112
|
-
|
|
113
|
-
{/* GFI → Gate Blocks */}
|
|
114
|
-
<section className="panel">
|
|
115
|
-
<h3>{t('feedback.gateBlocks')}</h3>
|
|
116
|
-
{gateBlocks.length === 0 ? (
|
|
117
|
-
<EmptyState title={t('feedback.noGateBlocks')} description={t('feedback.noGateBlocksDesc')} />
|
|
118
|
-
) : (
|
|
119
|
-
<div className="stack">
|
|
120
|
-
{gateBlocks.map((block, i) => (
|
|
121
|
-
<div className="row-card" key={i}>
|
|
122
|
-
<div>
|
|
123
|
-
<strong>{block.toolName}</strong>
|
|
124
|
-
<span>{new Date(block.timestamp).toLocaleString()}</span>
|
|
125
|
-
</div>
|
|
126
|
-
<div className="pill-row">
|
|
127
|
-
<span className="badge">GFI: {block.gfi}</span>
|
|
128
|
-
<span className="badge">Stage: {block.trustStage}</span>
|
|
129
|
-
</div>
|
|
130
|
-
</div>
|
|
131
|
-
))}
|
|
132
|
-
</div>
|
|
133
|
-
)}
|
|
134
|
-
</section>
|
|
135
|
-
</div>
|
|
136
|
-
</div>
|
|
137
|
-
);
|
|
138
|
-
}
|
|
@@ -1,136 +0,0 @@
|
|
|
1
|
-
import React, { useCallback, useEffect, useState } from 'react';
|
|
2
|
-
import { api } from '../api';
|
|
3
|
-
import type { GateStatsResponse, GateBlockItem } from '../types';
|
|
4
|
-
import { EmptyState } from '../charts';
|
|
5
|
-
import { useI18n } from '../i18n/ui';
|
|
6
|
-
import { formatDate } from '../utils/format';
|
|
7
|
-
import { Loading, ErrorState } from '../components';
|
|
8
|
-
import { useAutoRefresh } from '../hooks/useAutoRefresh';
|
|
9
|
-
|
|
10
|
-
export function GateMonitorPage() {
|
|
11
|
-
const { t } = useI18n();
|
|
12
|
-
const [gateStats, setGateStats] = useState<GateStatsResponse | null>(null);
|
|
13
|
-
const [gateBlocks, setGateBlocks] = useState<GateBlockItem[]>([]);
|
|
14
|
-
const [error, setError] = useState('');
|
|
15
|
-
|
|
16
|
-
const loadAll = useCallback(async () => {
|
|
17
|
-
try {
|
|
18
|
-
const [stats, blocks] = await Promise.all([
|
|
19
|
-
api.getGateStats(),
|
|
20
|
-
api.getGateBlocks(50),
|
|
21
|
-
]);
|
|
22
|
-
setGateStats(stats);
|
|
23
|
-
setGateBlocks(blocks);
|
|
24
|
-
setError('');
|
|
25
|
-
} catch (err) {
|
|
26
|
-
setError(String(err));
|
|
27
|
-
}
|
|
28
|
-
}, []);
|
|
29
|
-
|
|
30
|
-
const { isRefreshing } = useAutoRefresh(loadAll, {
|
|
31
|
-
intervalMs: 30000,
|
|
32
|
-
enabled: !!gateStats,
|
|
33
|
-
});
|
|
34
|
-
|
|
35
|
-
useEffect(() => {
|
|
36
|
-
loadAll();
|
|
37
|
-
}, [loadAll]);
|
|
38
|
-
|
|
39
|
-
if (error) return <ErrorState error={error} />;
|
|
40
|
-
if (!gateStats) return <Loading />;
|
|
41
|
-
|
|
42
|
-
return (
|
|
43
|
-
<div className="page">
|
|
44
|
-
<header className="page-header">
|
|
45
|
-
<div>
|
|
46
|
-
<span className="eyebrow">{t('gate.pageTitle')}</span>
|
|
47
|
-
<h2>{t('gate.pageSubtitle')}</h2>
|
|
48
|
-
</div>
|
|
49
|
-
</header>
|
|
50
|
-
|
|
51
|
-
{/* Today's Block Stats */}
|
|
52
|
-
<section className="panel" style={{ marginBottom: 'var(--space-5)' }}>
|
|
53
|
-
<h3>{t('gate.todayStats')}</h3>
|
|
54
|
-
<div className="kpi-grid" style={{ gridTemplateColumns: 'repeat(5, 1fr)' }}>
|
|
55
|
-
<article className="panel kpi">
|
|
56
|
-
<span className="label">{t('gate.gfiBlocks')}</span>
|
|
57
|
-
<span className="value">{gateStats.today.gfiBlocks}</span>
|
|
58
|
-
</article>
|
|
59
|
-
<article className="panel kpi">
|
|
60
|
-
<span className="label">{t('gate.stageBlocks')}</span>
|
|
61
|
-
<span className="value">{gateStats.today.stageBlocks}</span>
|
|
62
|
-
</article>
|
|
63
|
-
<article className="panel kpi">
|
|
64
|
-
<span className="label">{t('gate.p03Blocks')}</span>
|
|
65
|
-
<span className="value">{gateStats.today.p03Blocks}</span>
|
|
66
|
-
</article>
|
|
67
|
-
<article className="panel kpi">
|
|
68
|
-
<span className="label">{t('gate.bypassAttempts')}</span>
|
|
69
|
-
<span className="value" style={{ color: 'var(--error)' }}>{gateStats.today.bypassAttempts}</span>
|
|
70
|
-
</article>
|
|
71
|
-
<article className="panel kpi">
|
|
72
|
-
<span className="label">{t('gate.p16Exemptions')}</span>
|
|
73
|
-
<span className="value">{gateStats.today.p16Exemptions}</span>
|
|
74
|
-
</article>
|
|
75
|
-
</div>
|
|
76
|
-
</section>
|
|
77
|
-
|
|
78
|
-
{/* Trust & EP Dual Track */}
|
|
79
|
-
<div className="grid two-columns" style={{ marginBottom: 'var(--space-5)' }}>
|
|
80
|
-
<section className="panel">
|
|
81
|
-
<h3>🔐 {t('gate.trustEngine')}</h3>
|
|
82
|
-
<div style={{ padding: 'var(--space-3) 0' }}>
|
|
83
|
-
<div style={{ fontSize: '1.5rem', fontWeight: 700 }}>Stage {gateStats.trust.stage}: {gateStats.trust.status}</div>
|
|
84
|
-
<div style={{ marginTop: 'var(--space-2)', width: '100%', height: '12px', background: 'var(--bg-sunken)', borderRadius: '6px' }}>
|
|
85
|
-
<div style={{ width: `${gateStats.trust.score}%`, height: '100%', background: 'var(--info)', borderRadius: '6px' }} />
|
|
86
|
-
</div>
|
|
87
|
-
<div style={{ fontSize: '13px', color: 'var(--text-secondary)', marginTop: 'var(--space-1)' }}>
|
|
88
|
-
{t('gate.score')}: {gateStats.trust.score}/100
|
|
89
|
-
</div>
|
|
90
|
-
</div>
|
|
91
|
-
</section>
|
|
92
|
-
<section className="panel">
|
|
93
|
-
<h3>🌱 {t('gate.evolutionEngine')}</h3>
|
|
94
|
-
<div style={{ padding: 'var(--space-3) 0' }}>
|
|
95
|
-
<div style={{ fontSize: '1.5rem', fontWeight: 700 }}>{gateStats.evolution.tier} ({gateStats.evolution.status})</div>
|
|
96
|
-
<div style={{ marginTop: 'var(--space-2)', width: '100%', height: '12px', background: 'var(--bg-sunken)', borderRadius: '6px' }}>
|
|
97
|
-
<div style={{ width: `${Math.min(100, gateStats.evolution.points / 10)}%`, height: '100%', background: 'var(--success)', borderRadius: '6px' }} />
|
|
98
|
-
</div>
|
|
99
|
-
<div className="text-sm" style={{ color: 'var(--text-secondary)', marginTop: 'var(--space-1)' }}>
|
|
100
|
-
{t('gate.points')}: {gateStats.evolution.points}
|
|
101
|
-
</div>
|
|
102
|
-
</div>
|
|
103
|
-
</section>
|
|
104
|
-
</div>
|
|
105
|
-
|
|
106
|
-
{/* Block History */}
|
|
107
|
-
<section className="panel">
|
|
108
|
-
<h3>{t('gate.blockHistory')}</h3>
|
|
109
|
-
{gateBlocks.length === 0 ? (
|
|
110
|
-
<EmptyState title={t('gate.noGateBlocks')} description={t('gate.noGateBlocksDesc')} />
|
|
111
|
-
) : (
|
|
112
|
-
<div className="list-table">
|
|
113
|
-
{gateBlocks.map((block, i) => (
|
|
114
|
-
<div className="table-row" key={i}>
|
|
115
|
-
<div>
|
|
116
|
-
<strong>{block.toolName}</strong>
|
|
117
|
-
<span>{block.filePath || '—'}</span>
|
|
118
|
-
</div>
|
|
119
|
-
<div>
|
|
120
|
-
<span className="badge">{block.gateType}</span>
|
|
121
|
-
<span>{block.reason}</span>
|
|
122
|
-
</div>
|
|
123
|
-
<div className="align-right">
|
|
124
|
-
<span className="badge">GFI: {block.gfi}</span>
|
|
125
|
-
<span>{new Date(block.timestamp).toLocaleString()}</span>
|
|
126
|
-
</div>
|
|
127
|
-
</div>
|
|
128
|
-
))}
|
|
129
|
-
</div>
|
|
130
|
-
)}
|
|
131
|
-
</section>
|
|
132
|
-
</div>
|
|
133
|
-
);
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
// Main App Component
|