430 lines
9.0 KiB
Vue
430 lines
9.0 KiB
Vue
<template>
|
||
<div class="dns-lookup-demo">
|
||
<div class="domain-input">
|
||
<label>输入域名</label>
|
||
<input
|
||
type="text"
|
||
v-model="domain"
|
||
placeholder="例如: www.google.com"
|
||
class="input-field"
|
||
@keyup.enter="startLookup"
|
||
/>
|
||
<button class="lookup-btn" @click="startLookup">
|
||
🔍 开始解析
|
||
</button>
|
||
</div>
|
||
|
||
<div class="lookup-process" v-if="isLooking">
|
||
<div class="process-title">DNS 解析过程</div>
|
||
|
||
<div class="step-list">
|
||
<div
|
||
v-for="(step, index) in steps"
|
||
:key="index"
|
||
class="step-item"
|
||
:class="{
|
||
active: currentStep === index,
|
||
completed: currentStep > index
|
||
}"
|
||
>
|
||
<div class="step-icon">
|
||
{{ currentStep > index ? '✓' : index + 1 }}
|
||
</div>
|
||
<div class="step-content">
|
||
<div class="step-title">{{ step.title }}</div>
|
||
<div class="step-desc">{{ step.desc }}</div>
|
||
<div v-if="currentStep === index" class="step-animation">
|
||
{{ step.animation }}
|
||
</div>
|
||
</div>
|
||
<div class="step-arrow" v-if="index < steps.length - 1">
|
||
↓
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="result-box" v-if="completed">
|
||
<div class="result-title">✅ 解析完成</div>
|
||
<div class="result-content">
|
||
<div class="result-item">
|
||
<span class="label">域名:</span>
|
||
<span class="value">{{ domain }}</span>
|
||
</div>
|
||
<div class="result-item">
|
||
<span class="label">IP 地址:</span>
|
||
<span class="value">{{ resolvedIP }}</span>
|
||
</div>
|
||
<div class="result-item">
|
||
<span class="label">解析时间:</span>
|
||
<span class="value">{{ lookupTime }}ms</span>
|
||
</div>
|
||
</div>
|
||
<button class="reset-btn" @click="reset">
|
||
🔄 重新解析
|
||
</button>
|
||
</div>
|
||
|
||
<div class="info-box">
|
||
<div class="info-title">💡 DNS 知识点</div>
|
||
<div class="info-content">
|
||
<div class="info-item">
|
||
<strong>什么是 DNS?</strong>
|
||
<br>
|
||
DNS(域名系统)就像互联网的电话簿,将易记的域名(如 google.com)转换为计算机能识别的 IP 地址(如 142.250.185.238)。
|
||
</div>
|
||
<div class="info-item">
|
||
<strong>为什么需要 DNS?</strong>
|
||
<br>
|
||
• IP 地址难记:142.250.185.238 vs google.com
|
||
<br>
|
||
• IP 可能变化:服务器迁移时 IP 会变,域名不变
|
||
<br>
|
||
• 负载均衡:一个域名可以对应多个 IP
|
||
</div>
|
||
<div class="info-item">
|
||
<strong>DNS 解析的层次</strong>
|
||
<br>
|
||
1️⃣ 浏览器缓存:最近访问过的域名
|
||
<br>
|
||
2️⃣ 系统缓存:操作系统的 DNS 缓存
|
||
<br>
|
||
3️⃣ 路由器缓存:本地路由器的缓存
|
||
<br>
|
||
4️⃣ ISP DNS:网络服务商的 DNS 服务器
|
||
<br>
|
||
5️⃣ 根域名服务器:最高层级的 DNS
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref } from 'vue'
|
||
|
||
const domain = ref('www.google.com')
|
||
const isLooking = ref(false)
|
||
const currentStep = ref(-1)
|
||
const completed = ref(false)
|
||
const resolvedIP = ref('')
|
||
const lookupTime = ref(0)
|
||
|
||
const steps = [
|
||
{
|
||
title: '检查浏览器缓存',
|
||
desc: '查看最近是否访问过该域名',
|
||
animation: '🔍 正在搜索浏览器缓存...'
|
||
},
|
||
{
|
||
title: '检查系统缓存',
|
||
desc: '查看操作系统的 DNS 缓存',
|
||
animation: '💻 正在查询系统 DNS 缓存...'
|
||
},
|
||
{
|
||
title: '查询路由器 DNS',
|
||
desc: '向本地路由器发送 DNS 查询',
|
||
animation: '📡 正在向路由器发送查询...'
|
||
},
|
||
{
|
||
title: '查询 ISP DNS 服务器',
|
||
desc: '向网络服务商的 DNS 服务器查询',
|
||
animation: '🌐 正在联系 ISP DNS 服务器...'
|
||
},
|
||
{
|
||
title: '查询根域名服务器',
|
||
desc: '从 . 根服务器开始递归查询',
|
||
animation: '🔝 正在查询根域名服务器...'
|
||
},
|
||
{
|
||
title: '获取 IP 地址',
|
||
desc: '成功解析到 IP 地址',
|
||
animation: '✅ 找到 IP 地址!'
|
||
}
|
||
]
|
||
|
||
const ipAddresses = {
|
||
'www.google.com': '142.250.185.238',
|
||
'www.baidu.com': '110.242.68.4',
|
||
'www.github.com': '140.82.112.3',
|
||
'default': '93.184.216.34'
|
||
}
|
||
|
||
const startLookup = () => {
|
||
isLooking.value = true
|
||
completed.value = false
|
||
currentStep.value = -1
|
||
const startTime = Date.now()
|
||
|
||
// 模拟 DNS 查询过程
|
||
let stepIndex = 0
|
||
const interval = setInterval(() => {
|
||
if (stepIndex < steps.length) {
|
||
currentStep.value = stepIndex
|
||
stepIndex++
|
||
} else {
|
||
clearInterval(interval)
|
||
const endTime = Date.now()
|
||
lookupTime.value = endTime - startTime
|
||
resolvedIP.value = ipAddresses[domain.value.toLowerCase()] || ipAddresses['default']
|
||
completed.value = true
|
||
}
|
||
}, 800)
|
||
}
|
||
|
||
const reset = () => {
|
||
isLooking.value = false
|
||
currentStep.value = -1
|
||
completed.value = false
|
||
}
|
||
</script>
|
||
|
||
<style scoped>
|
||
.dns-lookup-demo {
|
||
border: 1px solid var(--vp-c-divider);
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
background: var(--vp-c-bg-soft);
|
||
margin: 20px 0;
|
||
}
|
||
|
||
.domain-input {
|
||
background: var(--vp-c-bg);
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
margin-bottom: 20px;
|
||
display: flex;
|
||
gap: 10px;
|
||
align-items: flex-end;
|
||
flex-wrap: wrap;
|
||
}
|
||
|
||
.domain-input label {
|
||
width: 100%;
|
||
font-size: 0.85rem;
|
||
font-weight: 600;
|
||
color: var(--vp-c-text-2);
|
||
margin-bottom: 8px;
|
||
}
|
||
|
||
.input-field {
|
||
flex: 1;
|
||
min-width: 200px;
|
||
padding: 12px;
|
||
border: 2px solid var(--vp-c-divider);
|
||
border-radius: 6px;
|
||
font-size: 0.9rem;
|
||
background: var(--vp-c-bg-soft);
|
||
color: var(--vp-c-text-1);
|
||
}
|
||
|
||
.input-field:focus {
|
||
outline: none;
|
||
border-color: var(--vp-c-brand);
|
||
}
|
||
|
||
.lookup-btn {
|
||
padding: 12px 24px;
|
||
background: var(--vp-c-brand);
|
||
color: white;
|
||
border: none;
|
||
border-radius: 6px;
|
||
font-size: 0.9rem;
|
||
font-weight: 600;
|
||
cursor: pointer;
|
||
transition: background 0.2s;
|
||
}
|
||
|
||
.lookup-btn:hover {
|
||
background: var(--vp-c-brand-dark);
|
||
}
|
||
|
||
.lookup-process {
|
||
background: var(--vp-c-bg);
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
margin-bottom: 20px;
|
||
}
|
||
|
||
.process-title {
|
||
font-size: 1rem;
|
||
font-weight: bold;
|
||
color: var(--vp-c-text-1);
|
||
margin-bottom: 20px;
|
||
text-align: center;
|
||
}
|
||
|
||
.step-list {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 0;
|
||
}
|
||
|
||
.step-item {
|
||
display: flex;
|
||
align-items: flex-start;
|
||
gap: 15px;
|
||
position: relative;
|
||
opacity: 0.3;
|
||
transition: opacity 0.3s;
|
||
}
|
||
|
||
.step-item.active {
|
||
opacity: 1;
|
||
}
|
||
|
||
.step-item.completed {
|
||
opacity: 0.7;
|
||
}
|
||
|
||
.step-icon {
|
||
width: 40px;
|
||
height: 40px;
|
||
border-radius: 50%;
|
||
background: var(--vp-c-divider);
|
||
color: var(--vp-c-text-3);
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
font-weight: bold;
|
||
font-size: 0.9rem;
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.step-item.active .step-icon {
|
||
background: var(--vp-c-brand);
|
||
color: white;
|
||
}
|
||
|
||
.step-item.completed .step-icon {
|
||
background: #22c55e;
|
||
color: white;
|
||
}
|
||
|
||
.step-content {
|
||
flex: 1;
|
||
padding-top: 5px;
|
||
}
|
||
|
||
.step-title {
|
||
font-size: 0.95rem;
|
||
font-weight: bold;
|
||
color: var(--vp-c-text-1);
|
||
margin-bottom: 4px;
|
||
}
|
||
|
||
.step-desc {
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-3);
|
||
margin-bottom: 6px;
|
||
}
|
||
|
||
.step-animation {
|
||
font-size: 0.8rem;
|
||
color: var(--vp-c-brand);
|
||
background: var(--vp-c-bg-soft);
|
||
padding: 8px;
|
||
border-radius: 4px;
|
||
font-family: monospace;
|
||
}
|
||
|
||
.step-arrow {
|
||
position: absolute;
|
||
left: 20px;
|
||
top: 40px;
|
||
width: 2px;
|
||
height: calc(100% - 20px);
|
||
background: var(--vp-c-divider);
|
||
}
|
||
|
||
.step-item.completed .step-arrow {
|
||
background: #22c55e;
|
||
}
|
||
|
||
.result-box {
|
||
background: var(--vp-c-bg);
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
margin-bottom: 20px;
|
||
border-left: 4px solid #22c55e;
|
||
}
|
||
|
||
.result-title {
|
||
font-size: 1rem;
|
||
font-weight: bold;
|
||
color: var(--vp-c-text-1);
|
||
margin-bottom: 15px;
|
||
}
|
||
|
||
.result-content {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 10px;
|
||
margin-bottom: 15px;
|
||
}
|
||
|
||
.result-item {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
padding: 10px;
|
||
background: var(--vp-c-bg-soft);
|
||
border-radius: 6px;
|
||
}
|
||
|
||
.result-item .label {
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-3);
|
||
font-weight: 600;
|
||
}
|
||
|
||
.result-item .value {
|
||
font-size: 0.9rem;
|
||
color: var(--vp-c-brand);
|
||
font-family: monospace;
|
||
font-weight: 600;
|
||
}
|
||
|
||
.reset-btn {
|
||
width: 100%;
|
||
padding: 10px;
|
||
background: var(--vp-c-brand);
|
||
color: white;
|
||
border: none;
|
||
border-radius: 6px;
|
||
font-size: 0.9rem;
|
||
font-weight: 600;
|
||
cursor: pointer;
|
||
transition: background 0.2s;
|
||
}
|
||
|
||
.reset-btn:hover {
|
||
background: var(--vp-c-brand-dark);
|
||
}
|
||
|
||
.info-box {
|
||
background: var(--vp-c-bg);
|
||
border-radius: 8px;
|
||
padding: 20px;
|
||
border-left: 4px solid var(--vp-c-brand);
|
||
}
|
||
|
||
.info-title {
|
||
font-size: 1rem;
|
||
font-weight: bold;
|
||
color: var(--vp-c-text-1);
|
||
margin-bottom: 15px;
|
||
}
|
||
|
||
.info-content {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 15px;
|
||
}
|
||
|
||
.info-item {
|
||
font-size: 0.85rem;
|
||
color: var(--vp-c-text-2);
|
||
line-height: 1.8;
|
||
}
|
||
</style>
|