This repository has been archived on 2026-02-15. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
breakpilot-pwa/pca-platform/demo/index.html
Benjamin Admin bfdaf63ba9 fix: Restore all files lost during destructive rebase
A previous `git pull --rebase origin main` dropped 177 local commits,
losing 3400+ files across admin-v2, backend, studio-v2, website,
klausur-service, and many other services. The partial restore attempt
(660295e2) only recovered some files.

This commit restores all missing files from pre-rebase ref 98933f5e
while preserving post-rebase additions (night-scheduler, night-mode UI,
NightModeWidget dashboard integration).

Restored features include:
- AI Module Sidebar (FAB), OCR Labeling, OCR Compare
- GPU Dashboard, RAG Pipeline, Magic Help
- Klausur-Korrektur (8 files), Abitur-Archiv (5+ files)
- Companion, Zeugnisse-Crawler, Screen Flow
- Full backend, studio-v2, website, klausur-service
- All compliance SDKs, agent-core, voice-service
- CI/CD configs, documentation, scripts

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-09 09:51:32 +01:00

445 lines
13 KiB
HTML

<!DOCTYPE html>
<html lang="de">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>PCA Platform Demo - Human vs Bot Detection</title>
<style>
* {
box-sizing: border-box;
margin: 0;
padding: 0;
}
body {
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
line-height: 1.6;
color: #333;
background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
min-height: 200vh; /* Long page for scroll testing */
}
.container {
max-width: 800px;
margin: 0 auto;
padding: 2rem;
}
header {
background: white;
border-radius: 12px;
padding: 2rem;
margin-bottom: 2rem;
box-shadow: 0 4px 20px rgba(0,0,0,0.1);
}
h1 {
color: #667eea;
margin-bottom: 0.5rem;
}
.subtitle {
color: #666;
font-size: 1.1rem;
}
.score-panel {
background: white;
border-radius: 12px;
padding: 2rem;
margin-bottom: 2rem;
box-shadow: 0 4px 20px rgba(0,0,0,0.1);
position: sticky;
top: 1rem;
z-index: 100;
}
.score-display {
display: flex;
align-items: center;
gap: 2rem;
}
.score-circle {
width: 120px;
height: 120px;
border-radius: 50%;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
font-weight: bold;
transition: all 0.3s ease;
}
.score-circle.low {
background: linear-gradient(135deg, #ff6b6b, #ee5a5a);
color: white;
}
.score-circle.medium {
background: linear-gradient(135deg, #feca57, #ff9f43);
color: white;
}
.score-circle.high {
background: linear-gradient(135deg, #1dd1a1, #10ac84);
color: white;
}
.score-value {
font-size: 2rem;
}
.score-label {
font-size: 0.8rem;
opacity: 0.9;
}
.score-details {
flex: 1;
}
.score-details h3 {
margin-bottom: 0.5rem;
}
.status-badge {
display: inline-block;
padding: 0.25rem 0.75rem;
border-radius: 20px;
font-size: 0.85rem;
font-weight: 600;
}
.status-badge.allow {
background: #d4edda;
color: #155724;
}
.status-badge.challenge {
background: #fff3cd;
color: #856404;
}
.metrics-grid {
display: grid;
grid-template-columns: repeat(4, 1fr);
gap: 1rem;
margin-top: 1rem;
}
.metric {
text-align: center;
padding: 0.5rem;
background: #f8f9fa;
border-radius: 8px;
}
.metric-value {
font-size: 1.5rem;
font-weight: bold;
color: #667eea;
}
.metric-label {
font-size: 0.75rem;
color: #666;
}
.content-section {
background: white;
border-radius: 12px;
padding: 2rem;
margin-bottom: 2rem;
box-shadow: 0 4px 20px rgba(0,0,0,0.1);
}
.content-section h2 {
color: #333;
margin-bottom: 1rem;
padding-bottom: 0.5rem;
border-bottom: 2px solid #667eea;
}
.content-section p {
margin-bottom: 1rem;
}
.demo-buttons {
display: flex;
gap: 1rem;
flex-wrap: wrap;
}
button {
padding: 0.75rem 1.5rem;
border: none;
border-radius: 8px;
font-size: 1rem;
cursor: pointer;
transition: transform 0.2s, box-shadow 0.2s;
}
button:hover {
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(0,0,0,0.15);
}
.btn-primary {
background: linear-gradient(135deg, #667eea, #764ba2);
color: white;
}
.btn-secondary {
background: #f8f9fa;
color: #333;
border: 1px solid #ddd;
}
.btn-warning {
background: linear-gradient(135deg, #feca57, #ff9f43);
color: white;
}
.protected-content {
background: #f8f9fa;
border: 2px dashed #ddd;
border-radius: 8px;
padding: 2rem;
text-align: center;
margin-top: 1rem;
}
.protected-content.unlocked {
border-color: #1dd1a1;
background: #d4edda;
}
.log-panel {
background: #1a1a2e;
color: #eee;
border-radius: 12px;
padding: 1rem;
font-family: 'Monaco', 'Menlo', monospace;
font-size: 0.85rem;
max-height: 300px;
overflow-y: auto;
}
.log-entry {
padding: 0.25rem 0;
border-bottom: 1px solid #333;
}
.log-entry:last-child {
border-bottom: none;
}
.log-time {
color: #667eea;
}
.log-score {
color: #1dd1a1;
}
footer {
text-align: center;
color: white;
padding: 2rem;
opacity: 0.8;
}
</style>
</head>
<body>
<div class="container">
<header>
<h1>PCA Platform Demo</h1>
<p class="subtitle">Person - Corporate - Agent | Human vs Bot Detection</p>
</header>
<div class="score-panel">
<div class="score-display">
<div class="score-circle low" id="scoreCircle">
<span class="score-value" id="scoreValue">0.00</span>
<span class="score-label">Human Score</span>
</div>
<div class="score-details">
<h3>Status: <span class="status-badge challenge" id="statusBadge">Initializing...</span></h3>
<p id="statusMessage">Collecting behavioral data...</p>
<div class="metrics-grid">
<div class="metric">
<div class="metric-value" id="metricDwell">0%</div>
<div class="metric-label">Dwell Time</div>
</div>
<div class="metric">
<div class="metric-value" id="metricScroll">0%</div>
<div class="metric-label">Scroll Depth</div>
</div>
<div class="metric">
<div class="metric-value" id="metricClicks">0</div>
<div class="metric-label">Clicks</div>
</div>
<div class="metric">
<div class="metric-value" id="metricMouse">0</div>
<div class="metric-label">Mouse Moves</div>
</div>
</div>
</div>
</div>
</div>
<section class="content-section">
<h2>How It Works</h2>
<p>
The PCA SDK analyzes your browsing behavior to distinguish humans from bots.
It tracks metrics like scroll depth, mouse movements, click patterns, and dwell time
- all without collecting personal information.
</p>
<p>
<strong>Scroll down</strong>, <strong>move your mouse</strong>, and <strong>click around</strong>
to increase your human score. Once you reach a score of 0.7+, you'll be recognized as human.
</p>
</section>
<section class="content-section">
<h2>Test the SDK</h2>
<div class="demo-buttons">
<button class="btn-primary" onclick="testEvaluate()">Evaluate Now</button>
<button class="btn-secondary" onclick="testTick()">Send Tick</button>
<button class="btn-warning" onclick="testStepUp()">Trigger Step-Up</button>
</div>
</section>
<section class="content-section">
<h2>Protected Content</h2>
<p>This content is protected and requires a human score of 0.7 or higher to access:</p>
<div class="protected-content" id="protectedContent">
<p>Content locked. Increase your score to unlock.</p>
</div>
</section>
<section class="content-section">
<h2>More Content (Scroll Test)</h2>
<p>Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.</p>
<p>Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.</p>
<p>Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.</p>
<p>Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim id est laborum.</p>
</section>
<section class="content-section">
<h2>Event Log</h2>
<div class="log-panel" id="logPanel">
<div class="log-entry"><span class="log-time">[--:--:--]</span> SDK initializing...</div>
</div>
</section>
<footer>
<p>PCA Platform v0.1.0 | GDPR Compliant | No PII Collected</p>
</footer>
</div>
<!-- PCA SDK -->
<script src="/sdk/pca-sdk.js"></script>
<script>
const logPanel = document.getElementById('logPanel');
const scoreCircle = document.getElementById('scoreCircle');
const scoreValue = document.getElementById('scoreValue');
const statusBadge = document.getElementById('statusBadge');
const statusMessage = document.getElementById('statusMessage');
const protectedContent = document.getElementById('protectedContent');
function log(message) {
const time = new Date().toLocaleTimeString();
const entry = document.createElement('div');
entry.className = 'log-entry';
entry.innerHTML = `<span class="log-time">[${time}]</span> ${message}`;
logPanel.appendChild(entry);
logPanel.scrollTop = logPanel.scrollHeight;
}
function updateUI(score, action, data) {
// Update score display
scoreValue.textContent = score.toFixed(2);
// Update score circle color
scoreCircle.className = 'score-circle';
if (score >= 0.7) {
scoreCircle.classList.add('high');
} else if (score >= 0.4) {
scoreCircle.classList.add('medium');
} else {
scoreCircle.classList.add('low');
}
// Update status badge
statusBadge.textContent = action === 'allow' ? 'Human Detected' : 'Verification Needed';
statusBadge.className = 'status-badge ' + action;
// Update status message
if (score >= 0.7) {
statusMessage.textContent = 'Congratulations! You are recognized as a human visitor.';
} else if (score >= 0.4) {
statusMessage.textContent = 'Almost there! Keep interacting with the page.';
} else {
statusMessage.textContent = 'Scroll, click, and move your mouse to prove you are human.';
}
// Update metrics
if (data && data.metrics) {
document.getElementById('metricDwell').textContent = Math.round(data.metrics.dwell_ratio * 100) + '%';
document.getElementById('metricScroll').textContent = Math.round(data.metrics.scroll_depth * 100) + '%';
document.getElementById('metricClicks').textContent = data.metrics.clicks;
document.getElementById('metricMouse').textContent = data.metrics.mouse_moves;
}
// Update protected content
if (score >= 0.7) {
protectedContent.className = 'protected-content unlocked';
protectedContent.innerHTML = `
<h3>Content Unlocked!</h3>
<p>This is the secret content that only humans can see.</p>
<p>Your session ID: ${PCA.getSessionId()}</p>
`;
}
log(`Score updated: <span class="log-score">${score.toFixed(3)}</span> | Action: ${action}`);
}
// Initialize SDK with custom config
const customConfig = {
thresholds: {
score_pass: 0.7,
score_challenge: 0.4
},
weights: {
dwell_ratio: 0.30,
scroll_score: 0.25,
pointer_variance: 0.20,
click_rate: 0.25
},
tick: {
endpoint: 'http://localhost:8085/pca/v1/tick',
interval_ms: 3000
},
step_up: {
methods: ['webauthn', 'pow'],
primary: 'pow',
webauthn: {
enabled: true,
challenge_endpoint: 'http://localhost:8085/pca/v1/webauthn-challenge'
},
pow: {
enabled: true,
difficulty: 4,
max_duration_ms: 5000
}
}
};
// Wait for SDK to load
if (typeof PCA !== 'undefined') {
PCA.init(customConfig);
log('SDK initialized with custom config');
// Subscribe to score updates
PCA.onScoreUpdate((score, action, data) => {
updateUI(score, action, PCA.evaluate());
});
// Update metrics every second
setInterval(() => {
const data = PCA.evaluate();
updateUI(data.score, data.score >= 0.7 ? 'allow' : 'challenge', data);
}, 1000);
} else {
log('Error: PCA SDK not loaded');
}
// Test functions
function testEvaluate() {
const result = PCA.evaluate();
log(`Manual evaluation: ${JSON.stringify(result)}`);
updateUI(result.score, result.score >= 0.7 ? 'allow' : 'challenge', result);
}
function testTick() {
PCA.tick();
log('Tick sent to server');
}
async function testStepUp() {
log('Starting step-up verification (PoW)...');
const success = await PCA.triggerStepUp();
if (success) {
log('Step-up verification successful!');
} else {
log('Step-up verification failed or cancelled');
}
}
</script>
</body>
</html>