2026-01-18 12:21:49 +08:00
|
|
|
|
<!--
|
|
|
|
|
|
AuthNvsAuthZDemo.vue
|
2026-01-19 11:25:10 +08:00
|
|
|
|
AuthN vs AuthZ(更可用:请求模拟器)
|
2026-01-18 12:21:49 +08:00
|
|
|
|
-->
|
|
|
|
|
|
<template>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="authn-authz-demo">
|
2026-01-18 12:21:49 +08:00
|
|
|
|
<div class="header">
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="title">🪪 AuthN vs 🛂 AuthZ:一个请求到底会经历什么?</div>
|
|
|
|
|
|
<div class="subtitle">
|
|
|
|
|
|
选择“谁在请求”与“要做什么”,看看认证/授权分别在哪一步起作用。
|
|
|
|
|
|
</div>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="grid">
|
|
|
|
|
|
<div class="card">
|
|
|
|
|
|
<div class="card-title">选择请求</div>
|
|
|
|
|
|
|
|
|
|
|
|
<label class="label">身份(AuthN:你是谁)</label>
|
|
|
|
|
|
<div class="row">
|
|
|
|
|
|
<button
|
|
|
|
|
|
v-for="u in users"
|
|
|
|
|
|
:key="u.id"
|
|
|
|
|
|
class="chip"
|
|
|
|
|
|
:class="{ active: userId === u.id }"
|
|
|
|
|
|
@click="userId = u.id"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{ u.name }}
|
|
|
|
|
|
</button>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<label class="label">操作(AuthZ:你能做什么)</label>
|
|
|
|
|
|
<div class="row">
|
|
|
|
|
|
<button
|
|
|
|
|
|
v-for="a in actions"
|
|
|
|
|
|
:key="a.id"
|
|
|
|
|
|
class="chip"
|
|
|
|
|
|
:class="{ active: actionId === a.id }"
|
|
|
|
|
|
@click="actionId = a.id"
|
|
|
|
|
|
>
|
|
|
|
|
|
{{ a.name }}
|
|
|
|
|
|
</button>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="hint">
|
|
|
|
|
|
真实系统里:认证先发生(解析
|
|
|
|
|
|
cookie/JWT),授权发生在路由/业务逻辑层(RBAC/ABAC)。
|
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">模拟结果</div>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="result">
|
|
|
|
|
|
<div class="line">
|
|
|
|
|
|
<span class="k">AuthN(认证)</span>
|
|
|
|
|
|
<span class="v" :class="authn.ok ? 'ok' : 'bad'">
|
|
|
|
|
|
{{ authn.ok ? '通过' : '失败' }}
|
|
|
|
|
|
</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="line">
|
|
|
|
|
|
<span class="k">AuthZ(授权)</span>
|
|
|
|
|
|
<span class="v" :class="authz.ok ? 'ok' : 'bad'">
|
|
|
|
|
|
{{ authz.ok ? '允许' : '拒绝' }}
|
|
|
|
|
|
</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
<div class="line">
|
|
|
|
|
|
<span class="k">HTTP</span>
|
|
|
|
|
|
<span class="v mono">{{ finalStatus }}</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
|
|
|
|
|
|
<pre class="code"><code>{{ decisionLog }}</code></pre>
|
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">关键点</div>
|
|
|
|
|
|
<ul class="list">
|
|
|
|
|
|
<li><strong>认证失败:</strong>你是谁都不确定 → 通常返回 401。</li>
|
|
|
|
|
|
<li>
|
|
|
|
|
|
<strong>认证通过但没权限:</strong>你是谁确定了,但不能做 → 通常返回
|
|
|
|
|
|
403。
|
|
|
|
|
|
</li>
|
|
|
|
|
|
<li>
|
|
|
|
|
|
<strong>授权规则要在服务端:</strong
|
|
|
|
|
|
>别相信前端的“是否显示按钮”,那只是 UX。
|
|
|
|
|
|
</li>
|
|
|
|
|
|
</ul>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
import { computed, ref } from 'vue'
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const users = [
|
|
|
|
|
|
{ id: 'anon', name: '匿名用户' },
|
|
|
|
|
|
{ id: 'user', name: '普通用户' },
|
|
|
|
|
|
{ id: 'admin', name: '管理员' }
|
2026-01-18 12:21:49 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const actions = [
|
|
|
|
|
|
{ id: 'view_profile', name: '查看个人资料(/api/me)' },
|
|
|
|
|
|
{ id: 'create_post', name: '发帖(POST /posts)' },
|
|
|
|
|
|
{ id: 'delete_user', name: '删除用户(DELETE /users/:id)' }
|
2026-01-18 12:21:49 +08:00
|
|
|
|
]
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const userId = ref('anon')
|
|
|
|
|
|
const actionId = ref('view_profile')
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const authn = computed(() => {
|
|
|
|
|
|
if (userId.value === 'anon')
|
|
|
|
|
|
return { ok: false, reason: '缺少有效凭证(cookie/JWT)' }
|
|
|
|
|
|
return { ok: true, reason: `识别为 ${userId.value}` }
|
|
|
|
|
|
})
|
2026-01-18 12:21:49 +08:00
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const authz = computed(() => {
|
|
|
|
|
|
if (!authn.value.ok)
|
|
|
|
|
|
return { ok: false, reason: '认证未通过,无法做授权判断' }
|
|
|
|
|
|
if (actionId.value === 'delete_user') {
|
|
|
|
|
|
return userId.value === 'admin'
|
|
|
|
|
|
? { ok: true, reason: 'admin 允许删除用户' }
|
|
|
|
|
|
: { ok: false, reason: '只有 admin 才能删除用户' }
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
2026-01-19 11:25:10 +08:00
|
|
|
|
return { ok: true, reason: '此操作对已登录用户开放' }
|
2026-01-18 12:21:49 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const finalStatus = computed(() => {
|
|
|
|
|
|
if (!authn.value.ok) return '401 Unauthorized'
|
|
|
|
|
|
if (!authz.value.ok) return '403 Forbidden'
|
|
|
|
|
|
return '200 OK'
|
2026-01-18 12:21:49 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
const decisionLog = computed(() => {
|
|
|
|
|
|
const lines = []
|
|
|
|
|
|
lines.push(`Request: ${actionId.value}`)
|
|
|
|
|
|
lines.push(
|
|
|
|
|
|
`AuthN: ${authn.value.ok ? 'PASS' : 'FAIL'} - ${authn.value.reason}`
|
|
|
|
|
|
)
|
|
|
|
|
|
lines.push(
|
|
|
|
|
|
`AuthZ: ${authz.value.ok ? 'ALLOW' : 'DENY'} - ${authz.value.reason}`
|
|
|
|
|
|
)
|
|
|
|
|
|
lines.push(`Result: ${finalStatus.value}`)
|
|
|
|
|
|
return lines.join('\n')
|
2026-01-18 12:21:49 +08:00
|
|
|
|
})
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.authn-authz-demo {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
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;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
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);
|
2026-01-19 11:25:10 +08:00
|
|
|
|
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;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.label {
|
|
|
|
|
|
display: block;
|
|
|
|
|
|
font-weight: 800;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
font-size: 0.875rem;
|
|
|
|
|
|
margin: 0.75rem 0 0.35rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.row {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
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
|
|
|
|
.chip {
|
|
|
|
|
|
padding: 0.4rem 0.65rem;
|
|
|
|
|
|
border-radius: 999px;
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
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
|
|
|
|
.chip.active {
|
|
|
|
|
|
border-color: rgba(var(--vp-c-brand-rgb), 0.35);
|
|
|
|
|
|
box-shadow: 0 0 0 3px rgba(var(--vp-c-brand-rgb), 0.12);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.hint {
|
|
|
|
|
|
margin-top: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
font-size: 0.9rem;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
line-height: 1.7;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.result {
|
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;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.line {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
gap: 0.75rem;
|
|
|
|
|
|
padding: 0.35rem 0;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.k {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
font-weight: 700;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.v {
|
|
|
|
|
|
font-weight: 800;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.v.ok {
|
|
|
|
|
|
color: var(--vp-c-green-1, #22c55e);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.v.bad {
|
|
|
|
|
|
color: var(--vp-c-red-1, #ef4444);
|
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
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.code {
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
padding: 0.75rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border-radius: 6px;
|
2026-01-19 11:25:10 +08:00
|
|
|
|
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;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-19 11:25:10 +08:00
|
|
|
|
.list {
|
|
|
|
|
|
margin: 0;
|
|
|
|
|
|
padding-left: 1.1rem;
|
|
|
|
|
|
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
|
|
|
|
@media (max-width: 720px) {
|
|
|
|
|
|
.grid {
|
|
|
|
|
|
grid-template-columns: 1fr;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|