mirror of
https://github.com/daeuniverse/dae.git
synced 2025-01-07 05:51:20 +07:00
105 lines
2.2 KiB
Go
105 lines
2.2 KiB
Go
/*
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
* Copyright (c) 2022-2023, daeuniverse Organization <dae@v2raya.org>
|
|
*/
|
|
|
|
package control
|
|
|
|
import (
|
|
"fmt"
|
|
"net/netip"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/daeuniverse/dae/component/sniffing"
|
|
)
|
|
|
|
const (
|
|
PacketSnifferTtl = 3 * time.Second
|
|
)
|
|
|
|
type PacketSniffer struct {
|
|
*sniffing.Sniffer
|
|
deadlineTimer *time.Timer
|
|
Mu sync.Mutex
|
|
}
|
|
|
|
// PacketSnifferPool is a full-cone udp conn pool
|
|
type PacketSnifferPool struct {
|
|
pool sync.Map
|
|
createMuMap sync.Map
|
|
}
|
|
type PacketSnifferOptions struct {
|
|
Ttl time.Duration
|
|
}
|
|
type PacketSnifferKey struct {
|
|
LAddr netip.AddrPort
|
|
RAddr netip.AddrPort
|
|
}
|
|
|
|
var DefaultPacketSnifferPool = NewPacketSnifferPool()
|
|
|
|
func NewPacketSnifferPool() *PacketSnifferPool {
|
|
return &PacketSnifferPool{}
|
|
}
|
|
|
|
func (p *PacketSnifferPool) Remove(key PacketSnifferKey, sniffer *PacketSniffer) (err error) {
|
|
if ue, ok := p.pool.LoadAndDelete(key); ok {
|
|
sniffer.Close()
|
|
if ue != sniffer {
|
|
return fmt.Errorf("target udp endpoint is not in the pool")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *PacketSnifferPool) Get(key PacketSnifferKey) *PacketSniffer {
|
|
_qs, ok := p.pool.Load(key)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return _qs.(*PacketSniffer)
|
|
}
|
|
|
|
func (p *PacketSnifferPool) GetOrCreate(key PacketSnifferKey, createOption *PacketSnifferOptions) (qs *PacketSniffer, isNew bool) {
|
|
_qs, ok := p.pool.Load(key)
|
|
begin:
|
|
if !ok {
|
|
createMu, _ := p.createMuMap.LoadOrStore(key, &sync.Mutex{})
|
|
createMu.(*sync.Mutex).Lock()
|
|
defer createMu.(*sync.Mutex).Unlock()
|
|
defer p.createMuMap.Delete(key)
|
|
_qs, ok = p.pool.Load(key)
|
|
if ok {
|
|
goto begin
|
|
}
|
|
// Create an PacketSniffer.
|
|
if createOption == nil {
|
|
createOption = &PacketSnifferOptions{}
|
|
}
|
|
if createOption.Ttl == 0 {
|
|
createOption.Ttl = PacketSnifferTtl
|
|
}
|
|
|
|
qs = &PacketSniffer{
|
|
Sniffer: sniffing.NewPacketSniffer(nil, createOption.Ttl),
|
|
Mu: sync.Mutex{},
|
|
deadlineTimer: nil,
|
|
}
|
|
qs.deadlineTimer = time.AfterFunc(createOption.Ttl, func() {
|
|
if _qs, ok := p.pool.LoadAndDelete(key); ok {
|
|
if _qs.(*PacketSniffer) == qs {
|
|
qs.Close()
|
|
} else {
|
|
// FIXME: ?
|
|
}
|
|
}
|
|
})
|
|
_qs = qs
|
|
p.pool.Store(key, qs)
|
|
// Receive UDP messages.
|
|
isNew = true
|
|
}
|
|
return _qs.(*PacketSniffer), isNew
|
|
}
|