2021-08-16 20:58:15 +02:00
|
|
|
// +build !windows
|
|
|
|
|
|
|
|
/* SPDX-License-Identifier: MIT
|
|
|
|
*
|
|
|
|
* Copyright (C) 2017-2021 WireGuard LLC. All Rights Reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
2021-08-20 19:32:50 +02:00
|
|
|
import (
|
2021-08-21 16:54:24 +02:00
|
|
|
"bytes"
|
2021-08-20 19:32:50 +02:00
|
|
|
"crypto/md5"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/hex"
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"os"
|
|
|
|
"os/signal"
|
|
|
|
"strconv"
|
|
|
|
"syscall"
|
2021-08-21 16:54:24 +02:00
|
|
|
"time"
|
2021-08-16 20:58:15 +02:00
|
|
|
|
2021-08-20 19:32:50 +02:00
|
|
|
"github.com/KusakabeSi/EtherGuardVPN/config"
|
|
|
|
"github.com/KusakabeSi/EtherGuardVPN/conn"
|
|
|
|
"github.com/KusakabeSi/EtherGuardVPN/device"
|
|
|
|
"github.com/KusakabeSi/EtherGuardVPN/ipc"
|
|
|
|
"github.com/KusakabeSi/EtherGuardVPN/path"
|
|
|
|
"github.com/KusakabeSi/EtherGuardVPN/tap"
|
|
|
|
yaml "gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
func printExampleSuperConf() {
|
|
|
|
sconfig := config.SuperConfig{
|
|
|
|
NodeName: "NodeSuper",
|
2021-08-24 10:43:55 +02:00
|
|
|
PrivKeyV4: "mL5IW0GuqbjgDeOJuPHBU2iJzBPNKhaNEXbIGwwYWWk=",
|
|
|
|
PrivKeyV6: "+EdOKIoBp/EvIusHDsvXhV1RJYbyN3Qr8nxlz35wl3I=",
|
2021-08-20 19:32:50 +02:00
|
|
|
ListenPort: 3000,
|
|
|
|
LogLevel: config.LoggerInfo{
|
|
|
|
LogLevel: "normal",
|
|
|
|
LogTransit: true,
|
2021-08-24 10:43:55 +02:00
|
|
|
LogControl: true,
|
2021-08-20 19:32:50 +02:00
|
|
|
},
|
2021-08-24 10:43:55 +02:00
|
|
|
RePushConfigInterval: 30,
|
2021-08-20 19:32:50 +02:00
|
|
|
Peers: []config.PeerInfo{
|
|
|
|
{
|
|
|
|
NodeID: 2,
|
|
|
|
PubKey: "NuYJ/3Ght+C4HovFq5Te/BrIazo6zwDJ8Bdu4rQCz0o=",
|
|
|
|
EndPoint: "127.0.0.1:3002",
|
|
|
|
Static: true,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
GraphRecalculateSetting: config.GraphRecalculateSetting{
|
2021-08-24 10:43:55 +02:00
|
|
|
JitterTolerance: 5,
|
|
|
|
JitterToleranceMultiplier: 1.01,
|
2021-08-20 19:32:50 +02:00
|
|
|
NodeReportTimeout: 40,
|
|
|
|
RecalculateCoolDown: 5,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
soprint, _ := yaml.Marshal(sconfig)
|
|
|
|
fmt.Print(string(soprint))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func Super(configPath string, useUAPI bool, printExample bool) (err error) {
|
|
|
|
if printExample {
|
|
|
|
printExampleSuperConf()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
var sconfig config.SuperConfig
|
|
|
|
err = readYaml(configPath, &sconfig)
|
|
|
|
if err != nil {
|
2021-08-24 10:43:55 +02:00
|
|
|
fmt.Printf("Error read config: %v\n", configPath)
|
2021-08-20 19:32:50 +02:00
|
|
|
return err
|
|
|
|
}
|
2021-08-25 13:54:13 +02:00
|
|
|
NodeName := sconfig.NodeName
|
|
|
|
if len(NodeName) > 32 {
|
|
|
|
return errors.New("Node name can't longer than 32 :" + NodeName)
|
|
|
|
}
|
|
|
|
|
2021-08-20 19:32:50 +02:00
|
|
|
var logLevel int
|
|
|
|
switch sconfig.LogLevel.LogLevel {
|
|
|
|
case "verbose", "debug":
|
|
|
|
logLevel = device.LogLevelVerbose
|
|
|
|
case "error":
|
|
|
|
logLevel = device.LogLevelError
|
|
|
|
case "silent":
|
|
|
|
logLevel = device.LogLevelSilent
|
|
|
|
}
|
|
|
|
|
|
|
|
logger4 := device.NewLogger(
|
|
|
|
logLevel,
|
2021-08-25 13:54:13 +02:00
|
|
|
fmt.Sprintf("(%s) ", NodeName+"_v4"),
|
2021-08-20 19:32:50 +02:00
|
|
|
)
|
|
|
|
logger6 := device.NewLogger(
|
|
|
|
logLevel,
|
2021-08-25 13:54:13 +02:00
|
|
|
fmt.Sprintf("(%s) ", NodeName+"_v6"),
|
2021-08-20 19:32:50 +02:00
|
|
|
)
|
|
|
|
|
2021-08-21 16:54:24 +02:00
|
|
|
http_PeerState = make(map[string]*PeerState)
|
|
|
|
http_PeerID2Map = make(map[config.Vertex]string)
|
2021-08-24 20:16:21 +02:00
|
|
|
http_PeerInfos = make(map[string]config.HTTP_Peerinfo)
|
2021-08-23 21:11:01 +02:00
|
|
|
http_HashSalt = []byte(config.RandomStr(32, "Salt generate failed"))
|
2021-08-25 10:13:53 +02:00
|
|
|
http_StatePWD = sconfig.StatePassword
|
2021-08-21 16:54:24 +02:00
|
|
|
|
2021-08-20 19:32:50 +02:00
|
|
|
super_chains := path.SUPER_Events{
|
|
|
|
Event_server_pong: make(chan path.PongMsg, 1<<5),
|
|
|
|
Event_server_register: make(chan path.RegisterMsg, 1<<5),
|
|
|
|
Event_server_NhTable_changed: make(chan struct{}, 1<<4),
|
|
|
|
}
|
|
|
|
|
|
|
|
thetap, _ := tap.CreateDummyTAP()
|
2021-08-25 13:54:13 +02:00
|
|
|
http_graph = path.NewGraph(3, true, sconfig.GraphRecalculateSetting, config.NTPinfo{}, sconfig.LogLevel.LogNTP)
|
2021-08-25 10:13:53 +02:00
|
|
|
http_device4 = device.NewDevice(thetap, config.SuperNodeMessage, conn.NewCustomBind(true, false), logger4, http_graph, true, configPath, nil, &sconfig, &super_chains)
|
|
|
|
http_device6 = device.NewDevice(thetap, config.SuperNodeMessage, conn.NewCustomBind(false, true), logger6, http_graph, true, configPath, nil, &sconfig, &super_chains)
|
2021-08-21 16:54:24 +02:00
|
|
|
defer http_device4.Close()
|
|
|
|
defer http_device6.Close()
|
2021-08-20 19:32:50 +02:00
|
|
|
var sk [32]byte
|
2021-08-24 10:43:55 +02:00
|
|
|
sk_slice, err := base64.StdEncoding.DecodeString(sconfig.PrivKeyV4)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Can't decode base64:%v\n", sconfig.PrivKeyV4)
|
|
|
|
return err
|
|
|
|
}
|
2021-08-20 19:32:50 +02:00
|
|
|
copy(sk[:], sk_slice)
|
2021-08-21 16:54:24 +02:00
|
|
|
http_device4.SetPrivateKey(sk)
|
2021-08-24 10:43:55 +02:00
|
|
|
sk_slice, err = base64.StdEncoding.DecodeString(sconfig.PrivKeyV6)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Can't decode base64:%v\n", sconfig.PrivKeyV6)
|
|
|
|
return err
|
|
|
|
}
|
2021-08-20 19:32:50 +02:00
|
|
|
copy(sk[:], sk_slice)
|
2021-08-21 16:54:24 +02:00
|
|
|
http_device6.SetPrivateKey(sk)
|
|
|
|
http_device4.IpcSet("fwmark=0\n")
|
|
|
|
http_device6.IpcSet("fwmark=0\n")
|
|
|
|
http_device4.IpcSet("listen_port=" + strconv.Itoa(sconfig.ListenPort) + "\n")
|
2021-08-24 10:43:55 +02:00
|
|
|
http_device6.IpcSet("listen_port=" + strconv.Itoa(sconfig.ListenPort) + "\n")
|
2021-08-21 16:54:24 +02:00
|
|
|
http_device4.IpcSet("replace_peers=true\n")
|
|
|
|
http_device6.IpcSet("replace_peers=true\n")
|
2021-08-20 19:32:50 +02:00
|
|
|
|
|
|
|
for _, peerconf := range sconfig.Peers {
|
2021-08-21 16:54:24 +02:00
|
|
|
var pk device.NoisePublicKey
|
|
|
|
|
|
|
|
pk_slice, err := base64.StdEncoding.DecodeString(peerconf.PubKey)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Error decode base64 ", err)
|
|
|
|
}
|
|
|
|
copy(pk[:], pk_slice)
|
2021-08-25 10:13:53 +02:00
|
|
|
if peerconf.NodeID >= config.SuperNodeMessage {
|
2021-08-20 19:32:50 +02:00
|
|
|
return errors.New(fmt.Sprintf("Invalid Node_id at peer %s\n", peerconf.PubKey))
|
|
|
|
}
|
2021-08-21 16:54:24 +02:00
|
|
|
http_PeerID2Map[peerconf.NodeID] = peerconf.PubKey
|
2021-08-24 20:16:21 +02:00
|
|
|
http_PeerInfos[peerconf.PubKey] = config.HTTP_Peerinfo{
|
2021-08-21 16:54:24 +02:00
|
|
|
NodeID: peerconf.NodeID,
|
|
|
|
PubKey: peerconf.PubKey,
|
|
|
|
PSKey: peerconf.PSKey,
|
|
|
|
Connurl: make(map[string]bool),
|
|
|
|
}
|
2021-08-24 10:43:55 +02:00
|
|
|
peer4, err := http_device4.NewPeer(pk, peerconf.NodeID)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error create peer id %v\n", peerconf.NodeID)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
peer4.StaticConn = true
|
|
|
|
peer4.ConnURL = peerconf.EndPoint
|
|
|
|
peer6, err := http_device6.NewPeer(pk, peerconf.NodeID)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error create peer id %v\n", peerconf.NodeID)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
peer6.StaticConn = true
|
|
|
|
peer6.ConnURL = peerconf.EndPoint
|
2021-08-21 16:54:24 +02:00
|
|
|
if peerconf.PSKey != "" {
|
|
|
|
var psk device.NoisePresharedKey
|
|
|
|
psk_slice, err := base64.StdEncoding.DecodeString(peerconf.PSKey)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Error decode base64 ", err)
|
|
|
|
}
|
|
|
|
copy(psk[:], psk_slice)
|
|
|
|
peer4.SetPSK(psk)
|
|
|
|
peer6.SetPSK(psk)
|
|
|
|
}
|
2021-08-20 19:32:50 +02:00
|
|
|
http_PeerState[peerconf.PubKey] = &PeerState{}
|
|
|
|
}
|
|
|
|
logger4.Verbosef("Device started")
|
|
|
|
|
|
|
|
errs := make(chan error, 1<<3)
|
|
|
|
term := make(chan os.Signal, 1)
|
|
|
|
if useUAPI {
|
2021-08-25 13:54:13 +02:00
|
|
|
uapi4, err := startUAPI(NodeName+"_v4", logger4, http_device4, errs)
|
2021-08-20 19:32:50 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer uapi4.Close()
|
2021-08-25 13:54:13 +02:00
|
|
|
uapi6, err := startUAPI(NodeName+"_v6", logger6, http_device6, errs)
|
2021-08-20 19:32:50 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer uapi6.Close()
|
|
|
|
}
|
|
|
|
signal.Notify(term, syscall.SIGTERM)
|
|
|
|
signal.Notify(term, os.Interrupt)
|
|
|
|
|
2021-08-21 16:54:24 +02:00
|
|
|
go Event_server_event_hendler(http_graph, super_chains)
|
|
|
|
go RoutinePushSettings(path.S2TD(sconfig.RePushConfigInterval))
|
|
|
|
go HttpServer(sconfig.ListenPort, "/api")
|
|
|
|
|
2021-08-20 19:32:50 +02:00
|
|
|
select {
|
|
|
|
case <-term:
|
|
|
|
case <-errs:
|
2021-08-21 16:54:24 +02:00
|
|
|
case <-http_device4.Wait():
|
|
|
|
case <-http_device6.Wait():
|
2021-08-20 19:32:50 +02:00
|
|
|
}
|
|
|
|
logger4.Verbosef("Shutting down")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-21 16:54:24 +02:00
|
|
|
func Event_server_event_hendler(graph *path.IG, events path.SUPER_Events) {
|
2021-08-20 19:32:50 +02:00
|
|
|
for {
|
2021-08-21 16:54:24 +02:00
|
|
|
select {
|
|
|
|
case reg_msg := <-events.Event_server_register:
|
2021-08-23 21:11:01 +02:00
|
|
|
copy(http_PeerState[http_PeerID2Map[reg_msg.Node_id]].NhTableState[:], reg_msg.NhStateHash[:])
|
|
|
|
copy(http_PeerState[http_PeerID2Map[reg_msg.Node_id]].PeerInfoState[:], reg_msg.PeerStateHash[:])
|
2021-08-24 20:16:21 +02:00
|
|
|
http_peerinfos.Store(reg_msg.Node_id, reg_msg.Name)
|
2021-08-21 16:54:24 +02:00
|
|
|
PubKey := http_PeerID2Map[reg_msg.Node_id]
|
|
|
|
if peer := http_device4.LookupPeerByStr(PubKey); peer != nil {
|
|
|
|
if connstr := peer.GetEndpointDstStr(); connstr != "" {
|
2021-08-24 20:16:21 +02:00
|
|
|
http_PeerInfos[PubKey].Connurl[connstr] = true
|
2021-08-21 16:54:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if peer := http_device6.LookupPeerByStr(PubKey); peer != nil {
|
|
|
|
if connstr := peer.GetEndpointDstStr(); connstr != "" {
|
2021-08-24 20:16:21 +02:00
|
|
|
http_PeerInfos[PubKey].Connurl[connstr] = true
|
2021-08-21 16:54:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
http_PeerInfoStr, _ = json.Marshal(&http_PeerInfos)
|
2021-08-23 21:11:01 +02:00
|
|
|
PeerInfo_hash_raw := md5.Sum(append(http_PeerInfoStr, http_HashSalt...))
|
2021-08-21 16:54:24 +02:00
|
|
|
PeerInfo_hash_str := hex.EncodeToString(PeerInfo_hash_raw[:])
|
|
|
|
PeerInfo_hash_str_byte := []byte(PeerInfo_hash_str)
|
|
|
|
if bytes.Equal(http_PeerInfo_hash[:], PeerInfo_hash_str_byte) == false {
|
|
|
|
copy(http_PeerInfo_hash[:], PeerInfo_hash_str_byte)
|
|
|
|
PushUpdate()
|
|
|
|
}
|
|
|
|
case <-events.Event_server_NhTable_changed:
|
2021-08-20 19:32:50 +02:00
|
|
|
NhTable := graph.GetNHTable(false)
|
|
|
|
NhTablestr, _ := json.Marshal(NhTable)
|
|
|
|
md5_hash_raw := md5.Sum(http_NhTableStr)
|
2021-08-21 16:54:24 +02:00
|
|
|
new_hash_str := hex.EncodeToString(md5_hash_raw[:])
|
|
|
|
new_hash_str_byte := []byte(new_hash_str)
|
|
|
|
copy(http_NhTable_Hash[:], new_hash_str_byte)
|
|
|
|
http_NhTableStr = NhTablestr
|
|
|
|
PushUpdate()
|
|
|
|
case pong_msg := <-events.Event_server_pong:
|
|
|
|
changed := graph.UpdateLentancy(pong_msg.Src_nodeID, pong_msg.Dst_nodeID, pong_msg.Timediff, true)
|
|
|
|
if changed {
|
|
|
|
NhTable := graph.GetNHTable(false)
|
|
|
|
NhTablestr, _ := json.Marshal(NhTable)
|
2021-08-23 21:11:01 +02:00
|
|
|
md5_hash_raw := md5.Sum(append(http_NhTableStr, http_HashSalt...))
|
2021-08-21 16:54:24 +02:00
|
|
|
new_hash_str := hex.EncodeToString(md5_hash_raw[:])
|
|
|
|
new_hash_str_byte := []byte(new_hash_str)
|
|
|
|
copy(http_NhTable_Hash[:], new_hash_str_byte)
|
2021-08-20 19:32:50 +02:00
|
|
|
http_NhTableStr = NhTablestr
|
2021-08-21 16:54:24 +02:00
|
|
|
PushNhTable()
|
2021-08-20 19:32:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-21 16:54:24 +02:00
|
|
|
func RoutinePushSettings(interval time.Duration) {
|
2021-08-20 19:32:50 +02:00
|
|
|
for {
|
2021-08-21 16:54:24 +02:00
|
|
|
time.Sleep(interval)
|
|
|
|
PushNhTable()
|
|
|
|
PushUpdate()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func PushNhTable() {
|
|
|
|
body, err := path.GetByte(path.UpdateNhTableMsg{
|
|
|
|
State_hash: http_NhTable_Hash,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Error get byte")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
buf := make([]byte, path.EgHeaderLen+len(body))
|
|
|
|
header, _ := path.NewEgHeader(buf[:path.EgHeaderLen])
|
2021-08-25 10:13:53 +02:00
|
|
|
header.SetDst(config.SuperNodeMessage)
|
2021-08-21 16:54:24 +02:00
|
|
|
header.SetPacketLength(uint16(len(body)))
|
2021-08-25 10:13:53 +02:00
|
|
|
header.SetSrc(config.SuperNodeMessage)
|
2021-08-21 16:54:24 +02:00
|
|
|
header.SetTTL(0)
|
|
|
|
header.SetUsage(path.UpdateNhTable)
|
|
|
|
copy(buf[path.EgHeaderLen:], body)
|
2021-08-25 11:18:54 +02:00
|
|
|
for pkstr, _ := range http_PeerState {
|
|
|
|
if peer := http_device4.LookupPeerByStr(pkstr); peer != nil && peer.GetEndpointDstStr() != "" {
|
|
|
|
http_device4.SendPacket(peer, buf, device.MessageTransportOffsetContent)
|
2021-08-20 19:32:50 +02:00
|
|
|
}
|
2021-08-25 11:18:54 +02:00
|
|
|
if peer := http_device6.LookupPeerByStr(pkstr); peer != nil && peer.GetEndpointDstStr() != "" {
|
|
|
|
http_device6.SendPacket(peer, buf, device.MessageTransportOffsetContent)
|
|
|
|
}
|
|
|
|
|
2021-08-21 16:54:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func PushUpdate() {
|
|
|
|
body, err := path.GetByte(path.UpdatePeerMsg{
|
|
|
|
State_hash: http_PeerInfo_hash,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println("Error get byte")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
buf := make([]byte, path.EgHeaderLen+len(body))
|
|
|
|
header, _ := path.NewEgHeader(buf[:path.EgHeaderLen])
|
2021-08-25 10:13:53 +02:00
|
|
|
header.SetDst(config.SuperNodeMessage)
|
2021-08-21 16:54:24 +02:00
|
|
|
header.SetPacketLength(uint16(len(body)))
|
2021-08-25 10:13:53 +02:00
|
|
|
header.SetSrc(config.SuperNodeMessage)
|
2021-08-21 16:54:24 +02:00
|
|
|
header.SetTTL(0)
|
|
|
|
header.SetUsage(path.UpdatePeer)
|
|
|
|
copy(buf[path.EgHeaderLen:], body)
|
|
|
|
for pkstr, peerstate := range http_PeerState {
|
|
|
|
if peerstate.PeerInfoState != http_PeerInfo_hash {
|
|
|
|
if peer := http_device4.LookupPeerByStr(pkstr); peer != nil {
|
|
|
|
http_device4.SendPacket(peer, buf, device.MessageTransportOffsetContent)
|
2021-08-20 19:32:50 +02:00
|
|
|
}
|
2021-08-21 16:54:24 +02:00
|
|
|
if peer := http_device6.LookupPeerByStr(pkstr); peer != nil {
|
|
|
|
http_device6.SendPacket(peer, buf, device.MessageTransportOffsetContent)
|
2021-08-20 19:32:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func startUAPI(interfaceName string, logger *device.Logger, the_device *device.Device, errs chan error) (net.Listener, error) {
|
|
|
|
fileUAPI, err := func() (*os.File, error) {
|
2021-08-24 10:43:55 +02:00
|
|
|
uapiFdStr := os.Getenv(ENV_EG_UAPI_FD)
|
2021-08-20 19:32:50 +02:00
|
|
|
if uapiFdStr == "" {
|
|
|
|
return ipc.UAPIOpen(interfaceName)
|
|
|
|
}
|
|
|
|
// use supplied fd
|
|
|
|
fd, err := strconv.ParseUint(uapiFdStr, 10, 32)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return os.NewFile(uintptr(fd), ""), nil
|
|
|
|
}()
|
2021-08-24 10:43:55 +02:00
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("Error create UAPI socket \n")
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-08-20 19:32:50 +02:00
|
|
|
uapi, err := ipc.UAPIListen(interfaceName, fileUAPI)
|
|
|
|
if err != nil {
|
|
|
|
logger.Errorf("Failed to listen on uapi socket: %v", err)
|
2021-08-24 10:43:55 +02:00
|
|
|
return nil, err
|
2021-08-20 19:32:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
conn, err := uapi.Accept()
|
|
|
|
if err != nil {
|
|
|
|
errs <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
go the_device.IpcHandle(conn)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
logger.Verbosef("UAPI listener started")
|
|
|
|
return uapi, err
|
2021-08-16 20:58:15 +02:00
|
|
|
}
|