246 lines
5.0 KiB
Go
246 lines
5.0 KiB
Go
package interfaces
|
|
|
|
import (
|
|
"encoding/binary"
|
|
"fmt"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/Sudo-Ivan/reticulum-go/pkg/common"
|
|
)
|
|
|
|
const (
|
|
BITRATE_MINIMUM = 1200 // Minimum bitrate in bits/second
|
|
MODE_FULL = 0x01
|
|
|
|
// Interface modes
|
|
MODE_GATEWAY = 0x02
|
|
MODE_ACCESS_POINT = 0x03
|
|
MODE_ROAMING = 0x04
|
|
MODE_BOUNDARY = 0x05
|
|
|
|
// Interface types
|
|
TYPE_UDP = 0x01
|
|
TYPE_TCP = 0x02
|
|
|
|
PROPAGATION_RATE = 0.02 // 2% of interface bandwidth
|
|
)
|
|
|
|
type Interface interface {
|
|
GetName() string
|
|
GetType() common.InterfaceType
|
|
GetMode() common.InterfaceMode
|
|
IsOnline() bool
|
|
IsDetached() bool
|
|
IsEnabled() bool
|
|
Detach()
|
|
Enable()
|
|
Disable()
|
|
Send(data []byte, addr string) error
|
|
SetPacketCallback(common.PacketCallback)
|
|
GetPacketCallback() common.PacketCallback
|
|
ProcessIncoming([]byte)
|
|
ProcessOutgoing([]byte) error
|
|
SendPathRequest([]byte) error
|
|
SendLinkPacket([]byte, []byte, time.Time) error
|
|
Start() error
|
|
Stop() error
|
|
GetMTU() int
|
|
GetConn() net.Conn
|
|
common.NetworkInterface
|
|
}
|
|
|
|
type BaseInterface struct {
|
|
Name string
|
|
Mode common.InterfaceMode
|
|
Type common.InterfaceType
|
|
Online bool
|
|
Enabled bool
|
|
Detached bool
|
|
IN bool
|
|
OUT bool
|
|
MTU int
|
|
Bitrate int64
|
|
TxBytes uint64
|
|
RxBytes uint64
|
|
lastTx time.Time
|
|
|
|
mutex sync.RWMutex
|
|
packetCallback common.PacketCallback
|
|
}
|
|
|
|
func NewBaseInterface(name string, ifType common.InterfaceType, enabled bool) BaseInterface {
|
|
return BaseInterface{
|
|
Name: name,
|
|
Mode: common.IF_MODE_FULL,
|
|
Type: ifType,
|
|
Online: false,
|
|
Enabled: enabled,
|
|
Detached: false,
|
|
IN: false,
|
|
OUT: false,
|
|
MTU: common.DEFAULT_MTU,
|
|
Bitrate: BITRATE_MINIMUM,
|
|
lastTx: time.Now(),
|
|
}
|
|
}
|
|
|
|
func (i *BaseInterface) SetPacketCallback(callback common.PacketCallback) {
|
|
i.mutex.Lock()
|
|
defer i.mutex.Unlock()
|
|
i.packetCallback = callback
|
|
}
|
|
|
|
func (i *BaseInterface) GetPacketCallback() common.PacketCallback {
|
|
i.mutex.RLock()
|
|
defer i.mutex.RUnlock()
|
|
return i.packetCallback
|
|
}
|
|
|
|
func (i *BaseInterface) ProcessIncoming(data []byte) {
|
|
i.mutex.RLock()
|
|
callback := i.packetCallback
|
|
i.mutex.RUnlock()
|
|
|
|
if callback != nil {
|
|
callback(data, i)
|
|
}
|
|
|
|
i.RxBytes += uint64(len(data))
|
|
}
|
|
|
|
func (i *BaseInterface) ProcessOutgoing(data []byte) error {
|
|
if !i.Online || i.Detached {
|
|
return fmt.Errorf("interface offline or detached")
|
|
}
|
|
i.TxBytes += uint64(len(data))
|
|
return nil
|
|
}
|
|
|
|
func (i *BaseInterface) SendPathRequest(packet []byte) error {
|
|
if !i.Online || i.Detached {
|
|
return fmt.Errorf("interface offline or detached")
|
|
}
|
|
|
|
frame := make([]byte, 0, len(packet)+1)
|
|
frame = append(frame, 0x01) // Path request type
|
|
frame = append(frame, packet...)
|
|
|
|
return i.ProcessOutgoing(frame)
|
|
}
|
|
|
|
func (i *BaseInterface) SendLinkPacket(dest []byte, data []byte, timestamp time.Time) error {
|
|
if !i.Online || i.Detached {
|
|
return fmt.Errorf("interface offline or detached")
|
|
}
|
|
|
|
frame := make([]byte, 0, len(dest)+len(data)+9)
|
|
frame = append(frame, 0x02) // Link packet type
|
|
frame = append(frame, dest...)
|
|
|
|
ts := make([]byte, 8)
|
|
binary.BigEndian.PutUint64(ts, uint64(timestamp.Unix()))
|
|
frame = append(frame, ts...)
|
|
frame = append(frame, data...)
|
|
|
|
return i.ProcessOutgoing(frame)
|
|
}
|
|
|
|
func (i *BaseInterface) Detach() {
|
|
i.mutex.Lock()
|
|
defer i.mutex.Unlock()
|
|
i.Detached = true
|
|
i.Online = false
|
|
}
|
|
|
|
func (i *BaseInterface) IsEnabled() bool {
|
|
i.mutex.RLock()
|
|
defer i.mutex.RUnlock()
|
|
return i.Enabled && i.Online && !i.Detached
|
|
}
|
|
|
|
func (i *BaseInterface) Enable() {
|
|
i.mutex.Lock()
|
|
defer i.mutex.Unlock()
|
|
i.Enabled = true
|
|
i.Online = true
|
|
}
|
|
|
|
func (i *BaseInterface) Disable() {
|
|
i.mutex.Lock()
|
|
defer i.mutex.Unlock()
|
|
i.Enabled = false
|
|
i.Online = false
|
|
}
|
|
|
|
func (i *BaseInterface) GetName() string {
|
|
return i.Name
|
|
}
|
|
|
|
func (i *BaseInterface) GetType() common.InterfaceType {
|
|
return i.Type
|
|
}
|
|
|
|
func (i *BaseInterface) GetMode() common.InterfaceMode {
|
|
return i.Mode
|
|
}
|
|
|
|
func (i *BaseInterface) GetMTU() int {
|
|
return i.MTU
|
|
}
|
|
|
|
func (i *BaseInterface) IsOnline() bool {
|
|
i.mutex.RLock()
|
|
defer i.mutex.RUnlock()
|
|
return i.Online
|
|
}
|
|
|
|
func (i *BaseInterface) IsDetached() bool {
|
|
i.mutex.RLock()
|
|
defer i.mutex.RUnlock()
|
|
return i.Detached
|
|
}
|
|
|
|
func (i *BaseInterface) Start() error {
|
|
return nil
|
|
}
|
|
|
|
func (i *BaseInterface) Stop() error {
|
|
return nil
|
|
}
|
|
|
|
func (i *BaseInterface) Send(data []byte, address string) error {
|
|
return i.ProcessOutgoing(data)
|
|
}
|
|
|
|
func (i *BaseInterface) GetConn() net.Conn {
|
|
return nil
|
|
}
|
|
|
|
func (i *BaseInterface) GetBandwidthAvailable() bool {
|
|
i.mutex.RLock()
|
|
defer i.mutex.RUnlock()
|
|
|
|
// If no transmission in last second, bandwidth is available
|
|
if time.Since(i.lastTx) > time.Second {
|
|
return true
|
|
}
|
|
|
|
// Calculate current bandwidth usage
|
|
bytesPerSec := float64(i.TxBytes) / time.Since(i.lastTx).Seconds()
|
|
currentUsage := bytesPerSec * 8 // Convert to bits/sec
|
|
|
|
// Check if usage is below threshold
|
|
maxUsage := float64(i.Bitrate) * PROPAGATION_RATE
|
|
return currentUsage < maxUsage
|
|
}
|
|
|
|
func (i *BaseInterface) updateBandwidthStats(bytes uint64) {
|
|
i.mutex.Lock()
|
|
defer i.mutex.Unlock()
|
|
|
|
i.TxBytes += bytes
|
|
i.lastTx = time.Now()
|
|
}
|