This commit is contained in:
anlicheng 2026-03-11 12:55:09 +08:00
parent d31586f0a4
commit ee913b937a
2 changed files with 44 additions and 32 deletions

View File

@ -102,6 +102,9 @@ actor SDLContextActor {
public func start() async {
self.startMonitor()
// puncher
await self.puncherActor.start()
await self.supervisor.addWorker(name: "quicClient") {
SDLLogger.shared.log("[SDLContext] try start quicClient")
let quicClient = try await self.startQUICClient()
@ -161,7 +164,7 @@ actor SDLContextActor {
await self.handleRegisterSuperNak(nakPacket: registerSuperNak)
case .peerInfo(let peerInfo):
SDLLogger.shared.log("[SDLContext] peer message: \(peerInfo)")
self.puncherActor.handlePeerInfo(using: self.udpHole, peerInfo: peerInfo)
await self.puncherActor.handlePeerInfo(using: self.udpHole, peerInfo: peerInfo)
case .event(let event):
await self.handleEvent(event: event)
case .policyReponse(let policyResponse):

View File

@ -9,13 +9,14 @@ import Foundation
import NIOCore
actor SDLPuncherActor {
nonisolated private let cooldown: Duration = .seconds(5)
nonisolated private let cooldownInterval: TimeInterval = 5
// dstMac
private var coolingDown: Set<Data> = []
private var coolingDown: [Data: (UInt32, Date)] = [:]
private var pktId: UInt32 = 1
//
private var pendingRequests: [UInt32: RegisterRequest] = [:]
private var cleanupTask: Task<Void, Never>?
struct RegisterRequest {
let srcMac: Data
@ -23,23 +24,35 @@ actor SDLPuncherActor {
let networkId: UInt32
}
//
func start() {
self.cleanupTask?.cancel()
self.cleanupTask = Task {
while !Task.isCancelled {
try? await Task.sleep(for: .seconds(cooldownInterval))
self.cleanExpiredCoolingDown()
}
}
}
func submitRegisterRequest(quicClient: SDLQUICClient?, request: RegisterRequest) {
let dstMac = request.dstMac
guard let quicClient, !coolingDown.contains(dstMac) else {
guard let quicClient, coolingDown[dstMac] == nil else {
return
}
//
coolingDown.insert(dstMac)
let pktId = self.nextPacketId()
self.tryHole(using: quicClient, pktId: pktId, request: request)
coolingDown[dstMac] = (pktId, Date().addingTimeInterval(cooldownInterval))
Task {
//
try? await Task.sleep(for: cooldown)
self.endCooldown(for: dstMac)
self.removePendingRequest(for: pktId)
//
var queryInfo = SDLQueryInfo()
queryInfo.pktID = pktId
queryInfo.dstMac = request.dstMac
self.pendingRequests[pktId] = request
if let queryData = try? queryInfo.serializedData() {
quicClient.send(type: .queryInfo, data: queryData)
}
}
@ -61,25 +74,6 @@ actor SDLPuncherActor {
}
}
private func endCooldown(for key: Data) {
self.coolingDown.remove(key)
}
private func removePendingRequest(for pktId: UInt32) {
self.pendingRequests.removeValue(forKey: pktId)
}
private func tryHole(using quicClient: SDLQUICClient, pktId: UInt32, request: RegisterRequest) {
var queryInfo = SDLQueryInfo()
queryInfo.pktID = pktId
queryInfo.dstMac = request.dstMac
self.pendingRequests[pktId] = request
if let queryData = try? queryInfo.serializedData() {
quicClient.send(type: .queryInfo, data: queryData)
}
}
private func nextPacketId() -> UInt32 {
let pktId = self.pktId
self.pktId &+= 1
@ -89,4 +83,19 @@ actor SDLPuncherActor {
return pktId
}
private func cleanExpiredCoolingDown() {
let date = Date()
for (key, (pktId, expireAt)) in coolingDown {
if expireAt < date {
self.coolingDown.removeValue(forKey: key)
self.pendingRequests.removeValue(forKey: pktId)
}
}
}
deinit {
self.cleanupTask?.cancel()
self.cleanupTask = nil
}
}