2026-02-21 10:04:47 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="ssh-auth-demo">
|
|
|
|
|
|
<div class="demo-header">
|
|
|
|
|
|
<span class="title">SSH 密钥认证:你的数字身份证</span>
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<span class="subtitle">对称加密 vs 非对称加密 · 密钥对生成 · 认证流程</span>
|
2026-02-21 10:04:47 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="control-panel">
|
|
|
|
|
|
<div class="scenario-btns">
|
|
|
|
|
|
<button
|
|
|
|
|
|
v-for="s in scenarios"
|
|
|
|
|
|
:key="s.id"
|
|
|
|
|
|
:class="['scenario-btn', { active: activeScenario === s.id }]"
|
|
|
|
|
|
@click="activeScenario = s.id"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{ s.label }}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="visualization-area">
|
|
|
|
|
|
<!-- Scenario 1: Password vs Key -->
|
|
|
|
|
|
<div v-if="activeScenario === 'compare'" class="compare-section">
|
|
|
|
|
|
<div class="compare-grid">
|
|
|
|
|
|
<div class="compare-card password">
|
|
|
|
|
|
<div class="card-icon">🔑</div>
|
|
|
|
|
|
<div class="card-title">密码登录</div>
|
|
|
|
|
|
<div class="card-flow">
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<div v-for="(step, i) in passwordFlow" :key="i" class="flow-step">
|
2026-02-21 10:04:47 +08:00
|
|
|
|
<span class="step-num">{{ i + 1 }}</span>
|
|
|
|
|
|
<span class="step-text">{{ step }}</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="card-verdict danger">
|
|
|
|
|
|
<span class="verdict-icon">⚠️</span>
|
|
|
|
|
|
<span>密码在网络上传输,可能被截获</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="compare-card key">
|
|
|
|
|
|
<div class="card-icon">🔐</div>
|
|
|
|
|
|
<div class="card-title">密钥登录</div>
|
|
|
|
|
|
<div class="card-flow">
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<div v-for="(step, i) in keyFlow" :key="i" class="flow-step">
|
2026-02-21 10:04:47 +08:00
|
|
|
|
<span class="step-num">{{ i + 1 }}</span>
|
|
|
|
|
|
<span class="step-text">{{ step }}</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="card-verdict success">
|
|
|
|
|
|
<span class="verdict-icon">✅</span>
|
|
|
|
|
|
<span>私钥永远不离开你的电脑</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- Scenario 2: Key Pair Generation -->
|
|
|
|
|
|
<div v-if="activeScenario === 'keygen'" class="keygen-section">
|
|
|
|
|
|
<div class="keygen-visual">
|
|
|
|
|
|
<div class="keygen-command">
|
|
|
|
|
|
<code>ssh-keygen -t ed25519 -C "your@email.com"</code>
|
|
|
|
|
|
<button
|
|
|
|
|
|
class="gen-btn"
|
|
|
|
|
|
:disabled="isGenerating"
|
|
|
|
|
|
@click="generateKeys"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{ isGenerating ? '生成中...' : '生成密钥对' }}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="key-pair" :class="{ generated: keysGenerated }">
|
|
|
|
|
|
<div class="key-card private" :class="{ visible: keysGenerated }">
|
|
|
|
|
|
<div class="key-header">
|
|
|
|
|
|
<span class="key-icon">🔒</span>
|
|
|
|
|
|
<span class="key-name">私钥 (Private Key)</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="key-location">~/.ssh/id_ed25519</div>
|
|
|
|
|
|
<div class="key-content">
|
|
|
|
|
|
<code>{{ privateKeyDisplay }}</code>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="key-rule danger">绝不外泄 · 留在本机</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="key-arrow" :class="{ visible: keysGenerated }">
|
|
|
|
|
|
<span class="arrow-text">数学关联</span>
|
|
|
|
|
|
<span class="arrow-icon">↔</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="key-card public" :class="{ visible: keysGenerated }">
|
|
|
|
|
|
<div class="key-header">
|
|
|
|
|
|
<span class="key-icon">🌍</span>
|
|
|
|
|
|
<span class="key-name">公钥 (Public Key)</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="key-location">~/.ssh/id_ed25519.pub</div>
|
|
|
|
|
|
<div class="key-content">
|
|
|
|
|
|
<code>{{ publicKeyDisplay }}</code>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="key-rule success">可以给任何人 · 放到服务器</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div v-if="keysGenerated" class="key-analogy">
|
|
|
|
|
|
<strong>生活类比:</strong>公钥 = 锁(可以随便装)· 私钥 =
|
|
|
|
|
|
钥匙(只有你有)· 用锁锁住的东西,只有对应的钥匙能打开
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- Scenario 3: Auth Flow -->
|
|
|
|
|
|
<div v-if="activeScenario === 'auth'" class="auth-section">
|
|
|
|
|
|
<div class="auth-controls">
|
|
|
|
|
|
<button
|
|
|
|
|
|
class="action-btn"
|
|
|
|
|
|
:disabled="authStep > 0 && authStep < 5"
|
|
|
|
|
|
@click="startAuth"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{
|
|
|
|
|
|
authStep === 0
|
|
|
|
|
|
? '开始认证'
|
|
|
|
|
|
: authStep >= 5
|
|
|
|
|
|
? '重新演示'
|
|
|
|
|
|
: '认证中...'
|
|
|
|
|
|
}}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="auth-flow">
|
|
|
|
|
|
<div class="auth-parties">
|
|
|
|
|
|
<div class="party client">
|
|
|
|
|
|
<div class="party-icon">💻</div>
|
|
|
|
|
|
<div class="party-name">你的电脑</div>
|
|
|
|
|
|
<div class="party-has">持有:私钥</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="auth-messages">
|
|
|
|
|
|
<div
|
|
|
|
|
|
:class="['msg', { active: authStep >= 1 }]"
|
|
|
|
|
|
class="msg-right"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span class="msg-label">① 请求连接</span>
|
|
|
|
|
|
<span class="msg-detail">"我要用密钥登录"</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div :class="['msg', { active: authStep >= 2 }]" class="msg-left">
|
|
|
|
|
|
<span class="msg-label">② 发送随机挑战</span>
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<span class="msg-detail">"请证明你有私钥:用它签名这段随机数据"</span>
|
2026-02-21 10:04:47 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div
|
|
|
|
|
|
:class="['msg', { active: authStep >= 3 }]"
|
|
|
|
|
|
class="msg-right"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span class="msg-label">③ 返回签名</span>
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<span class="msg-detail">"用私钥签名后的结果(私钥本身不发送)"</span>
|
2026-02-21 10:04:47 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div :class="['msg', { active: authStep >= 4 }]" class="msg-left">
|
|
|
|
|
|
<span class="msg-label">④ 用公钥验证</span>
|
|
|
|
|
|
<span class="msg-detail">"用存储的公钥验证签名 → 匹配!"</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div :class="['msg', 'msg-result', { active: authStep >= 5 }]">
|
|
|
|
|
|
<span class="msg-label">⑤ 认证成功</span>
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<span class="msg-detail">"欢迎登录!从始至终,私钥没离开过你的电脑"</span>
|
2026-02-21 10:04:47 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="party server">
|
|
|
|
|
|
<div class="party-icon">🖥️</div>
|
|
|
|
|
|
<div class="party-name">远程服务器</div>
|
|
|
|
|
|
<div class="party-has">持有:公钥</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- Scenario 4: Common Uses -->
|
|
|
|
|
|
<div v-if="activeScenario === 'uses'" class="uses-section">
|
|
|
|
|
|
<div class="uses-grid">
|
|
|
|
|
|
<div v-for="use in commonUses" :key="use.name" class="use-card">
|
|
|
|
|
|
<div class="use-icon">{{ use.icon }}</div>
|
|
|
|
|
|
<div class="use-name">{{ use.name }}</div>
|
|
|
|
|
|
<div class="use-cmd">
|
|
|
|
|
|
<code>{{ use.command }}</code>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="use-desc">{{ use.desc }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="config-tips">
|
|
|
|
|
|
<div class="tip-title">~/.ssh/config 快捷配置</div>
|
|
|
|
|
|
<pre class="tip-code"><code>Host my-server
|
|
|
|
|
|
HostName 192.168.1.100
|
|
|
|
|
|
User deploy
|
|
|
|
|
|
IdentityFile ~/.ssh/id_ed25519
|
|
|
|
|
|
|
|
|
|
|
|
Host github.com
|
|
|
|
|
|
HostName github.com
|
|
|
|
|
|
User git
|
|
|
|
|
|
IdentityFile ~/.ssh/id_ed25519</code></pre>
|
|
|
|
|
|
<div class="tip-result">
|
|
|
|
|
|
配置后:<code>ssh my-server</code> 即可一键连接
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="info-box">
|
|
|
|
|
|
<strong>核心思想:</strong>
|
2026-02-24 00:18:09 +08:00
|
|
|
|
<span v-if="activeScenario === 'compare'">SSH
|
|
|
|
|
|
密钥登录比密码更安全,因为私钥从不在网络上传输,无法被中间人窃取。</span>
|
|
|
|
|
|
<span v-else-if="activeScenario === 'keygen'">一次 ssh-keygen
|
|
|
|
|
|
生成一对密钥:私钥自己保管,公钥放到目标服务器或平台。</span>
|
|
|
|
|
|
<span v-else-if="activeScenario === 'auth'">认证过程基于"挑战-响应"机制:服务器出题,你的私钥签名作答,公钥验证答案。全程私钥不离开本机。</span>
|
|
|
|
|
|
<span v-else>SSH 密钥不仅用于服务器登录,也是 Git (GitHub/GitLab)
|
|
|
|
|
|
等开发工具的标准身份认证方式。</span>
|
2026-02-21 10:04:47 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
|
|
|
|
|
import { ref } from 'vue'
|
|
|
|
|
|
|
|
|
|
|
|
const activeScenario = ref('compare')
|
|
|
|
|
|
|
|
|
|
|
|
const scenarios = [
|
|
|
|
|
|
{ id: 'compare', label: '密码 vs 密钥' },
|
|
|
|
|
|
{ id: 'keygen', label: '生成密钥对' },
|
|
|
|
|
|
{ id: 'auth', label: '认证流程' },
|
|
|
|
|
|
{ id: 'uses', label: '常见用途' }
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
const passwordFlow = [
|
|
|
|
|
|
'输入用户名和密码',
|
|
|
|
|
|
'密码通过网络发送到服务器',
|
|
|
|
|
|
'服务器比对密码是否正确',
|
|
|
|
|
|
'每次都要输密码'
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
const keyFlow = [
|
|
|
|
|
|
'事先把公钥放到服务器',
|
|
|
|
|
|
'连接时发送身份标识(不发私钥)',
|
|
|
|
|
|
'服务器用公钥出"数学题"',
|
|
|
|
|
|
'你的私钥在本地"答题",只发答案'
|
|
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
const isGenerating = ref(false)
|
|
|
|
|
|
const keysGenerated = ref(false)
|
|
|
|
|
|
const privateKeyDisplay = ref(
|
|
|
|
|
|
'-----BEGIN OPENSSH PRIVATE KEY-----\n(等待生成...)\n-----END OPENSSH PRIVATE KEY-----'
|
|
|
|
|
|
)
|
|
|
|
|
|
const publicKeyDisplay = ref('(等待生成...)')
|
|
|
|
|
|
|
|
|
|
|
|
const generateKeys = async () => {
|
|
|
|
|
|
if (isGenerating.value) return
|
|
|
|
|
|
isGenerating.value = true
|
|
|
|
|
|
keysGenerated.value = false
|
|
|
|
|
|
|
|
|
|
|
|
await new Promise((r) => setTimeout(r, 800))
|
|
|
|
|
|
|
|
|
|
|
|
privateKeyDisplay.value =
|
|
|
|
|
|
'-----BEGIN OPENSSH PRIVATE KEY-----\nb3BlbnNzaC1rZXktdjEAAAA...\n(2048 位密钥,绝不外传)\n-----END OPENSSH PRIVATE KEY-----'
|
|
|
|
|
|
publicKeyDisplay.value =
|
|
|
|
|
|
'ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAA\nIGx...kF your@email.com'
|
|
|
|
|
|
|
|
|
|
|
|
keysGenerated.value = true
|
|
|
|
|
|
isGenerating.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const authStep = ref(0)
|
|
|
|
|
|
|
|
|
|
|
|
const startAuth = async () => {
|
|
|
|
|
|
if (authStep.value > 0 && authStep.value < 5) return
|
|
|
|
|
|
authStep.value = 0
|
|
|
|
|
|
|
|
|
|
|
|
for (let i = 1; i <= 5; i++) {
|
|
|
|
|
|
await new Promise((r) => setTimeout(r, 800))
|
|
|
|
|
|
authStep.value = i
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const commonUses = [
|
|
|
|
|
|
{
|
|
|
|
|
|
icon: '🖥️',
|
|
|
|
|
|
name: '远程服务器',
|
|
|
|
|
|
command: 'ssh user@server',
|
|
|
|
|
|
desc: '免密码登录 Linux/Mac 服务器'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
icon: '🐙',
|
|
|
|
|
|
name: 'GitHub',
|
|
|
|
|
|
command: 'git push origin main',
|
|
|
|
|
|
desc: '用 SSH 协议推送代码'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
icon: '🦊',
|
|
|
|
|
|
name: 'GitLab',
|
|
|
|
|
|
command: 'git clone git@gitlab.com:...',
|
|
|
|
|
|
desc: '克隆私有仓库'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
icon: '📦',
|
|
|
|
|
|
name: 'SCP 传文件',
|
|
|
|
|
|
command: 'scp file.txt user@server:~/',
|
|
|
|
|
|
desc: '安全复制文件到远程'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
icon: '🚇',
|
|
|
|
|
|
name: 'SSH 隧道',
|
|
|
|
|
|
command: 'ssh -L 8080:localhost:3000 server',
|
|
|
|
|
|
desc: '将远程端口映射到本地'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
icon: '🐳',
|
|
|
|
|
|
name: '部署服务',
|
|
|
|
|
|
command: 'ssh deploy@prod "docker pull..."',
|
|
|
|
|
|
desc: '远程执行部署命令'
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.ssh-auth-demo {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
background: var(--vp-c-bg-soft);
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
margin: 0.5rem 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
align-items: baseline;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header .title {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 1rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header .subtitle {
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.control-panel {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.5rem 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.scenario-btns {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.scenario-btn {
|
|
|
|
|
|
padding: 0.35rem 0.7rem;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
transition: all 0.2s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.scenario-btn.active {
|
|
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Compare Section */
|
|
|
|
|
|
.compare-grid {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: 1fr 1fr;
|
|
|
|
|
|
gap: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.compare-card {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-icon {
|
|
|
|
|
|
font-size: 1.5rem;
|
|
|
|
|
|
margin-bottom: 0.25rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-title {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
|
margin-bottom: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-flow {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
gap: 0.35rem;
|
|
|
|
|
|
margin-bottom: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.flow-step {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.step-num {
|
|
|
|
|
|
width: 20px;
|
|
|
|
|
|
height: 20px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
border-radius: 50%;
|
|
|
|
|
|
font-size: 0.7rem;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-verdict {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.35rem;
|
|
|
|
|
|
padding: 0.35rem 0.5rem;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-verdict.danger {
|
|
|
|
|
|
background: rgba(239, 68, 68, 0.1);
|
|
|
|
|
|
color: var(--vp-c-danger-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.card-verdict.success {
|
|
|
|
|
|
background: rgba(16, 185, 129, 0.1);
|
|
|
|
|
|
color: var(--vp-c-green-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Keygen Section */
|
|
|
|
|
|
.keygen-command {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.5rem 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.keygen-command code {
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.25rem 0.5rem;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
min-width: 200px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.gen-btn {
|
|
|
|
|
|
padding: 0.35rem 0.7rem;
|
|
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
border: none;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.gen-btn:disabled {
|
|
|
|
|
|
opacity: 0.6;
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-pair {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: 1fr auto 1fr;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-card {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border: 2px solid var(--vp-c-divider);
|
|
|
|
|
|
opacity: 0.4;
|
|
|
|
|
|
transition: all 0.5s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-card.visible {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-card.private {
|
|
|
|
|
|
border-color: var(--vp-c-danger-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-card.public {
|
|
|
|
|
|
border-color: var(--vp-c-green-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.35rem;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
margin-bottom: 0.25rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-location {
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
margin-bottom: 0.35rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-content code {
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
font-size: 0.72rem;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.35rem;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
white-space: pre-wrap;
|
|
|
|
|
|
word-break: break-all;
|
|
|
|
|
|
line-height: 1.4;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-rule {
|
|
|
|
|
|
margin-top: 0.35rem;
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
padding: 0.2rem;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-rule.danger {
|
|
|
|
|
|
background: rgba(239, 68, 68, 0.1);
|
|
|
|
|
|
color: var(--vp-c-danger-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-rule.success {
|
|
|
|
|
|
background: rgba(16, 185, 129, 0.1);
|
|
|
|
|
|
color: var(--vp-c-green-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-arrow {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.15rem;
|
|
|
|
|
|
opacity: 0.3;
|
|
|
|
|
|
transition: opacity 0.5s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-arrow.visible {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.arrow-text {
|
|
|
|
|
|
font-size: 0.7rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.arrow-icon {
|
|
|
|
|
|
font-size: 1.2rem;
|
|
|
|
|
|
color: var(--vp-c-brand);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-analogy {
|
|
|
|
|
|
margin-top: 0.75rem;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.5rem 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Auth Section */
|
|
|
|
|
|
.auth-controls {
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-btn {
|
|
|
|
|
|
padding: 0.4rem 1rem;
|
|
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
border: none;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.action-btn:disabled {
|
|
|
|
|
|
opacity: 0.6;
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.auth-flow {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.auth-parties {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: 100px 1fr 100px;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
align-items: start;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.party {
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
padding: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.party-icon {
|
|
|
|
|
|
font-size: 1.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.party-name {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
margin: 0.15rem 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.party-has {
|
|
|
|
|
|
font-size: 0.72rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.auth-messages {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
gap: 0.35rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg {
|
|
|
|
|
|
padding: 0.35rem 0.5rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
|
opacity: 0.2;
|
|
|
|
|
|
transition: all 0.4s;
|
|
|
|
|
|
border: 1px solid transparent;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg.active {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg-right {
|
|
|
|
|
|
background: rgba(59, 130, 246, 0.08);
|
|
|
|
|
|
border-color: rgba(59, 130, 246, 0.2);
|
|
|
|
|
|
margin-right: 20%;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg-left {
|
|
|
|
|
|
background: rgba(16, 185, 129, 0.08);
|
|
|
|
|
|
border-color: rgba(16, 185, 129, 0.2);
|
|
|
|
|
|
margin-left: 20%;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg-result {
|
|
|
|
|
|
background: rgba(16, 185, 129, 0.15);
|
|
|
|
|
|
border-color: rgba(16, 185, 129, 0.3);
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg-label {
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 0.78rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.msg-detail {
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
margin-top: 0.1rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Uses Section */
|
|
|
|
|
|
.uses-grid {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.use-card {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.5rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.use-icon {
|
|
|
|
|
|
font-size: 1.2rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.use-name {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
margin: 0.15rem 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.use-cmd code {
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
font-size: 0.7rem;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.15rem 0.3rem;
|
|
|
|
|
|
border-radius: 3px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.use-desc {
|
|
|
|
|
|
font-size: 0.72rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
margin-top: 0.2rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.config-tips {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tip-title {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
margin-bottom: 0.35rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tip-code {
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.5rem;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
margin: 0 0 0.35rem 0;
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
overflow-x: auto;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tip-code code {
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tip-result {
|
|
|
|
|
|
font-size: 0.82rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.tip-result code {
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.1rem 0.3rem;
|
|
|
|
|
|
border-radius: 3px;
|
|
|
|
|
|
font-size: 0.78rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Info Box */
|
|
|
|
|
|
.info-box {
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
margin-top: 0.75rem;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
gap: 0.25rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.info-box strong {
|
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
flex-shrink: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@media (max-width: 640px) {
|
|
|
|
|
|
.compare-grid {
|
|
|
|
|
|
grid-template-columns: 1fr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-pair {
|
|
|
|
|
|
grid-template-columns: 1fr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.key-arrow {
|
|
|
|
|
|
flex-direction: row;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.auth-parties {
|
|
|
|
|
|
grid-template-columns: 1fr;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.party {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
text-align: left;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|