Files

820 lines
19 KiB
Vue
Raw Permalink Normal View History

<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>
</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">
<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">
<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>
</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>
</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>
</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>
</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...\n2048 位密钥,绝不外传)\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>