punchnet-macos/Tun/Punchnet/Actors/SDLPuncherActor.swift
2026-03-11 12:55:09 +08:00

102 lines
2.9 KiB
Swift

//
// SDLPuncherActor.swift
// Tun
//
// Created by on 2026/1/7.
//
import Foundation
import NIOCore
actor SDLPuncherActor {
nonisolated private let cooldownInterval: TimeInterval = 5
// dstMac
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
let dstMac: Data
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[dstMac] == nil else {
return
}
let pktId = self.nextPacketId()
coolingDown[dstMac] = (pktId, Date().addingTimeInterval(cooldownInterval))
//
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)
}
}
func handlePeerInfo(using udpHole: SDLUDPHole?, peerInfo: SDLPeerInfo) async {
guard let udpHole, let request = pendingRequests.removeValue(forKey: peerInfo.pktID),
let remoteAddress = try? await peerInfo.v4Info.socketAddress() else {
return
}
SDLLogger.shared.log("[SDLContext] hole sock address: \(remoteAddress)", level: .debug)
// register
var register = SDLRegister()
register.networkID = request.networkId
register.srcMac = request.srcMac
register.dstMac = request.dstMac
if let registerData = try? register.serializedData() {
udpHole.send(type: .register, data: registerData, remoteAddress: remoteAddress)
}
}
private func nextPacketId() -> UInt32 {
let pktId = self.pktId
self.pktId &+= 1
if self.pktId == 0 {
self.pktId = 1
}
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
}
}