2026-01-15 20:10:19 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="tcp-handshake-demo">
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<div class="diagram">
|
|
|
|
|
|
<!-- Client Column -->
|
|
|
|
|
|
<div class="column client">
|
|
|
|
|
|
<div class="actor-icon">💻 Client</div>
|
|
|
|
|
|
<div class="state-label">{{ clientState }}</div>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<!-- Interaction Area -->
|
|
|
|
|
|
<div class="interaction-zone">
|
|
|
|
|
|
<!-- Step 1: SYN -->
|
|
|
|
|
|
<div class="packet-row" :class="{ active: step === 1, done: step > 1 }">
|
|
|
|
|
|
<button
|
|
|
|
|
|
@click="sendSyn"
|
|
|
|
|
|
:disabled="step !== 0"
|
|
|
|
|
|
class="packet-btn syn"
|
2026-01-15 20:10:19 +08:00
|
|
|
|
>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
SYN (SEQ=x) →
|
|
|
|
|
|
</button>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<!-- Step 2: SYN-ACK -->
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="packet-row reverse"
|
|
|
|
|
|
:class="{ active: step === 2, done: step > 2 }"
|
|
|
|
|
|
>
|
|
|
|
|
|
<button
|
|
|
|
|
|
@click="sendSynAck"
|
|
|
|
|
|
:disabled="step !== 1"
|
|
|
|
|
|
class="packet-btn syn-ack"
|
|
|
|
|
|
>
|
|
|
|
|
|
← SYN-ACK (ACK=x+1, SEQ=y)
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<!-- Step 3: ACK -->
|
|
|
|
|
|
<div class="packet-row" :class="{ active: step === 3, done: step > 3 }">
|
|
|
|
|
|
<button
|
|
|
|
|
|
@click="sendAck"
|
|
|
|
|
|
:disabled="step !== 2"
|
|
|
|
|
|
class="packet-btn ack"
|
|
|
|
|
|
>
|
|
|
|
|
|
ACK (ACK=y+1) →
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<!-- Server Column -->
|
|
|
|
|
|
<div class="column server">
|
|
|
|
|
|
<div class="actor-icon">🖥️ Server</div>
|
|
|
|
|
|
<div class="state-label">{{ serverState }}</div>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
<div class="status-message">
|
|
|
|
|
|
<p v-if="step === 0">点击 <strong>SYN</strong> 开始连接。</p>
|
|
|
|
|
|
<p v-if="step === 1">
|
|
|
|
|
|
服务器收到了请求,现在需要回复 <strong>SYN-ACK</strong>。
|
|
|
|
|
|
</p>
|
|
|
|
|
|
<p v-if="step === 2">
|
|
|
|
|
|
客户端收到了确认,最后发送 <strong>ACK</strong> 完成握手。
|
|
|
|
|
|
</p>
|
|
|
|
|
|
<p v-if="step === 3" class="success">🎉 连接已建立 (ESTABLISHED)!</p>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
</div>
|
2026-01-16 19:10:21 +08:00
|
|
|
|
|
|
|
|
|
|
<button v-if="step === 3" @click="reset" class="reset-btn">Reset</button>
|
2026-01-15 20:10:19 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
|
|
|
|
|
import { ref } from 'vue'
|
|
|
|
|
|
|
|
|
|
|
|
const step = ref(0)
|
2026-01-16 19:10:21 +08:00
|
|
|
|
const clientState = ref('CLOSED')
|
|
|
|
|
|
const serverState = ref('LISTEN')
|
2026-01-15 20:10:19 +08:00
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
const sendSyn = () => {
|
|
|
|
|
|
step.value = 1
|
|
|
|
|
|
clientState.value = 'SYN_SENT'
|
|
|
|
|
|
}
|
2026-01-15 20:10:19 +08:00
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
const sendSynAck = () => {
|
|
|
|
|
|
step.value = 2
|
|
|
|
|
|
serverState.value = 'SYN_RCVD'
|
|
|
|
|
|
}
|
2026-01-15 20:10:19 +08:00
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
const sendAck = () => {
|
|
|
|
|
|
step.value = 3
|
|
|
|
|
|
clientState.value = 'ESTABLISHED'
|
|
|
|
|
|
serverState.value = 'ESTABLISHED'
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const reset = () => {
|
|
|
|
|
|
step.value = 0
|
2026-01-16 19:10:21 +08:00
|
|
|
|
clientState.value = 'CLOSED'
|
|
|
|
|
|
serverState.value = 'LISTEN'
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.tcp-handshake-demo {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
border-radius: 8px;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
background-color: var(--vp-c-bg-soft);
|
|
|
|
|
|
padding: 1.5rem;
|
|
|
|
|
|
margin: 1rem 0;
|
|
|
|
|
|
font-family: var(--vp-font-family-mono);
|
|
|
|
|
|
text-align: center;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.diagram {
|
2026-01-15 20:10:19 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
margin-bottom: 2rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.column {
|
|
|
|
|
|
width: 120px;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
gap: 1rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.actor-icon {
|
|
|
|
|
|
font-size: 1.2rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.state-label {
|
|
|
|
|
|
padding: 0.5rem;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
border-radius: 4px;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
font-size: 0.8rem;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.interaction-zone {
|
2026-01-15 20:10:19 +08:00
|
|
|
|
flex: 1;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
justify-content: center;
|
|
|
|
|
|
gap: 1.5rem;
|
|
|
|
|
|
padding: 0 2rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-row {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: flex-start;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
opacity: 0.3;
|
|
|
|
|
|
transition: all 0.3s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-row.reverse {
|
|
|
|
|
|
justify-content: flex-end;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-row.active {
|
|
|
|
|
|
opacity: 1;
|
|
|
|
|
|
transform: scale(1.05);
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-row.done {
|
|
|
|
|
|
opacity: 1;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-btn {
|
|
|
|
|
|
padding: 0.5rem 1rem;
|
|
|
|
|
|
border-radius: 20px;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
border: none;
|
|
|
|
|
|
cursor: pointer;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
font-family: monospace;
|
|
|
|
|
|
font-size: 0.85rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
transition: all 0.2s;
|
2026-01-16 19:10:21 +08:00
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-btn:not(:disabled):hover {
|
|
|
|
|
|
transform: translateY(-2px);
|
|
|
|
|
|
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-btn.syn {
|
|
|
|
|
|
background: #3b82f6;
|
|
|
|
|
|
color: white;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-btn.syn-ack {
|
|
|
|
|
|
background: #f59e0b;
|
|
|
|
|
|
color: white;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-btn.ack {
|
|
|
|
|
|
background: #10b981;
|
|
|
|
|
|
color: white;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.packet-btn:disabled {
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
color: var(--vp-c-text-3);
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
border-color: transparent;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.status-message {
|
|
|
|
|
|
height: 2rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.status-message .success {
|
|
|
|
|
|
color: #10b981;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.reset-btn {
|
|
|
|
|
|
padding: 0.5rem 1.5rem;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
background: var(--vp-c-bg);
|
2026-01-16 19:10:21 +08:00
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
cursor: pointer;
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-01-16 19:10:21 +08:00
|
|
|
|
.reset-btn:hover {
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
2026-01-15 20:10:19 +08:00
|
|
|
|
}
|
|
|
|
|
|
</style>
|