2026-01-18 12:21:49 +08:00
|
|
|
|
<!--
|
|
|
|
|
|
JWTWorkflowDemo.vue
|
2026-01-19 11:25:10 +08:00
|
|
|
|
JWT 工作流程(手动推进,更贴近真实使用)
|
2026-01-18 12:21:49 +08:00
|
|
|
|
-->
|
|
|
|
|
|
<template>
|
|
|
|
|
|
<div class="jwt-workflow-demo">
|
|
|
|
|
|
<div class="header">
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="title">🎫 JWT:生成 → 发送 → 验证 → 解析</div>
|
|
|
|
|
|
<div class="subtitle">
|
|
|
|
|
|
默认“手动推进”,不自动下一步;避免把演示误当成真实系统的安全边界。
|
|
|
|
|
|
</div>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="controls">
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<button class="btn primary" @click="start" :disabled="step !== 0">
|
|
|
|
|
|
开始
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</button>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<button class="btn" @click="prev" :disabled="step <= 1">上一步</button>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
<button
|
2026-01-19 11:25:10 +08:00
|
|
|
|
class="btn primary"
|
|
|
|
|
|
@click="next"
|
|
|
|
|
|
:disabled="step === 0 || step >= maxStep"
|
2026-01-18 12:21:49 +08:00
|
|
|
|
>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
下一步
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</button>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<button class="btn" @click="reset">重置</button>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div v-if="step > 0" class="progress">
|
|
|
|
|
|
Step {{ step }} / {{ maxStep }} · {{ steps[step - 1]?.title }}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="grid">
|
|
|
|
|
|
<div class="card">
|
|
|
|
|
|
<div class="card-title">用户声明(Payload 示例)</div>
|
|
|
|
|
|
<pre class="code"><code>{{ payloadJson }}</code></pre>
|
|
|
|
|
|
<div class="hint">
|
|
|
|
|
|
注意:JWT 的 payload 只是 Base64Url
|
|
|
|
|
|
编码,任何人都能解码,所以不要放密码、手机号等敏感数据。
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="card">
|
|
|
|
|
|
<div class="card-title">JWT Token(示意)</div>
|
|
|
|
|
|
<div class="token">
|
|
|
|
|
|
<div class="part" :class="{ active: step >= 1 }">
|
2026-01-18 12:21:49 +08:00
|
|
|
|
<div class="part-label">Header</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<code class="mono">{{ step >= 1 ? headerB64 : '...' }}</code>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="dot">.</div>
|
|
|
|
|
|
<div class="part" :class="{ active: step >= 2 }">
|
2026-01-18 12:21:49 +08:00
|
|
|
|
<div class="part-label">Payload</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<code class="mono">{{ step >= 2 ? payloadB64 : '...' }}</code>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="dot">.</div>
|
|
|
|
|
|
<div class="part" :class="{ active: step >= 3 }">
|
2026-01-18 12:21:49 +08:00
|
|
|
|
<div class="part-label">Signature</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<code class="mono">{{ step >= 3 ? signatureB64 : '...' }}</code>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="mono-box" v-if="step >= 4">
|
|
|
|
|
|
<div class="mono-label">完整 Token</div>
|
|
|
|
|
|
<code class="mono">{{ token }}</code>
|
|
|
|
|
|
<button class="copy" @click="copy(token)">
|
|
|
|
|
|
{{ copied ? '已复制' : '复制 Token' }}
|
|
|
|
|
|
</button>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="mono-box" v-if="step >= 5">
|
|
|
|
|
|
<div class="mono-label">请求头示例</div>
|
|
|
|
|
|
<code class="mono">Authorization: Bearer {{ token }}</code>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="card">
|
|
|
|
|
|
<div class="card-title">{{ steps[step - 1]?.title || '流程说明' }}</div>
|
|
|
|
|
|
<div class="desc">{{ steps[step - 1]?.desc }}</div>
|
|
|
|
|
|
<div v-if="steps[step - 1]?.warn" class="warn">
|
|
|
|
|
|
<div class="warn-title">注意</div>
|
|
|
|
|
|
<div class="warn-text">{{ steps[step - 1]?.warn }}</div>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
import { computed, ref } from 'vue'
|
|
|
|
|
|
|
|
|
|
|
|
const maxStep = 6
|
|
|
|
|
|
const step = ref(0)
|
|
|
|
|
|
const copied = ref(false)
|
|
|
|
|
|
|
|
|
|
|
|
const headerObj = { alg: 'HS256', typ: 'JWT' }
|
|
|
|
|
|
const payloadObj = computed(() => ({
|
|
|
|
|
|
user_id: 123,
|
|
|
|
|
|
username: 'alice',
|
|
|
|
|
|
role: 'admin',
|
|
|
|
|
|
iat: Math.floor(Date.now() / 1000),
|
|
|
|
|
|
exp: Math.floor(Date.now() / 1000) + 3600
|
|
|
|
|
|
}))
|
|
|
|
|
|
|
|
|
|
|
|
const payloadJson = computed(() => JSON.stringify(payloadObj.value, null, 2))
|
|
|
|
|
|
const headerB64 = computed(() => btoa(JSON.stringify(headerObj)))
|
|
|
|
|
|
const payloadB64 = computed(() => btoa(JSON.stringify(payloadObj.value)))
|
|
|
|
|
|
const signatureB64 = computed(() =>
|
|
|
|
|
|
btoa(`${headerB64.value}.${payloadB64.value}.your-secret-key`)
|
|
|
|
|
|
)
|
|
|
|
|
|
const token = computed(
|
|
|
|
|
|
() => `${headerB64.value}.${payloadB64.value}.${signatureB64.value}`
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
const steps = [
|
|
|
|
|
|
{
|
|
|
|
|
|
title: '1) 生成 Header',
|
|
|
|
|
|
desc: 'Header 描述使用的算法与 token 类型(JWT)。'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
title: '2) 生成 Payload',
|
|
|
|
|
|
desc: 'Payload 放业务声明(claims)。它可被解码,所以不要放敏感信息。'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
title: '3) 生成 Signature',
|
|
|
|
|
|
desc: 'Signature 用密钥对 header.payload 做签名,用来防篡改。',
|
|
|
|
|
|
warn: '只有“签名校验”能保证 payload 未被改过;Base64 不是加密。'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
title: '4) 拼接 Token',
|
|
|
|
|
|
desc: '把三段用 “.” 连接:header.payload.signature。'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
title: '5) 客户端发送请求',
|
|
|
|
|
|
desc: '通常放在 Authorization: Bearer <token>。'
|
|
|
|
|
|
},
|
|
|
|
|
|
{
|
|
|
|
|
|
title: '6) 服务端验证与授权',
|
|
|
|
|
|
desc: '服务端校验签名与过期时间,再按 role/权限做授权判断。',
|
|
|
|
|
|
warn: 'JWT 无法“立刻全局注销”:常用解法是短 access token + refresh token + 黑名单/版本号。'
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
2026-01-19 11:25:10 +08:00
|
|
|
|
]
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const start = () => {
|
|
|
|
|
|
step.value = 1
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const next = () => {
|
|
|
|
|
|
step.value = Math.min(maxStep, step.value + 1)
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const prev = () => {
|
|
|
|
|
|
step.value = Math.max(1, step.value - 1)
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const reset = () => {
|
|
|
|
|
|
step.value = 0
|
|
|
|
|
|
copied.value = false
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const copy = async (text) => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
await navigator.clipboard.writeText(text)
|
|
|
|
|
|
copied.value = true
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
copied.value = false
|
|
|
|
|
|
}, 800)
|
|
|
|
|
|
} catch {
|
|
|
|
|
|
copied.value = false
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.jwt-workflow-demo {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
background: var(--vp-c-bg-soft);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
padding: 1.5rem;
|
2026-02-14 20:23:34 +08:00
|
|
|
|
margin: 0.5rem 0;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.header {
|
2026-01-19 11:25:10 +08:00
|
|
|
|
margin-bottom: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.title {
|
2026-01-19 11:25:10 +08:00
|
|
|
|
font-weight: 800;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.subtitle {
|
2026-01-19 11:25:10 +08:00
|
|
|
|
margin-top: 0.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.controls {
|
|
|
|
|
|
display: flex;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
gap: 0.5rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
flex-wrap: wrap;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.btn {
|
|
|
|
|
|
padding: 0.5rem 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
cursor: pointer;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
font-weight: 700;
|
|
|
|
|
|
font-size: 0.875rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.btn.primary {
|
|
|
|
|
|
background: var(--vp-c-brand);
|
|
|
|
|
|
border-color: var(--vp-c-brand);
|
|
|
|
|
|
color: var(--vp-c-bg);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.btn:disabled {
|
|
|
|
|
|
opacity: 0.5;
|
|
|
|
|
|
cursor: not-allowed;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.progress {
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.grid {
|
|
|
|
|
|
display: grid;
|
|
|
|
|
|
grid-template-columns: 1fr 1fr;
|
|
|
|
|
|
gap: 1rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.card {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
padding: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.card-title {
|
|
|
|
|
|
font-weight: 800;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.code {
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-01-19 11:25:10 +08:00
|
|
|
|
overflow-x: auto;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.hint {
|
|
|
|
|
|
margin-top: 0.75rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
line-height: 1.7;
|
|
|
|
|
|
font-size: 0.9rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.token {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
align-items: stretch;
|
|
|
|
|
|
gap: 0.5rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
flex-wrap: wrap;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.part {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
flex: 1;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
min-width: 220px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
padding: 0.75rem;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
opacity: 0.6;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.part.active {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
border-color: rgba(var(--vp-c-brand-rgb), 0.35);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.part-label {
|
|
|
|
|
|
font-weight: 800;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
margin-bottom: 0.35rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.dot {
|
|
|
|
|
|
display: none;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.mono {
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-size: 0.85rem;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
word-break: break-all;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.mono-box {
|
|
|
|
|
|
margin-top: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-01-19 11:25:10 +08:00
|
|
|
|
background: var(--vp-c-bg-alt);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
padding: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.mono-label {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
2026-01-19 11:25:10 +08:00
|
|
|
|
font-size: 0.8rem;
|
|
|
|
|
|
margin-bottom: 0.35rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.copy {
|
|
|
|
|
|
margin-top: 0.5rem;
|
|
|
|
|
|
padding: 0.35rem 0.6rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
background: var(--vp-c-bg);
|
2026-01-19 11:25:10 +08:00
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
cursor: pointer;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-weight: 700;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
font-size: 0.875rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.desc {
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
line-height: 1.75;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.warn {
|
|
|
|
|
|
margin-top: 0.75rem;
|
|
|
|
|
|
border: 1px solid rgba(var(--vp-c-brand-rgb), 0.18);
|
|
|
|
|
|
background: rgba(var(--vp-c-brand-rgb), 0.06);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
padding: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.warn-title {
|
|
|
|
|
|
font-weight: 800;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
margin-bottom: 0.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.warn-text {
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
line-height: 1.7;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
@media (max-width: 720px) {
|
|
|
|
|
|
.grid {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
grid-template-columns: 1fr;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|