| <!DOCTYPE html> |
| <html lang="en"> |
| <head> |
| <meta charset="UTF-8"> |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> |
| <title>EE Secure Client</title> |
| <link href="https://fonts.googleapis.com/css2?family=IBM+Plex+Mono:wght@400;500&family=IBM+Plex+Sans:wght@300;400;500;600&display=swap" rel="stylesheet"> |
| <style> |
| :root { |
| --bg: #0d0f12; |
| --surface: #161a20; |
| --border: #252b34; |
| --border-hover: #3a4452; |
| --accent: #00d4aa; |
| --accent-dim: rgba(0, 212, 170, 0.12); |
| --text: #e2e8f0; |
| --text-muted: #64748b; |
| --text-dim: #94a3b8; |
| --danger: #f87171; |
| --danger-dim: rgba(248, 113, 113, 0.1); |
| --mono: 'IBM Plex Mono', monospace; |
| --sans: 'IBM Plex Sans', sans-serif; |
| } |
| |
| * { box-sizing: border-box; margin: 0; padding: 0; } |
| |
| body { |
| background: var(--bg); |
| color: var(--text); |
| font-family: var(--sans); |
| min-height: 100vh; |
| padding: 48px 24px; |
| } |
| |
| .page { |
| max-width: 620px; |
| margin: 0 auto; |
| } |
| |
| |
| .header { |
| margin-bottom: 40px; |
| } |
| .header-badge { |
| display: inline-flex; |
| align-items: center; |
| gap: 7px; |
| background: var(--accent-dim); |
| border: 1px solid rgba(0,212,170,0.25); |
| color: var(--accent); |
| font-family: var(--mono); |
| font-size: 11px; |
| letter-spacing: 0.08em; |
| text-transform: uppercase; |
| padding: 5px 12px; |
| border-radius: 4px; |
| margin-bottom: 16px; |
| } |
| .lock-dot { |
| width: 7px; height: 7px; |
| background: var(--accent); |
| border-radius: 50%; |
| animation: blink 2.5s ease-in-out infinite; |
| } |
| @keyframes blink { 0%,100%{opacity:1} 50%{opacity:0.3} } |
| |
| h1 { |
| font-size: 26px; |
| font-weight: 600; |
| letter-spacing: -0.02em; |
| color: var(--text); |
| margin-bottom: 6px; |
| } |
| .subtitle { |
| font-size: 14px; |
| color: var(--text-muted); |
| line-height: 1.5; |
| } |
| |
| |
| .card { |
| background: var(--surface); |
| border: 1px solid var(--border); |
| border-radius: 10px; |
| padding: 28px; |
| margin-bottom: 20px; |
| } |
| |
| .section-label { |
| font-family: var(--mono); |
| font-size: 10px; |
| letter-spacing: 0.1em; |
| text-transform: uppercase; |
| color: var(--text-muted); |
| margin-bottom: 16px; |
| padding-bottom: 10px; |
| border-bottom: 1px solid var(--border); |
| } |
| |
| |
| .field { margin-bottom: 20px; } |
| .field:last-child { margin-bottom: 0; } |
| |
| label { |
| display: block; |
| font-size: 13px; |
| font-weight: 500; |
| color: var(--text-dim); |
| margin-bottom: 7px; |
| } |
| label span { |
| font-family: var(--mono); |
| font-size: 11px; |
| color: var(--text-muted); |
| font-weight: 400; |
| } |
| |
| input, textarea { |
| width: 100%; |
| background: var(--bg); |
| border: 1px solid var(--border); |
| border-radius: 6px; |
| padding: 10px 14px; |
| color: var(--text); |
| font-family: var(--sans); |
| font-size: 14px; |
| transition: border-color 0.15s; |
| outline: none; |
| -webkit-appearance: none; |
| } |
| input:focus, textarea:focus { |
| border-color: var(--accent); |
| box-shadow: 0 0 0 3px rgba(0,212,170,0.08); |
| } |
| input::placeholder, textarea::placeholder { |
| color: var(--text-muted); |
| } |
| textarea { |
| resize: vertical; |
| min-height: 120px; |
| line-height: 1.6; |
| font-family: var(--mono); |
| font-size: 13px; |
| } |
| |
| |
| .field-row { |
| display: grid; |
| grid-template-columns: 1fr 1fr; |
| gap: 16px; |
| } |
| |
| |
| .btn { |
| width: 100%; |
| padding: 13px; |
| background: var(--accent); |
| color: #0d0f12; |
| border: none; |
| border-radius: 6px; |
| font-family: var(--sans); |
| font-size: 14px; |
| font-weight: 600; |
| cursor: pointer; |
| letter-spacing: 0.01em; |
| transition: opacity 0.15s, transform 0.1s; |
| margin-top: 4px; |
| } |
| .btn:hover { opacity: 0.88; } |
| .btn:active { transform: scale(0.99); } |
| .btn:disabled { opacity: 0.4; cursor: not-allowed; } |
| |
| |
| .result-card { |
| background: var(--surface); |
| border: 1px solid var(--border); |
| border-radius: 10px; |
| overflow: hidden; |
| } |
| .result-header { |
| display: flex; |
| align-items: center; |
| justify-content: space-between; |
| padding: 14px 20px; |
| border-bottom: 1px solid var(--border); |
| background: rgba(0,212,170,0.04); |
| } |
| .result-header-label { |
| font-family: var(--mono); |
| font-size: 11px; |
| letter-spacing: 0.08em; |
| text-transform: uppercase; |
| color: var(--accent); |
| } |
| .result-body { |
| padding: 20px; |
| font-family: var(--mono); |
| font-size: 13px; |
| line-height: 1.7; |
| color: var(--text); |
| white-space: pre-wrap; |
| word-break: break-word; |
| } |
| |
| |
| .error-card { |
| background: var(--danger-dim); |
| border: 1px solid rgba(248,113,113,0.25); |
| border-radius: 8px; |
| padding: 14px 18px; |
| font-size: 13px; |
| color: var(--danger); |
| font-family: var(--mono); |
| line-height: 1.6; |
| word-break: break-all; |
| } |
| .error-card::before { |
| content: "ERROR — "; |
| font-weight: 500; |
| } |
| |
| |
| .btn.loading { |
| position: relative; |
| color: transparent; |
| } |
| .btn.loading::after { |
| content: ''; |
| position: absolute; |
| inset: 0; |
| margin: auto; |
| width: 18px; height: 18px; |
| border: 2px solid rgba(13,15,18,0.3); |
| border-top-color: #0d0f12; |
| border-radius: 50%; |
| animation: spin 0.7s linear infinite; |
| } |
| @keyframes spin { to { transform: rotate(360deg); } } |
| |
| .help-text { |
| font-size: 12px; |
| color: var(--text-muted); |
| margin-top: 5px; |
| } |
| </style> |
| </head> |
| <body> |
| <div class="page"> |
|
|
| <div class="header"> |
| <div class="header-badge"> |
| <span class="lock-dot"></span> |
| End-to-end encrypted |
| </div> |
| <h1>Equivariant Encryption Client</h1> |
| <p class="subtitle">Prompts are encrypted locally before leaving this Space — the inference server only ever sees scrambled embeddings.</p> |
| </div> |
|
|
| <form method="POST" id="mainForm"> |
|
|
| <div class="card"> |
| <div class="section-label">Connection</div> |
|
|
| <div class="field"> |
| <label>Server Space URL</label> |
| <input type="url" name="server_url" |
| placeholder="https://your-ee-server.hf.space" |
| value="https://broadfield-dev-equivariant-encryption-server.hf.space" |
| required readonly> |
| </div> |
|
|
| <div class="field"> |
| <label>EE Model <span>— from Hugging Face Hub</span></label> |
| <input type="text" name="ee_model_name" |
| placeholder="username/model-ee" |
| value="broadfield-dev/Qwen3-0.6B-dp-ee" |
| required readonly> |
| </div> |
| </div> |
|
|
| <div class="card"> |
| <div class="section-label">Encryption Key</div> |
|
|
| <div class="field"> |
| <label>Secret Seed <span>— never leaves this client</span></label> |
| <input type="number" name="ee_seed" |
| value="{{ form.get('ee_seed', '424242') }}" |
| required> |
| <p class="help-text">Use the same seed that was used when transforming the model.</p> |
| </div> |
| </div> |
|
|
| <div class="card"> |
| <div class="section-label">Prompt</div> |
|
|
| <div class="field"> |
| <label>Your message</label> |
| <textarea name="prompt" required |
| placeholder="Enter your prompt here...">{{ form.get('prompt', '') }}</textarea> |
| </div> |
|
|
| <div class="field"> |
| <label>Max new tokens</label> |
| <input type="number" name="max_tokens" |
| value="{{ form.get('max_tokens', '256') }}" |
| min="1" max="2048"> |
| </div> |
| </div> |
|
|
| <button type="submit" class="btn" id="submitBtn"> |
| Encrypt & Send → |
| </button> |
|
|
| </form> |
|
|
| {% if error %} |
| <div style="margin-top:20px"> |
| <div class="error-card">{{ error }}</div> |
| </div> |
| {% endif %} |
|
|
| {% if result %} |
| <div style="margin-top:20px"> |
| <div class="result-card"> |
| <div class="result-header"> |
| <span class="result-header-label">Server response</span> |
| <span style="font-size:12px; color: var(--text-muted); font-family: var(--mono);">decrypted locally</span> |
| </div> |
| <div class="result-body">{{ result }}</div> |
| </div> |
| </div> |
| {% endif %} |
|
|
| </div> |
|
|
| <script> |
| document.getElementById('mainForm').addEventListener('submit', function() { |
| const btn = document.getElementById('submitBtn'); |
| btn.classList.add('loading'); |
| btn.disabled = true; |
| }); |
| </script> |
| </body> |
| </html> |