mirror of
https://github.com/amnezia-vpn/amneziawg-go.git
synced 2025-07-20 10:12:50 +02:00
691 lines
19 KiB
Go
691 lines
19 KiB
Go
/* SPDX-License-Identifier: MIT
|
|
*
|
|
* Copyright (C) 2017-2025 WireGuard LLC. All Rights Reserved.
|
|
*/
|
|
|
|
package device
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net"
|
|
"net/netip"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/amnezia-vpn/amneziawg-go/device/awg"
|
|
"github.com/amnezia-vpn/amneziawg-go/ipc"
|
|
)
|
|
|
|
type IPCError struct {
|
|
code int64 // error code
|
|
err error // underlying/wrapped error
|
|
}
|
|
|
|
func (s IPCError) Error() string {
|
|
return fmt.Sprintf("IPC error %d: %v", s.code, s.err)
|
|
}
|
|
|
|
func (s IPCError) Unwrap() error {
|
|
return s.err
|
|
}
|
|
|
|
func (s IPCError) ErrorCode() int64 {
|
|
return s.code
|
|
}
|
|
|
|
func ipcErrorf(code int64, msg string, args ...any) *IPCError {
|
|
return &IPCError{code: code, err: fmt.Errorf(msg, args...)}
|
|
}
|
|
|
|
var byteBufferPool = &sync.Pool{
|
|
New: func() any { return new(bytes.Buffer) },
|
|
}
|
|
|
|
// IpcGetOperation implements the WireGuard configuration protocol "get" operation.
|
|
// See https://www.wireguard.com/xplatform/#configuration-protocol for details.
|
|
func (device *Device) IpcGetOperation(w io.Writer) error {
|
|
device.ipcMutex.RLock()
|
|
defer device.ipcMutex.RUnlock()
|
|
|
|
buf := byteBufferPool.Get().(*bytes.Buffer)
|
|
buf.Reset()
|
|
defer byteBufferPool.Put(buf)
|
|
sendf := func(format string, args ...any) {
|
|
fmt.Fprintf(buf, format, args...)
|
|
buf.WriteByte('\n')
|
|
}
|
|
keyf := func(prefix string, key *[32]byte) {
|
|
buf.Grow(len(key)*2 + 2 + len(prefix))
|
|
buf.WriteString(prefix)
|
|
buf.WriteByte('=')
|
|
const hex = "0123456789abcdef"
|
|
for i := 0; i < len(key); i++ {
|
|
buf.WriteByte(hex[key[i]>>4])
|
|
buf.WriteByte(hex[key[i]&0xf])
|
|
}
|
|
buf.WriteByte('\n')
|
|
}
|
|
|
|
func() {
|
|
// lock required resources
|
|
|
|
device.net.RLock()
|
|
defer device.net.RUnlock()
|
|
|
|
device.staticIdentity.RLock()
|
|
defer device.staticIdentity.RUnlock()
|
|
|
|
device.peers.RLock()
|
|
defer device.peers.RUnlock()
|
|
|
|
// serialize device related values
|
|
|
|
if !device.staticIdentity.privateKey.IsZero() {
|
|
keyf("private_key", (*[32]byte)(&device.staticIdentity.privateKey))
|
|
}
|
|
|
|
if device.net.port != 0 {
|
|
sendf("listen_port=%d", device.net.port)
|
|
}
|
|
|
|
if device.net.fwmark != 0 {
|
|
sendf("fwmark=%d", device.net.fwmark)
|
|
}
|
|
|
|
if device.isAWG() {
|
|
if device.awg.ASecCfg.JunkPacketCount != 0 {
|
|
sendf("jc=%d", device.awg.ASecCfg.JunkPacketCount)
|
|
}
|
|
if device.awg.ASecCfg.JunkPacketMinSize != 0 {
|
|
sendf("jmin=%d", device.awg.ASecCfg.JunkPacketMinSize)
|
|
}
|
|
if device.awg.ASecCfg.JunkPacketMaxSize != 0 {
|
|
sendf("jmax=%d", device.awg.ASecCfg.JunkPacketMaxSize)
|
|
}
|
|
if device.awg.ASecCfg.InitHeaderJunkSize != 0 {
|
|
sendf("s1=%d", device.awg.ASecCfg.InitHeaderJunkSize)
|
|
}
|
|
if device.awg.ASecCfg.ResponseHeaderJunkSize != 0 {
|
|
sendf("s2=%d", device.awg.ASecCfg.ResponseHeaderJunkSize)
|
|
}
|
|
if device.awg.ASecCfg.CookieReplyHeaderJunkSize != 0 {
|
|
sendf("s3=%d", device.awg.ASecCfg.CookieReplyHeaderJunkSize)
|
|
}
|
|
if device.awg.ASecCfg.TransportHeaderJunkSize != 0 {
|
|
sendf("s4=%d", device.awg.ASecCfg.TransportHeaderJunkSize)
|
|
}
|
|
if device.awg.ASecCfg.InitPacketMagicHeader != 0 {
|
|
sendf("h1=%d", device.awg.ASecCfg.InitPacketMagicHeader)
|
|
}
|
|
if device.awg.ASecCfg.ResponsePacketMagicHeader != 0 {
|
|
sendf("h2=%d", device.awg.ASecCfg.ResponsePacketMagicHeader)
|
|
}
|
|
if device.awg.ASecCfg.UnderloadPacketMagicHeader != 0 {
|
|
sendf("h3=%d", device.awg.ASecCfg.UnderloadPacketMagicHeader)
|
|
}
|
|
if device.awg.ASecCfg.TransportPacketMagicHeader != 0 {
|
|
sendf("h4=%d", device.awg.ASecCfg.TransportPacketMagicHeader)
|
|
}
|
|
|
|
specialJunkIpcFields := device.awg.HandshakeHandler.SpecialJunk.IpcGetFields()
|
|
for _, field := range specialJunkIpcFields {
|
|
sendf("%s=%s", field.Key, field.Value)
|
|
}
|
|
controlledJunkIpcFields := device.awg.HandshakeHandler.ControlledJunk.IpcGetFields()
|
|
for _, field := range controlledJunkIpcFields {
|
|
sendf("%s=%s", field.Key, field.Value)
|
|
}
|
|
if device.awg.HandshakeHandler.ITimeout != 0 {
|
|
sendf("itime=%d", device.awg.HandshakeHandler.ITimeout/time.Second)
|
|
}
|
|
}
|
|
|
|
for _, peer := range device.peers.keyMap {
|
|
// Serialize peer state.
|
|
peer.handshake.mutex.RLock()
|
|
keyf("public_key", (*[32]byte)(&peer.handshake.remoteStatic))
|
|
keyf("preshared_key", (*[32]byte)(&peer.handshake.presharedKey))
|
|
peer.handshake.mutex.RUnlock()
|
|
sendf("protocol_version=1")
|
|
peer.endpoint.Lock()
|
|
if peer.endpoint.val != nil {
|
|
sendf("endpoint=%s", peer.endpoint.val.DstToString())
|
|
}
|
|
peer.endpoint.Unlock()
|
|
|
|
nano := peer.lastHandshakeNano.Load()
|
|
secs := nano / time.Second.Nanoseconds()
|
|
nano %= time.Second.Nanoseconds()
|
|
|
|
sendf("last_handshake_time_sec=%d", secs)
|
|
sendf("last_handshake_time_nsec=%d", nano)
|
|
sendf("tx_bytes=%d", peer.txBytes.Load())
|
|
sendf("rx_bytes=%d", peer.rxBytes.Load())
|
|
sendf("persistent_keepalive_interval=%d", peer.persistentKeepaliveInterval.Load())
|
|
|
|
device.allowedips.EntriesForPeer(peer, func(prefix netip.Prefix) bool {
|
|
sendf("allowed_ip=%s", prefix.String())
|
|
return true
|
|
})
|
|
}
|
|
}()
|
|
|
|
// send lines (does not require resource locks)
|
|
if _, err := w.Write(buf.Bytes()); err != nil {
|
|
return ipcErrorf(ipc.IpcErrorIO, "failed to write output: %w", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// IpcSetOperation implements the WireGuard configuration protocol "set" operation.
|
|
// See https://www.wireguard.com/xplatform/#configuration-protocol for details.
|
|
func (device *Device) IpcSetOperation(r io.Reader) (err error) {
|
|
device.ipcMutex.Lock()
|
|
defer device.ipcMutex.Unlock()
|
|
|
|
defer func() {
|
|
if err != nil {
|
|
device.log.Errorf("%v", err)
|
|
}
|
|
}()
|
|
|
|
peer := new(ipcSetPeer)
|
|
deviceConfig := true
|
|
|
|
tempAwg := awg.Protocol{}
|
|
scanner := bufio.NewScanner(r)
|
|
for scanner.Scan() {
|
|
line := scanner.Text()
|
|
if line == "" {
|
|
// Blank line means terminate operation.
|
|
err := device.handlePostConfig(&tempAwg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
peer.handlePostConfig()
|
|
return nil
|
|
}
|
|
key, value, ok := strings.Cut(line, "=")
|
|
if !ok {
|
|
return ipcErrorf(
|
|
ipc.IpcErrorProtocol,
|
|
"failed to parse line %q",
|
|
line,
|
|
)
|
|
}
|
|
|
|
if key == "public_key" {
|
|
if deviceConfig {
|
|
deviceConfig = false
|
|
}
|
|
peer.handlePostConfig()
|
|
// Load/create the peer we are now configuring.
|
|
err := device.handlePublicKeyLine(peer, value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
continue
|
|
}
|
|
|
|
var err error
|
|
if deviceConfig {
|
|
err = device.handleDeviceLine(key, value, &tempAwg)
|
|
} else {
|
|
err = device.handlePeerLine(peer, key, value)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
err = device.handlePostConfig(&tempAwg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
peer.handlePostConfig()
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
return ipcErrorf(ipc.IpcErrorIO, "failed to read input: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (device *Device) handleDeviceLine(key, value string, tempAwg *awg.Protocol) error {
|
|
switch key {
|
|
case "private_key":
|
|
var sk NoisePrivateKey
|
|
err := sk.FromMaybeZeroHex(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to set private_key: %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating private key")
|
|
device.SetPrivateKey(sk)
|
|
|
|
case "listen_port":
|
|
port, err := strconv.ParseUint(value, 10, 16)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse listen_port: %w", err)
|
|
}
|
|
|
|
// update port and rebind
|
|
device.log.Verbosef("UAPI: Updating listen port")
|
|
|
|
device.net.Lock()
|
|
device.net.port = uint16(port)
|
|
device.net.Unlock()
|
|
|
|
if err := device.BindUpdate(); err != nil {
|
|
return ipcErrorf(ipc.IpcErrorPortInUse, "failed to set listen_port: %w", err)
|
|
}
|
|
|
|
case "fwmark":
|
|
mark, err := strconv.ParseUint(value, 10, 32)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "invalid fwmark: %w", err)
|
|
}
|
|
|
|
device.log.Verbosef("UAPI: Updating fwmark")
|
|
if err := device.BindSetMark(uint32(mark)); err != nil {
|
|
return ipcErrorf(ipc.IpcErrorPortInUse, "failed to update fwmark: %w", err)
|
|
}
|
|
|
|
case "replace_peers":
|
|
if value != "true" {
|
|
return ipcErrorf(
|
|
ipc.IpcErrorInvalid,
|
|
"failed to set replace_peers, invalid value: %v",
|
|
value,
|
|
)
|
|
}
|
|
device.log.Verbosef("UAPI: Removing all peers")
|
|
device.RemoveAllPeers()
|
|
|
|
case "jc":
|
|
junkPacketCount, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse junk_packet_count %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating junk_packet_count")
|
|
tempAwg.ASecCfg.JunkPacketCount = junkPacketCount
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "jmin":
|
|
junkPacketMinSize, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse junk_packet_min_size %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating junk_packet_min_size")
|
|
tempAwg.ASecCfg.JunkPacketMinSize = junkPacketMinSize
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "jmax":
|
|
junkPacketMaxSize, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse junk_packet_max_size %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating junk_packet_max_size")
|
|
tempAwg.ASecCfg.JunkPacketMaxSize = junkPacketMaxSize
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "s1":
|
|
initPacketJunkSize, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse init_packet_junk_size %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating init_packet_junk_size")
|
|
tempAwg.ASecCfg.InitHeaderJunkSize = initPacketJunkSize
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "s2":
|
|
responsePacketJunkSize, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse response_packet_junk_size %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating response_packet_junk_size")
|
|
tempAwg.ASecCfg.ResponseHeaderJunkSize = responsePacketJunkSize
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "s3":
|
|
cookieReplyPacketJunkSize, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse cookie_reply_packet_junk_size %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating cookie_reply_packet_junk_size")
|
|
tempAwg.ASecCfg.CookieReplyHeaderJunkSize = cookieReplyPacketJunkSize
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "s4":
|
|
transportPacketJunkSize, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse transport_packet_junk_size %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating transport_packet_junk_size")
|
|
tempAwg.ASecCfg.TransportHeaderJunkSize = transportPacketJunkSize
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "h1":
|
|
initPacketMagicHeader, err := strconv.ParseUint(value, 10, 32)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse init_packet_magic_header %w", err)
|
|
}
|
|
tempAwg.ASecCfg.InitPacketMagicHeader = uint32(initPacketMagicHeader)
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "h2":
|
|
responsePacketMagicHeader, err := strconv.ParseUint(value, 10, 32)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse response_packet_magic_header %w", err)
|
|
}
|
|
tempAwg.ASecCfg.ResponsePacketMagicHeader = uint32(responsePacketMagicHeader)
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "h3":
|
|
underloadPacketMagicHeader, err := strconv.ParseUint(value, 10, 32)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse underload_packet_magic_header %w", err)
|
|
}
|
|
tempAwg.ASecCfg.UnderloadPacketMagicHeader = uint32(underloadPacketMagicHeader)
|
|
tempAwg.ASecCfg.IsSet = true
|
|
|
|
case "h4":
|
|
transportPacketMagicHeader, err := strconv.ParseUint(value, 10, 32)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse transport_packet_magic_header %w", err)
|
|
}
|
|
tempAwg.ASecCfg.TransportPacketMagicHeader = uint32(transportPacketMagicHeader)
|
|
tempAwg.ASecCfg.IsSet = true
|
|
case "i1", "i2", "i3", "i4", "i5":
|
|
if len(value) == 0 {
|
|
device.log.Verbosef("UAPI: received empty %s", key)
|
|
return nil
|
|
}
|
|
|
|
generators, err := awg.Parse(key, value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "invalid %s: %w", key, err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating %s", key)
|
|
tempAwg.HandshakeHandler.SpecialJunk.AppendGenerator(generators)
|
|
tempAwg.HandshakeHandler.IsSet = true
|
|
case "j1", "j2", "j3":
|
|
if len(value) == 0 {
|
|
device.log.Verbosef("UAPI: received empty %s", key)
|
|
return nil
|
|
}
|
|
|
|
generators, err := awg.Parse(key, value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "invalid %s: %w", key, err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating %s", key)
|
|
|
|
tempAwg.HandshakeHandler.ControlledJunk.AppendGenerator(generators)
|
|
tempAwg.HandshakeHandler.IsSet = true
|
|
case "itime":
|
|
if len(value) == 0 {
|
|
device.log.Verbosef("UAPI: received empty itime")
|
|
return nil
|
|
}
|
|
|
|
itime, err := strconv.ParseInt(value, 10, 64)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "parse itime %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Updating itime")
|
|
|
|
tempAwg.HandshakeHandler.ITimeout = time.Duration(itime) * time.Second
|
|
tempAwg.HandshakeHandler.IsSet = true
|
|
default:
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "invalid UAPI device key: %v", key)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// An ipcSetPeer is the current state of an IPC set operation on a peer.
|
|
type ipcSetPeer struct {
|
|
*Peer // Peer is the current peer being operated on
|
|
dummy bool // dummy reports whether this peer is a temporary, placeholder peer
|
|
created bool // new reports whether this is a newly created peer
|
|
pkaOn bool // pkaOn reports whether the peer had the persistent keepalive turn on
|
|
}
|
|
|
|
func (peer *ipcSetPeer) handlePostConfig() {
|
|
if peer.Peer == nil || peer.dummy {
|
|
return
|
|
}
|
|
if peer.created {
|
|
peer.endpoint.disableRoaming = peer.device.net.brokenRoaming && peer.endpoint.val != nil
|
|
}
|
|
if peer.device.isUp() {
|
|
peer.Start()
|
|
if peer.pkaOn {
|
|
peer.SendKeepalive()
|
|
}
|
|
peer.SendStagedPackets()
|
|
}
|
|
}
|
|
|
|
func (device *Device) handlePublicKeyLine(
|
|
peer *ipcSetPeer,
|
|
value string,
|
|
) error {
|
|
// Load/create the peer we are configuring.
|
|
var publicKey NoisePublicKey
|
|
err := publicKey.FromHex(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to get peer by public key: %w", err)
|
|
}
|
|
|
|
// Ignore peer with the same public key as this device.
|
|
device.staticIdentity.RLock()
|
|
peer.dummy = device.staticIdentity.publicKey.Equals(publicKey)
|
|
device.staticIdentity.RUnlock()
|
|
|
|
if peer.dummy {
|
|
peer.Peer = &Peer{}
|
|
} else {
|
|
peer.Peer = device.LookupPeer(publicKey)
|
|
}
|
|
|
|
peer.created = peer.Peer == nil
|
|
if peer.created {
|
|
peer.Peer, err = device.NewPeer(publicKey)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to create new peer: %w", err)
|
|
}
|
|
device.log.Verbosef("%v - UAPI: Created", peer.Peer)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (device *Device) handlePeerLine(
|
|
peer *ipcSetPeer,
|
|
key, value string,
|
|
) error {
|
|
switch key {
|
|
case "update_only":
|
|
// allow disabling of creation
|
|
if value != "true" {
|
|
return ipcErrorf(
|
|
ipc.IpcErrorInvalid,
|
|
"failed to set update only, invalid value: %v",
|
|
value,
|
|
)
|
|
}
|
|
if peer.created && !peer.dummy {
|
|
device.RemovePeer(peer.handshake.remoteStatic)
|
|
peer.Peer = &Peer{}
|
|
peer.dummy = true
|
|
}
|
|
|
|
case "remove":
|
|
// remove currently selected peer from device
|
|
if value != "true" {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to set remove, invalid value: %v", value)
|
|
}
|
|
if !peer.dummy {
|
|
device.log.Verbosef("%v - UAPI: Removing", peer.Peer)
|
|
device.RemovePeer(peer.handshake.remoteStatic)
|
|
}
|
|
peer.Peer = &Peer{}
|
|
peer.dummy = true
|
|
|
|
case "preshared_key":
|
|
device.log.Verbosef("%v - UAPI: Updating preshared key", peer.Peer)
|
|
|
|
peer.handshake.mutex.Lock()
|
|
err := peer.handshake.presharedKey.FromHex(value)
|
|
peer.handshake.mutex.Unlock()
|
|
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to set preshared key: %w", err)
|
|
}
|
|
|
|
case "endpoint":
|
|
device.log.Verbosef("%v - UAPI: Updating endpoint", peer.Peer)
|
|
endpoint, err := device.net.bind.ParseEndpoint(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to set endpoint %v: %w", value, err)
|
|
}
|
|
peer.endpoint.Lock()
|
|
defer peer.endpoint.Unlock()
|
|
peer.endpoint.val = endpoint
|
|
|
|
case "persistent_keepalive_interval":
|
|
device.log.Verbosef("%v - UAPI: Updating persistent keepalive interval", peer.Peer)
|
|
|
|
secs, err := strconv.ParseUint(value, 10, 16)
|
|
if err != nil {
|
|
return ipcErrorf(
|
|
ipc.IpcErrorInvalid,
|
|
"failed to set persistent keepalive interval: %w",
|
|
err,
|
|
)
|
|
}
|
|
|
|
old := peer.persistentKeepaliveInterval.Swap(uint32(secs))
|
|
|
|
// Send immediate keepalive if we're turning it on and before it wasn't on.
|
|
peer.pkaOn = old == 0 && secs != 0
|
|
|
|
case "replace_allowed_ips":
|
|
device.log.Verbosef("%v - UAPI: Removing all allowedips", peer.Peer)
|
|
if value != "true" {
|
|
return ipcErrorf(
|
|
ipc.IpcErrorInvalid,
|
|
"failed to replace allowedips, invalid value: %v",
|
|
value,
|
|
)
|
|
}
|
|
if peer.dummy {
|
|
return nil
|
|
}
|
|
device.allowedips.RemoveByPeer(peer.Peer)
|
|
|
|
case "allowed_ip":
|
|
add := true
|
|
verb := "Adding"
|
|
if len(value) > 0 && value[0] == '-' {
|
|
add = false
|
|
verb = "Removing"
|
|
value = value[1:]
|
|
}
|
|
device.log.Verbosef("%v - UAPI: %s allowedip", peer.Peer, verb)
|
|
prefix, err := netip.ParsePrefix(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to set allowed ip: %w", err)
|
|
}
|
|
if peer.dummy {
|
|
return nil
|
|
}
|
|
if add {
|
|
device.allowedips.Insert(prefix, peer.Peer)
|
|
} else {
|
|
device.allowedips.Remove(prefix, peer.Peer)
|
|
}
|
|
|
|
case "protocol_version":
|
|
if value != "1" {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "invalid protocol version: %v", value)
|
|
}
|
|
|
|
default:
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "invalid UAPI peer key: %v", key)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (device *Device) IpcGet() (string, error) {
|
|
buf := new(strings.Builder)
|
|
if err := device.IpcGetOperation(buf); err != nil {
|
|
return "", err
|
|
}
|
|
return buf.String(), nil
|
|
}
|
|
|
|
func (device *Device) IpcSet(uapiConf string) error {
|
|
return device.IpcSetOperation(strings.NewReader(uapiConf))
|
|
}
|
|
|
|
func (device *Device) IpcHandle(socket net.Conn) {
|
|
defer socket.Close()
|
|
|
|
buffered := func(s io.ReadWriter) *bufio.ReadWriter {
|
|
reader := bufio.NewReader(s)
|
|
writer := bufio.NewWriter(s)
|
|
return bufio.NewReadWriter(reader, writer)
|
|
}(socket)
|
|
|
|
for {
|
|
op, err := buffered.ReadString('\n')
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// handle operation
|
|
switch op {
|
|
case "set=1\n":
|
|
err = device.IpcSetOperation(buffered.Reader)
|
|
case "get=1\n":
|
|
var nextByte byte
|
|
nextByte, err = buffered.ReadByte()
|
|
if err != nil {
|
|
return
|
|
}
|
|
if nextByte != '\n' {
|
|
err = ipcErrorf(
|
|
ipc.IpcErrorInvalid,
|
|
"trailing character in UAPI get: %q",
|
|
nextByte,
|
|
)
|
|
break
|
|
}
|
|
err = device.IpcGetOperation(buffered.Writer)
|
|
default:
|
|
device.log.Errorf("invalid UAPI operation: %v", op)
|
|
return
|
|
}
|
|
|
|
// write status
|
|
var status *IPCError
|
|
if err != nil && !errors.As(err, &status) {
|
|
// shouldn't happen
|
|
status = ipcErrorf(ipc.IpcErrorUnknown, "other UAPI error: %w", err)
|
|
}
|
|
if status != nil {
|
|
device.log.Errorf("%v", status)
|
|
fmt.Fprintf(buffered, "errno=%d\n\n", status.ErrorCode())
|
|
} else {
|
|
fmt.Fprintf(buffered, "errno=0\n\n")
|
|
}
|
|
buffered.Flush()
|
|
}
|
|
}
|