mirror of
https://github.com/amnezia-vpn/amneziawg-go.git
synced 2026-06-02 06:23:45 +02:00
792 lines
19 KiB
Go
792 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/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.junk.count != 0 {
|
|
sendf("jc=%d", device.junk.count)
|
|
}
|
|
|
|
if device.junk.min != 0 {
|
|
sendf("jmin=%d", device.junk.min)
|
|
}
|
|
|
|
if device.junk.max != 0 {
|
|
sendf("jmax=%d", device.junk.max)
|
|
}
|
|
|
|
if device.paddings.init != 0 {
|
|
sendf("s1=%d", device.paddings.init)
|
|
}
|
|
|
|
if device.paddings.response != 0 {
|
|
sendf("s2=%d", device.paddings.response)
|
|
}
|
|
|
|
if device.paddings.cookie != 0 {
|
|
sendf("s3=%d", device.paddings.cookie)
|
|
}
|
|
|
|
if device.paddings.transport != 0 {
|
|
sendf("s4=%d", device.paddings.transport)
|
|
}
|
|
|
|
if device.headers.init != nil {
|
|
sendf("h1=%s", device.headers.init.GenSpec())
|
|
}
|
|
|
|
if device.headers.response != nil {
|
|
sendf("h2=%s", device.headers.response.GenSpec())
|
|
}
|
|
|
|
if device.headers.cookie != nil {
|
|
sendf("h3=%s", device.headers.cookie.GenSpec())
|
|
}
|
|
|
|
if device.headers.transport != nil {
|
|
sendf("h4=%s", device.headers.transport.GenSpec())
|
|
}
|
|
|
|
for i, ipacket := range device.ipackets {
|
|
if ipacket != nil {
|
|
sendf("i%d=%s", i+1, ipacket.Spec)
|
|
}
|
|
}
|
|
|
|
if device.dpackets.init != nil {
|
|
sendf("di=%s", device.dpackets.init.Spec)
|
|
}
|
|
|
|
if device.dpackets.response != nil {
|
|
sendf("dr=%s", device.dpackets.response.Spec)
|
|
}
|
|
|
|
if device.dpackets.cookie != nil {
|
|
sendf("dc=%s", device.dpackets.cookie.Spec)
|
|
}
|
|
|
|
if device.dpackets.transport != nil {
|
|
sendf("dt=%s", device.dpackets.transport.Spec)
|
|
}
|
|
|
|
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)
|
|
}
|
|
}()
|
|
|
|
ipcDev := new(ipcSetDevice)
|
|
peer := new(ipcSetPeer)
|
|
deviceConfig := true
|
|
|
|
scanner := bufio.NewScanner(r)
|
|
for scanner.Scan() {
|
|
line := scanner.Text()
|
|
if line == "" {
|
|
// Blank line means terminate operation.
|
|
err := ipcDev.mergeWithDevice(device)
|
|
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)
|
|
} else {
|
|
err = device.handlePeerLine(peer, key, value)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
err = ipcDev.mergeWithDevice(device)
|
|
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) 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":
|
|
jc, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse jc: %w", err)
|
|
}
|
|
if jc <= 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "jc must be a positive value")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating junk count")
|
|
device.junk.count = jc
|
|
|
|
case "jmin":
|
|
jmin, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse jmin: %w", err)
|
|
}
|
|
if jmin <= 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "jmin must be a positive value")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating junk min")
|
|
device.junk.min = jmin
|
|
|
|
case "jmax":
|
|
jmax, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse jmax: %w", err)
|
|
}
|
|
if jmax <= 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "jmax must be a positive value")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating junk max")
|
|
device.junk.max = jmax
|
|
|
|
case "s1":
|
|
padding, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse s1: %w", err)
|
|
}
|
|
if padding < 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "s1 must be non-negative")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating s1 padding")
|
|
device.paddings.init = padding
|
|
|
|
case "s2":
|
|
padding, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse s2: %w", err)
|
|
}
|
|
if padding < 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "s2 must be non-negative")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating s2 padding")
|
|
device.paddings.response = padding
|
|
|
|
case "s3":
|
|
padding, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse s3: %w", err)
|
|
}
|
|
if padding < 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "s3 must be non-negative")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating s3 padding")
|
|
device.paddings.cookie = padding
|
|
|
|
case "s4":
|
|
padding, err := strconv.Atoi(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse s4: %w", err)
|
|
}
|
|
if padding < 0 {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "s4 must be non-negative")
|
|
}
|
|
device.log.Verbosef("UAPI: Updating s4 padding")
|
|
device.paddings.transport = padding
|
|
|
|
case "h1":
|
|
header, err := newMagicHeader(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse H1: %w", err)
|
|
}
|
|
device.headers.init = header
|
|
|
|
case "h2":
|
|
header, err := newMagicHeader(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse H2: %w", err)
|
|
}
|
|
device.headers.response = header
|
|
|
|
case "h3":
|
|
header, err := newMagicHeader(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse H3: %w", err)
|
|
}
|
|
device.headers.cookie = header
|
|
|
|
case "h4":
|
|
header, err := newMagicHeader(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse H4: %w", err)
|
|
}
|
|
device.headers.transport = header
|
|
|
|
case "i1":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse I1: %w", err)
|
|
}
|
|
device.ipackets[0] = chain
|
|
|
|
case "i2":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse I2: %w", err)
|
|
}
|
|
device.ipackets[1] = chain
|
|
|
|
case "i3":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse I3: %w", err)
|
|
}
|
|
device.ipackets[2] = chain
|
|
|
|
case "i4":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse I4: %w", err)
|
|
}
|
|
device.ipackets[3] = chain
|
|
|
|
case "i5":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse I5: %w", err)
|
|
}
|
|
device.ipackets[4] = chain
|
|
|
|
case "di":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse DI: %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Set DI format")
|
|
device.dpackets.init = chain
|
|
|
|
case "dr":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse DR: %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Set DR format")
|
|
device.dpackets.response = chain
|
|
|
|
case "dc":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse DC: %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Set DC format")
|
|
device.dpackets.cookie = chain
|
|
|
|
case "dt":
|
|
chain, err := newObfChain(value)
|
|
if err != nil {
|
|
return ipcErrorf(ipc.IpcErrorInvalid, "failed to parse DT: %w", err)
|
|
}
|
|
device.log.Verbosef("UAPI: Set DT format")
|
|
device.dpackets.transport = chain
|
|
|
|
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()
|
|
}
|
|
}
|
|
|
|
type ipcSetDevice struct {
|
|
headers struct {
|
|
init *magicHeader
|
|
response *magicHeader
|
|
cookie *magicHeader
|
|
transport *magicHeader
|
|
}
|
|
}
|
|
|
|
func (d *ipcSetDevice) mergeWithDevice(device *Device) error {
|
|
if d.headers.init == nil {
|
|
d.headers.init = device.headers.init
|
|
}
|
|
|
|
if d.headers.response == nil {
|
|
d.headers.response = device.headers.response
|
|
}
|
|
|
|
if d.headers.cookie == nil {
|
|
d.headers.cookie = device.headers.cookie
|
|
}
|
|
|
|
if d.headers.transport == nil {
|
|
d.headers.transport = device.headers.transport
|
|
}
|
|
|
|
headers := []*magicHeader{d.headers.init, d.headers.response, d.headers.cookie, d.headers.transport}
|
|
for i := 0; i < len(headers); i++ {
|
|
for j := i + 1; j < len(headers); j++ {
|
|
left := headers[i]
|
|
right := headers[j]
|
|
|
|
if left.start <= right.end && right.start <= left.end {
|
|
return errors.New("headers must not overlap")
|
|
}
|
|
}
|
|
}
|
|
|
|
device.headers.init = d.headers.init
|
|
device.headers.response = d.headers.response
|
|
device.headers.cookie = d.headers.cookie
|
|
device.headers.transport = d.headers.transport
|
|
|
|
return nil
|
|
}
|