/* * SPDX-License-Identifier: AGPL-3.0-only * Copyright (c) 2022-2024, daeuniverse Organization */ 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 }