12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451 |
- package functions
- import (
- "fmt"
- "errors"
- "context"
- "net/http"
- "io/ioutil"
- "strings"
- "log"
- "net"
- "os"
- "strconv"
- "os/exec"
- "github.com/gravitl/netmaker/netclient/config"
- nodepb "github.com/gravitl/netmaker/grpc"
- "golang.zx2c4.com/wireguard/wgctrl"
- "google.golang.org/grpc"
- "encoding/base64"
- "google.golang.org/grpc/metadata"
- "golang.zx2c4.com/wireguard/wgctrl/wgtypes"
- //homedir "github.com/mitchellh/go-homedir"
- )
- var (
- wcclient nodepb.NodeServiceClient
- )
- func ListPorts() error{
- wgclient, err := wgctrl.New()
- if err != nil {
- return err
- }
- devices, err := wgclient.Devices()
- if err != nil {
- return err
- }
- fmt.Println("Here are your ports:")
- for _, i := range devices {
- fmt.Println(i.ListenPort)
- }
- return err
- }
- func GetFreePort(rangestart int32) (int32, error){
- wgclient, err := wgctrl.New()
- if err != nil {
- return 0, err
- }
- devices, err := wgclient.Devices()
- if err != nil {
- return 0, err
- }
- var portno int32
- portno = 0
- for x := rangestart; x <= 60000; x++ {
- conflict := false
- for _, i := range devices {
- if int32(i.ListenPort) == x {
- conflict = true
- break;
- }
- }
- if conflict {
- continue
- }
- portno = x
- break
- }
- return portno, err
- }
- func Install(accesskey string, password string, server string, network string, noauto bool, accesstoken string, inputname string, pubip string, dnsoff bool, ipforward string) error {
- tserver := ""
- tnetwork := ""
- tkey := ""
- trange := ""
- var localrange *net.IPNet
- islocal := false
- if FileExists("/etc/systemd/system/netclient-"+network+".timer") ||
- FileExists("/etc/netclient/netconfig-"+network) {
- err := errors.New("ALREADY_INSTALLED. Netclient appears to already be installed for network " + network + ". To re-install, please remove by executing 'sudo netclient -c remove -n " + network + "'. Then re-run the install command.")
- return err
- }
- if accesstoken != "" && accesstoken != "badtoken" {
- btoken, err := base64.StdEncoding.DecodeString(accesstoken)
- if err != nil {
- log.Fatalf("Something went wrong decoding your token: %v", err)
- }
- token := string(btoken)
- tokenvals := strings.Split(token, "|")
- tserver = tokenvals[0]
- tnetwork = tokenvals[1]
- tkey = tokenvals[2]
- trange = tokenvals[3]
- printrange := ""
- if server == "localhost:50051" {
- server = tserver
- }
- if network == "nonetwork" {
- network = tnetwork
- }
- if accesskey == "badkey" {
- accesskey = tkey
- }
- fmt.Println(trange)
- if trange != "" {
- fmt.Println("This is a local network. Proceeding with local address as endpoint.")
- islocal = true
- _, localrange, err = net.ParseCIDR(trange)
- if err == nil {
- printrange = localrange.String()
- } else {
- //localrange = ""
- }
- } else {
- printrange = "Not a local network. Will use public address for endpoint."
- }
- fmt.Println("Decoded values from token:")
- fmt.Println(" Server: " + server)
- fmt.Println(" Network: " + network)
- fmt.Println(" Key: " + accesskey)
- fmt.Println(" Local Range: " + printrange)
- }
- wgclient, err := wgctrl.New()
- if err != nil {
- log.Fatalf("failed to open client: %v", err)
- }
- defer wgclient.Close()
- cfg, err := config.ReadConfig(network)
- if err != nil {
- log.Printf("No Config Yet. Will Write: %v", err)
- }
- nodecfg := cfg.Node
- servercfg := cfg.Server
- fmt.Println("SERVER SETTINGS:")
- nodecfg.DNSOff = dnsoff
- nodecfg.IPForwarding = ipforward
- if server == "" {
- if servercfg.Address == "" && tserver == "" {
- log.Fatal("no server provided")
- } else {
- server = servercfg.Address
- }
- }
- fmt.Println(" Server: " + server)
- if accesskey == "" {
- if servercfg.AccessKey == "" && tkey == "" {
- fmt.Println("no access key provided.Proceeding anyway.")
- } else {
- accesskey = servercfg.AccessKey
- }
- }
- fmt.Println(" AccessKey: " + accesskey)
- err = config.WriteServer(server, accesskey, network)
- if err != nil {
- fmt.Println("Error encountered while writing Server Config.")
- return err
- }
- fmt.Println("NODE REQUESTING SETTINGS:")
- if password == "" {
- if nodecfg.Password == "" {
- //create error here
- log.Fatal("no password provided")
- } else {
- password = nodecfg.Password
- }
- }
- fmt.Println(" Password: " + password)
- if network == "badnetwork" {
- if nodecfg.Network == "" && tnetwork == "" {
- //create error here
- log.Fatal("no network provided")
- } else {
- network = nodecfg.Network
- }
- }
- fmt.Println(" Network: " + network)
- var macaddress string
- var localaddress string
- var listenport int32
- var keepalive int32
- var publickey wgtypes.Key
- var privatekey wgtypes.Key
- var privkeystring string
- var endpoint string
- var postup string
- var postdown string
- var name string
- var wginterface string
- if nodecfg.LocalAddress == "" {
- ifaces, err := net.Interfaces()
- if err != nil {
- return err
- }
- var local string
- found := false
- for _, i := range ifaces {
- if i.Flags&net.FlagUp == 0 {
- continue // interface down
- }
- if i.Flags&net.FlagLoopback != 0 {
- continue // loopback interface
- }
- addrs, err := i.Addrs()
- if err != nil {
- return err
- }
- for _, addr := range addrs {
- var ip net.IP
- switch v := addr.(type) {
- case *net.IPNet:
- if !found {
- ip = v.IP
- local = ip.String()
- if islocal {
- found = localrange.Contains(ip)
- } else {
- found = true
- }
- }
- case *net.IPAddr:
- if !found {
- ip = v.IP
- local = ip.String()
- if islocal {
- found = localrange.Contains(ip)
- } else {
- found = true
- }
- }
- }
- }
- }
- localaddress = local
- } else {
- localaddress = nodecfg.LocalAddress
- }
- fmt.Println(" Local Address: " + localaddress)
- if pubip != "" && pubip != "nopubip" {
- endpoint = pubip
- } else {
- if nodecfg.Endpoint == "" {
- if islocal && localaddress != "" {
- endpoint = localaddress
- fmt.Println("Endpoint is local. Setting to address: " + endpoint)
- } else {
- endpoint, err = getPublicIP()
- if err != nil {
- fmt.Println("Error setting endpoint.")
- return err
- }
- fmt.Println("Endpoint is public. Setting to address: " + endpoint)
- }
- } else {
- endpoint = nodecfg.Endpoint
- fmt.Println("Endpoint set in config. Setting to address: " + endpoint)
- }
- }
- fmt.Println(" Endpoint: " + endpoint)
- if nodecfg.Name != "" {
- name = nodecfg.Name
- }
- if inputname != "" && inputname != "noname" {
- name = inputname
- }
- fmt.Println(" Name: " + name)
- if nodecfg.Interface != "" {
- wginterface = nodecfg.Interface
- }
- fmt.Println(" Interface: " + wginterface)
- if nodecfg.PostUp != "" {
- postup = nodecfg.PostUp
- }
- fmt.Println(" PostUp: " + postup)
- if nodecfg.PostDown!= "" {
- postdown = nodecfg.PostDown
- }
- fmt.Println(" PostDown: " + postdown)
- if nodecfg.KeepAlive != 0 {
- keepalive = nodecfg.KeepAlive
- }
- fmt.Println(" KeepAlive: " + wginterface)
- if nodecfg.Port != 0 {
- listenport = nodecfg.Port
- }
- if listenport == 0 {
- listenport, err = GetFreePort(51821)
- if err != nil {
- fmt.Printf("Error retrieving port: %v", err)
- }
- }
- fmt.Printf(" Port: %v", listenport)
- fmt.Println("")
- if nodecfg.PrivateKey != "" {
- privkeystring = nodecfg.PrivateKey
- privatekey, err := wgtypes.ParseKey(nodecfg.PrivateKey)
- if err != nil {
- log.Fatal(err)
- }
- if nodecfg.PublicKey != "" {
- publickey, err = wgtypes.ParseKey(nodecfg.PublicKey)
- if err != nil {
- log.Fatal(err)
- }
- } else {
- publickey = privatekey.PublicKey()
- }
- } else {
- privatekey, err := wgtypes.GeneratePrivateKey()
- if err != nil {
- log.Fatal(err)
- }
- privkeystring = privatekey.String()
- publickey = privatekey.PublicKey()
- }
- if nodecfg.MacAddress != "" {
- macaddress = nodecfg.MacAddress
- } else {
- macs, err := getMacAddr()
- if err != nil {
- return err
- } else if len(macs) == 0 {
- log.Fatal()
- } else {
- macaddress = macs[0]
- }
- }
- fmt.Println(" Mac Address: " + macaddress)
- fmt.Println(" Private Key: " + privatekey.String())
- fmt.Println(" Public Key: " + publickey.String())
- var wcclient nodepb.NodeServiceClient
- var requestOpts grpc.DialOption
- requestOpts = grpc.WithInsecure()
- conn, err := grpc.Dial(server, requestOpts)
- if err != nil {
- log.Fatalf("Unable to establish client connection to localhost:50051: %v", err)
- }
- wcclient = nodepb.NewNodeServiceClient(conn)
- postnode := &nodepb.Node{
- Password: password,
- Macaddress: macaddress,
- Accesskey: accesskey,
- Nodenetwork: network,
- Listenport: listenport,
- Postup: postup,
- Postdown: postdown,
- Keepalive: keepalive,
- Localaddress: localaddress,
- Interface: wginterface,
- Publickey: publickey.String(),
- Name: name,
- Endpoint: endpoint,
- }
- fmt.Println("Writing node settings to netconfig file.")
- err = modConfig(postnode)
- if err != nil {
- return err
- }
- res, err := wcclient.CreateNode(
- context.TODO(),
- &nodepb.CreateNodeReq{
- Node: postnode,
- },
- )
- if err != nil {
- return err
- }
- node := res.Node
- fmt.Println("Setting local config from server response")
- if err != nil {
- return err
- }
- fmt.Println("NODE RECIEVED SETTINGS: ")
- fmt.Println(" Password: " + node.Password)
- fmt.Println(" WG Address: " + node.Address)
- fmt.Println(" WG ipv6 Address: " + node.Address6)
- fmt.Println(" Network: " + node.Nodenetwork)
- fmt.Println(" Public Endpoint: " + node.Endpoint)
- fmt.Println(" Local Address: " + node.Localaddress)
- fmt.Println(" Name: " + node.Name)
- fmt.Println(" Interface: " + node.Interface)
- fmt.Println(" PostUp: " + node.Postup)
- fmt.Println(" PostDown: " + node.Postdown)
- fmt.Println(" Port: " + strconv.FormatInt(int64(node.Listenport), 10))
- fmt.Println(" KeepAlive: " + strconv.FormatInt(int64(node.Keepalive), 10))
- fmt.Println(" Public Key: " + node.Publickey)
- fmt.Println(" Mac Address: " + node.Macaddress)
- fmt.Println(" Is Local?: " + strconv.FormatBool(node.Islocal))
- fmt.Println(" Is Dual Stack?: " + strconv.FormatBool(node.Isdualstack))
- fmt.Println(" Local Range: " + node.Localrange)
- if node.Dnsoff==true && !nodecfg.DNSOff {
- nodecfg.DNSOff = true
- }
- if !islocal && node.Islocal && node.Localrange != "" {
- fmt.Println("Resetting local settings for local network.")
- node.Localaddress, err = getLocalIP(node.Localrange)
- if err != nil {
- return err
- }
- node.Endpoint = node.Localaddress
- }
- err = modConfig(node)
- if err != nil {
- return err
- }
- if node.Ispending {
- fmt.Println("Node is marked as PENDING.")
- fmt.Println("Awaiting approval from Admin before configuring WireGuard.")
- if !noauto {
- fmt.Println("Configuring Netmaker Service.")
- err = ConfigureSystemD(network)
- return err
- }
- }
- peers, hasGateway, gateways, err := getPeers(node.Macaddress, network, server, node.Isdualstack, node.Isingressgateway)
- if err != nil {
- return err
- }
- fmt.Println("retrived peers, setting wireguard config.")
- err = storePrivKey(privkeystring, network)
- if err != nil {
- return err
- }
- err = initWireguard(node, privkeystring, peers, hasGateway, gateways)
- if err != nil {
- return err
- }
- if !noauto {
- err = ConfigureSystemD(network)
- }
- if err != nil {
- return err
- }
- return err
- }
- func getLocalIP(localrange string) (string, error) {
- _, localRange, err := net.ParseCIDR(localrange)
- if err != nil {
- return "", err
- }
- ifaces, err := net.Interfaces()
- if err != nil {
- return "", err
- }
- var local string
- found := false
- for _, i := range ifaces {
- if i.Flags&net.FlagUp == 0 {
- continue // interface down
- }
- if i.Flags&net.FlagLoopback != 0 {
- continue // loopback interface
- }
- addrs, err := i.Addrs()
- if err != nil {
- return "", err
- }
- for _, addr := range addrs {
- var ip net.IP
- switch v := addr.(type) {
- case *net.IPNet:
- if !found {
- ip = v.IP
- local = ip.String()
- found = localRange.Contains(ip)
- }
- case *net.IPAddr:
- if !found {
- ip = v.IP
- local = ip.String()
- found = localRange.Contains(ip)
- }
- }
- }
- }
- if !found || local == "" {
- return "", errors.New("Failed to find local IP in range " + localrange)
- }
- return local, nil
- }
- func getPublicIP() (string, error) {
- iplist := []string{"http://ip.client.gravitl.com","https://ifconfig.me", "http://api.ipify.org", "http://ipinfo.io/ip"}
- endpoint := ""
- var err error
- for _, ipserver := range iplist {
- resp, err := http.Get(ipserver)
- if err != nil {
- continue
- }
- defer resp.Body.Close()
- if resp.StatusCode == http.StatusOK {
- bodyBytes, err := ioutil.ReadAll(resp.Body)
- if err != nil {
- continue
- }
- endpoint = string(bodyBytes)
- break
- }
- }
- if err == nil && endpoint == "" {
- err = errors.New("Public Address Not Found.")
- }
- return endpoint, err
- }
- func modConfig(node *nodepb.Node) error{
- network := node.Nodenetwork
- if network == "" {
- return errors.New("No Network Provided")
- }
- modconfig, err := config.ReadConfig(network)
- if err != nil {
- return err
- }
- nodecfg := modconfig.Node
- if node.Name != ""{
- nodecfg.Name = node.Name
- }
- if node.Interface != ""{
- nodecfg.Interface = node.Interface
- }
- if node.Nodenetwork != ""{
- nodecfg.Network = node.Nodenetwork
- }
- if node.Macaddress != ""{
- nodecfg.MacAddress = node.Macaddress
- }
- if node.Localaddress != ""{
- nodecfg.LocalAddress = node.Localaddress
- }
- if node.Postup != ""{
- nodecfg.PostUp = node.Postup
- }
- if node.Postdown != ""{
- nodecfg.PostDown = node.Postdown
- }
- if node.Listenport != 0{
- nodecfg.Port = node.Listenport
- }
- if node.Keepalive != 0{
- nodecfg.KeepAlive = node.Keepalive
- }
- if node.Publickey != ""{
- nodecfg.PublicKey = node.Publickey
- }
- if node.Endpoint != ""{
- nodecfg.Endpoint = node.Endpoint
- }
- if node.Password != ""{
- nodecfg.Password = node.Password
- }
- if node.Address != ""{
- nodecfg.WGAddress = node.Address
- }
- if node.Address6 != ""{
- nodecfg.WGAddress6 = node.Address6
- }
- if node.Postchanges != "" {
- nodecfg.PostChanges = node.Postchanges
- }
- if node.Dnsoff == true {
- nodecfg.DNSOff = node.Dnsoff
- }
- if node.Isdualstack == true {
- nodecfg.IsDualStack = true
- }
- if node.Localrange != "" && node.Islocal {
- nodecfg.IsLocal = true
- nodecfg.LocalRange = node.Localrange
- }
- modconfig.Node = nodecfg
- err = config.Write(modconfig, network)
- return err
- }
- func getMacAddr() ([]string, error) {
- ifas, err := net.Interfaces()
- if err != nil {
- return nil, err
- }
- var as []string
- for _, ifa := range ifas {
- a := ifa.HardwareAddr.String()
- if a != "" {
- as = append(as, a)
- }
- }
- return as, nil
- }
- func initWireguard(node *nodepb.Node, privkey string, peers []wgtypes.PeerConfig, hasGateway bool, gateways []string) error {
- ipExec, err := exec.LookPath("ip")
- if err != nil {
- return err
- }
- key, err := wgtypes.ParseKey(privkey)
- if err != nil {
- return err
- }
- wgclient, err := wgctrl.New()
- //modcfg := config.Config
- //modcfg.ReadConfig()
- modcfg, err := config.ReadConfig(node.Nodenetwork)
- if err != nil {
- return err
- }
- nodecfg := modcfg.Node
- servercfg := modcfg.Server
- fmt.Println("beginning local WG config")
- if err != nil {
- log.Fatalf("failed to open client: %v", err)
- }
- defer wgclient.Close()
- fmt.Println("setting local settings")
- ifacename := node.Interface
- if nodecfg.Interface != "" {
- ifacename = nodecfg.Interface
- } else if node.Interface != "" {
- ifacename = node.Interface
- } else {
- log.Fatal("no interface to configure")
- }
- if node.Address == "" {
- log.Fatal("no address to configure")
- }
- nameserver := servercfg.Address
- nameserver = strings.Split(nameserver, ":")[0]
- network := node.Nodenetwork
- if nodecfg.Network != "" {
- network = nodecfg.Network
- } else if node.Nodenetwork != "" {
- network = node.Nodenetwork
- }
- cmdIPDevLinkAdd := &exec.Cmd {
- Path: ipExec,
- Args: []string{ ipExec, "link", "add", "dev", ifacename, "type", "wireguard" },
- Stdout: os.Stdout,
- Stderr: os.Stdout,
- }
- cmdIPAddrAdd := &exec.Cmd {
- Path: ipExec,
- Args: []string{ ipExec, "address", "add", "dev", ifacename, node.Address+"/24"},
- Stdout: os.Stdout,
- Stderr: os.Stdout,
- }
- currentiface, err := net.InterfaceByName(ifacename)
- if err != nil {
- err = cmdIPDevLinkAdd.Run()
- if err != nil && !strings.Contains(err.Error(), "exists") {
- fmt.Println("Error creating interface")
- //fmt.Println(err.Error())
- //return err
- }
- }
- match := false
- addrs, _ := currentiface.Addrs()
- for _, a := range addrs {
- if strings.Contains(a.String(), node.Address){
- match = true
- }
- }
- if !match {
- err = cmdIPAddrAdd.Run()
- if err != nil {
- fmt.Println("Error adding address")
- //return err
- }
- }
- var nodeport int
- nodeport = int(node.Listenport)
- fmt.Println("setting WG config from node and peers")
- //pubkey := privkey.PublicKey()
- conf := wgtypes.Config{
- PrivateKey: &key,
- ListenPort: &nodeport,
- ReplacePeers: true,
- Peers: peers,
- }
- _, err = wgclient.Device(ifacename)
- if err != nil {
- if os.IsNotExist(err) {
- fmt.Println("Device does not exist: ")
- fmt.Println(err)
- } else {
- log.Fatalf("Unknown config error: %v", err)
- }
- }
- fmt.Println("configuring WG device")
- err = wgclient.ConfigureDevice(ifacename, conf)
- if err != nil {
- if os.IsNotExist(err) {
- fmt.Println("Device does not exist: ")
- fmt.Println(err)
- } else {
- fmt.Printf("This is inconvenient: %v", err)
- }
- }
- //=========DNS Setup==========\\
- if nodecfg.DNSOff != true {
- _, err := exec.LookPath("resolvectl")
- if err != nil {
- fmt.Println(err)
- fmt.Println("WARNING: resolvectl not present. Unable to set dns. Install resolvectl or run manually.")
- } else {
- _, err = exec.Command("resolvectl", "domain", ifacename, "~"+network).Output()
- if err != nil {
- fmt.Println(err)
- fmt.Println("WARNING: Error encountered setting dns. Aborted setting dns.")
- } else {
- _, err = exec.Command("resolvectl", "default-route", ifacename, "false").Output()
- if err != nil {
- fmt.Println(err)
- fmt.Println("WARNING: Error encountered setting dns. Aborted setting dns.")
- } else {
- _, err = exec.Command("resolvectl", "dns", ifacename, nameserver).Output()
- fmt.Println(err)
- }
- }
- }
- }
- //=========End DNS Setup=======\\
- cmdIPLinkUp := &exec.Cmd {
- Path: ipExec,
- Args: []string{ ipExec, "link", "set", "up", "dev", ifacename},
- Stdout: os.Stdout,
- Stderr: os.Stdout,
- }
- cmdIPLinkDown := &exec.Cmd {
- Path: ipExec,
- Args: []string{ ipExec, "link", "set", "down", "dev", ifacename},
- Stdout: os.Stdout,
- Stderr: os.Stdout,
- }
- err = cmdIPLinkDown.Run()
- if nodecfg.PostDown != "" {
- runcmds := strings.Split(nodecfg.PostDown, "; ")
- err = runCmds(runcmds)
- if err != nil {
- fmt.Println("Error encountered running PostDown: " + err.Error())
- }
- }
- err = cmdIPLinkUp.Run()
- if err != nil {
- return err
- }
- if nodecfg.PostUp != "" {
- runcmds := strings.Split(nodecfg.PostUp, "; ")
- err = runCmds(runcmds)
- if err != nil {
- fmt.Println("Error encountered running PostUp: " + err.Error())
- }
- }
- if (hasGateway) {
- for _, gateway := range gateways {
- out, err := exec.Command(ipExec,"-4","route","add",gateway,"dev",ifacename).Output()
- fmt.Println(string(out))
- if err != nil {
- fmt.Println("Error encountered adding gateway: " + err.Error())
- }
- }
- }
- if (node.Address6 != "" && node.Isdualstack) {
- fmt.Println("Adding address: " + node.Address6)
- out, err := exec.Command(ipExec, "address", "add", "dev", ifacename, node.Address6+"/64").Output()
- if err != nil {
- fmt.Println(out)
- fmt.Println("Error encountered adding ipv6: " + err.Error())
- }
- }
- return err
- }
- func runCmds(commands []string) error {
- var err error
- for _, command := range commands {
- fmt.Println("Running command: " + command)
- args := strings.Fields(command)
- out, err := exec.Command(args[0], args[1:]...).Output()
- fmt.Println(string(out))
- if err != nil {
- return err
- }
- }
- return err
- }
- func setWGKeyConfig(network string, serveraddr string) error {
- ctx := context.Background()
- var header metadata.MD
- var wcclient nodepb.NodeServiceClient
- var requestOpts grpc.DialOption
- requestOpts = grpc.WithInsecure()
- conn, err := grpc.Dial(serveraddr, requestOpts)
- if err != nil {
- fmt.Printf("Cant dial GRPC server: %v", err)
- return err
- }
- wcclient = nodepb.NewNodeServiceClient(conn)
- fmt.Println("Authenticating with GRPC Server")
- ctx, err = SetJWT(wcclient, network)
- if err != nil {
- fmt.Printf("Failed to authenticate: %v", err)
- return err
- }
- fmt.Println("Authenticated")
- node := getNode(network)
- privatekey, err := wgtypes.GeneratePrivateKey()
- if err != nil {
- return err
- }
- privkeystring := privatekey.String()
- publickey := privatekey.PublicKey()
- node.Publickey = publickey.String()
- err = storePrivKey(privkeystring, network)
- if err != nil {
- return err
- }
- err = modConfig(&node)
- if err != nil {
- return err
- }
- postnode := getNode(network)
- req := &nodepb.UpdateNodeReq{
- Node: &postnode,
- }
- _, err = wcclient.UpdateNode(ctx, req, grpc.Header(&header))
- if err != nil {
- return err
- }
- err = setWGConfig(network)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- return err
- }
- func setWGConfig(network string) error {
- cfg, err := config.ReadConfig(network)
- if err != nil {
- return err
- }
- servercfg := cfg.Server
- nodecfg := cfg.Node
- node := getNode(network)
- peers, hasGateway, gateways, err := getPeers(node.Macaddress, nodecfg.Network, servercfg.Address, node.Isdualstack, node.Isingressgateway)
- if err != nil {
- return err
- }
- privkey, err := retrievePrivKey(network)
- if err != nil {
- return err
- }
- err = initWireguard(&node, privkey, peers, hasGateway, gateways)
- if err != nil {
- return err
- }
- return err
- }
- func storePrivKey(key string, network string) error{
- d1 := []byte(key)
- err := ioutil.WriteFile("/etc/netclient/wgkey-" + network, d1, 0644)
- return err
- }
- func retrievePrivKey(network string) (string, error) {
- dat, err := ioutil.ReadFile("/etc/netclient/wgkey-" + network)
- return string(dat), err
- }
- func getPrivateAddr() (string, error) {
- ifaces, err := net.Interfaces()
- if err != nil {
- return "", err
- }
- var local string
- found := false
- for _, i := range ifaces {
- if i.Flags&net.FlagUp == 0 {
- continue // interface down
- }
- if i.Flags&net.FlagLoopback != 0 {
- continue // loopback interface
- }
- addrs, err := i.Addrs()
- if err != nil {
- return "", err
- }
- for _, addr := range addrs {
- var ip net.IP
- switch v := addr.(type) {
- case *net.IPNet:
- if !found {
- ip = v.IP
- local = ip.String()
- found = true
- }
- case *net.IPAddr:
- if !found {
- ip = v.IP
- local = ip.String()
- found = true
- }
- }
- }
- }
- if !found {
- err := errors.New("Local Address Not Found.")
- return "", err
- }
- return local, err
- }
- func CheckIn(network string) error {
- node := getNode(network)
- cfg, err := config.ReadConfig(network)
- if err != nil {
- return err
- }
- nodecfg := cfg.Node
- servercfg := cfg.Server
- fmt.Println("Checking into server: " + servercfg.Address)
- setupcheck := true
- ipchange := false
- if !(nodecfg.IPForwarding == "off") {
- out, err := exec.Command("sysctl", "net.ipv4.ip_forward").Output()
- if err != nil {
- fmt.Println(err)
- fmt.Println("WARNING: Error encountered setting ip forwarding. This can break functionality.")
- } else {
- s := strings.Fields(string(out))
- if s[2] != "1" {
- _, err = exec.Command("sysctl", "-w", "net.ipv4.ip_forward=1").Output()
- if err != nil {
- fmt.Println(err)
- fmt.Println("WARNING: Error encountered setting ip forwarding. You may want to investigate this.")
- }
- }
- }
- }
- if !nodecfg.RoamingOff {
- if !nodecfg.IsLocal {
- fmt.Println("Checking to see if public addresses have changed")
- extIP, err := getPublicIP()
- if err != nil {
- fmt.Printf("Error encountered checking ip addresses: %v", err)
- }
- if nodecfg.Endpoint != extIP && extIP != "" {
- fmt.Println("Endpoint has changed from " +
- nodecfg.Endpoint + " to " + extIP)
- fmt.Println("Updating address")
- nodecfg.Endpoint = extIP
- nodecfg.PostChanges = "true"
- node.Endpoint = extIP
- node.Postchanges = "true"
- ipchange = true
- }
- intIP, err := getPrivateAddr()
- if err != nil {
- fmt.Printf("Error encountered checking ip addresses: %v", err)
- }
- if nodecfg.LocalAddress != intIP && intIP != "" {
- fmt.Println("Local Address has changed from " +
- nodecfg.LocalAddress + " to " + intIP)
- fmt.Println("Updating address")
- nodecfg.LocalAddress = intIP
- nodecfg.PostChanges = "true"
- node.Localaddress = intIP
- node.Postchanges = "true"
- ipchange = true
- }
- } else {
- fmt.Println("Checking to see if local addresses have changed")
- localIP, err := getLocalIP(nodecfg.LocalRange)
- if err != nil {
- fmt.Printf("Error encountered checking ip addresses: %v", err)
- }
- if nodecfg.Endpoint != localIP && localIP != "" {
- fmt.Println("Endpoint has changed from " +
- nodecfg.Endpoint + " to " + localIP)
- fmt.Println("Updating address")
- nodecfg.Endpoint = localIP
- nodecfg.LocalAddress = localIP
- nodecfg.PostChanges = "true"
- node.Endpoint = localIP
- node.Localaddress = localIP
- node.Postchanges = "true"
- ipchange = true
- }
- }
- if node.Postchanges != "true" {
- fmt.Println("Addresses have not changed.")
- }
- }
- if ipchange {
- err := modConfig(&node)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- err = setWGConfig(network)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- node = getNode(network)
- cfg, err := config.ReadConfig(network)
- if err != nil {
- return err
- }
- nodecfg = cfg.Node
- }
- var wcclient nodepb.NodeServiceClient
- var requestOpts grpc.DialOption
- requestOpts = grpc.WithInsecure()
- conn, err := grpc.Dial(servercfg.Address, requestOpts)
- if err != nil {
- fmt.Printf("Cant dial GRPC server: %v", err)
- return err
- }
- wcclient = nodepb.NewNodeServiceClient(conn)
- ctx := context.Background()
- fmt.Println("Authenticating with GRPC Server")
- ctx, err = SetJWT(wcclient, network)
- if err != nil {
- fmt.Printf("Failed to authenticate: %v", err)
- return err
- }
- fmt.Println("Authenticated")
- fmt.Println("Checking In.")
- var header metadata.MD
- node.Nodenetwork = network
- checkinres, err := wcclient.CheckIn(
- ctx,
- &nodepb.CheckInReq{
- Node: &node,
- },
- grpc.Header(&header),
- )
- if err != nil {
- if checkinres != nil && checkinres.Checkinresponse.Ispending {
- fmt.Println("Node is in pending status. Waiting for Admin approval of node before making further updates.")
- return nil
- }
- fmt.Printf("Unable to process Check In request: %v", err)
- return err
- }
- fmt.Println("Checked in.")
- if checkinres.Checkinresponse.Ispending {
- fmt.Println("Node is in pending status. Waiting for Admin approval of node before making further updates.")
- return err
- }
- newinterface := getNode(network).Interface
- readreq := &nodepb.ReadNodeReq{
- Macaddress: node.Macaddress,
- Network: node.Nodenetwork,
- }
- readres, err := wcclient.ReadNode(ctx, readreq, grpc.Header(&header))
- if err != nil {
- fmt.Printf("Error: %v", err)
- } else {
- currentiface := readres.Node.Interface
- ifaceupdate := newinterface != currentiface
- if err != nil {
- log.Printf("Error retrieving interface: %v", err)
- }
- if ifaceupdate {
- fmt.Println("Interface update: " + currentiface +
- " >>>> " + newinterface)
- err := DeleteInterface(currentiface, nodecfg.PostDown)
- if err != nil {
- fmt.Println("ERROR DELETING INTERFACE: " + currentiface)
- }
- err = setWGConfig(network)
- if err != nil {
- log.Printf("Error updating interface: %v", err)
- }
- }
- }
- if checkinres.Checkinresponse.Needconfigupdate {
- fmt.Println("Server has requested that node update config.")
- fmt.Println("Updating config from remote server.")
- req := &nodepb.ReadNodeReq{
- Macaddress: node.Macaddress,
- Network: node.Nodenetwork,
- }
- readres, err := wcclient.ReadNode(ctx, req, grpc.Header(&header))
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- err = modConfig(readres.Node)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- err = setWGConfig(network)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- setupcheck = false
- } else if nodecfg.PostChanges == "true" {
- fmt.Println("Node has requested to update remote config.")
- fmt.Println("Posting local config to remote server.")
- postnode := getNode(network)
- req := &nodepb.UpdateNodeReq{
- Node: &postnode,
- }
- res, err := wcclient.UpdateNode(ctx, req, grpc.Header(&header))
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- res.Node.Postchanges = "false"
- err = modConfig(res.Node)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- err = setWGConfig(network)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- setupcheck = false
- }
- if checkinres.Checkinresponse.Needkeyupdate {
- fmt.Println("Server has requested that node update key pairs.")
- fmt.Println("Proceeding to re-generate key pairs for Wiregard.")
- err = setWGKeyConfig(network, servercfg.Address)
- if err != nil {
- return err
- log.Fatalf("Unable to process reset keys request: %v", err)
- }
- setupcheck = false
- }
- if checkinres.Checkinresponse.Needpeerupdate {
- fmt.Println("Server has requested that node update peer list.")
- fmt.Println("Updating peer list from remote server.")
- err = setWGConfig(network)
- if err != nil {
- return err
- log.Fatalf("Unable to process Set Peers request: %v", err)
- }
- setupcheck = false
- }
- if checkinres.Checkinresponse.Needdelete {
- fmt.Println("This machine got the delete signal. Deleting.")
- err := Remove(network)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- }
- if setupcheck {
- iface := nodecfg.Interface
- _, err := net.InterfaceByName(iface)
- if err != nil {
- fmt.Println("interface " + iface + " does not currently exist. Setting up WireGuard.")
- err = setWGKeyConfig(network, servercfg.Address)
- if err != nil {
- return err
- log.Fatalf("Error: %v", err)
- }
- }
- }
- return nil
- }
- func needInterfaceUpdate(ctx context.Context, mac string, network string, iface string) (bool, string, error) {
- var header metadata.MD
- req := &nodepb.ReadNodeReq{
- Macaddress: mac,
- Network: network,
- }
- readres, err := wcclient.ReadNode(ctx, req, grpc.Header(&header))
- if err != nil {
- return false, "", err
- log.Fatalf("Error: %v", err)
- }
- oldiface := readres.Node.Interface
- return iface != oldiface, oldiface, err
- }
- func getNode(network string) nodepb.Node {
- modcfg, err := config.ReadConfig(network)
- if err != nil {
- log.Fatalf("Error: %v", err)
- }
- nodecfg := modcfg.Node
- var node nodepb.Node
- node.Name = nodecfg.Name
- node.Interface = nodecfg.Interface
- node.Nodenetwork = nodecfg.Network
- node.Localaddress = nodecfg.LocalAddress
- node.Address = nodecfg.WGAddress
- node.Address6 = nodecfg.WGAddress6
- node.Listenport = nodecfg.Port
- node.Keepalive = nodecfg.KeepAlive
- node.Postup = nodecfg.PostUp
- node.Postdown = nodecfg.PostDown
- node.Publickey = nodecfg.PublicKey
- node.Macaddress = nodecfg.MacAddress
- node.Endpoint = nodecfg.Endpoint
- node.Password = nodecfg.Password
- node.Dnsoff = nodecfg.DNSOff
- node.Isdualstack = nodecfg.IsDualStack
- return node
- }
- func Remove(network string) error {
- //need to implement checkin on server side
- cfg, err := config.ReadConfig(network)
- if err != nil {
- return err
- }
- servercfg := cfg.Server
- node := cfg.Node
- fmt.Println("Deleting remote node with MAC: " + node.MacAddress)
- var wcclient nodepb.NodeServiceClient
- var requestOpts grpc.DialOption
- requestOpts = grpc.WithInsecure()
- conn, err := grpc.Dial(servercfg.Address, requestOpts)
- if err != nil {
- log.Printf("Unable to establish client connection to " + servercfg.Address + ": %v", err)
- //return err
- }else {
- wcclient = nodepb.NewNodeServiceClient(conn)
- ctx := context.Background()
- fmt.Println("Authenticating with GRPC Server")
- ctx, err = SetJWT(wcclient, network)
- if err != nil {
- //return err
- log.Printf("Failed to authenticate: %v", err)
- } else {
- fmt.Println("Authenticated")
- var header metadata.MD
- _, err = wcclient.DeleteNode(
- ctx,
- &nodepb.DeleteNodeReq{
- Macaddress: node.MacAddress,
- NetworkName: node.Network,
- },
- grpc.Header(&header),
- )
- if err != nil {
- log.Printf("Encountered error deleting node: %v", err)
- fmt.Println(err)
- } else {
- fmt.Println("Deleted node " + node.MacAddress)
- }
- }
- }
- err = WipeLocal(network)
- if err != nil {
- log.Printf("Unable to wipe local config: %v", err)
- }
- err = RemoveSystemDServices(network)
- if err != nil {
- return err
- log.Printf("Unable to remove systemd services: %v", err)
- }
- fmt.Printf("Please investigate any stated errors to ensure proper removal.")
- fmt.Printf("Failure to delete node from server via gRPC will mean node still exists and needs to be manually deleted by administrator.")
- return nil
- }
- func WipeLocal(network string) error{
- cfg, err := config.ReadConfig(network)
- if err != nil {
- return err
- }
- nodecfg := cfg.Node
- ifacename := nodecfg.Interface
- //home, err := homedir.Dir()
- home := "/etc/netclient"
- err = os.Remove(home + "/netconfig-" + network)
- if err != nil {
- fmt.Println(err)
- }
- err = os.Remove(home + "/nettoken-" + network)
- if err != nil {
- fmt.Println(err)
- }
- err = os.Remove(home + "/wgkey-" + network)
- if err != nil {
- fmt.Println(err)
- }
- ipExec, err := exec.LookPath("ip")
- if ifacename != "" {
- cmdIPLinkDel := &exec.Cmd {
- Path: ipExec,
- Args: []string{ ipExec, "link", "del", ifacename },
- Stdout: os.Stdout,
- Stderr: os.Stdout,
- }
- err = cmdIPLinkDel.Run()
- if err != nil {
- fmt.Println(err)
- }
- if nodecfg.PostDown != "" {
- runcmds := strings.Split(nodecfg.PostDown, "; ")
- err = runCmds(runcmds)
- if err != nil {
- fmt.Println("Error encountered running PostDown: " + err.Error())
- }
- }
- }
- return err
- }
- func DeleteInterface(ifacename string, postdown string) error{
- ipExec, err := exec.LookPath("ip")
- cmdIPLinkDel := &exec.Cmd {
- Path: ipExec,
- Args: []string{ ipExec, "link", "del", ifacename },
- Stdout: os.Stdout,
- Stderr: os.Stdout,
- }
- err = cmdIPLinkDel.Run()
- if err != nil {
- fmt.Println(err)
- }
- if postdown != "" {
- runcmds := strings.Split(postdown, "; ")
- err = runCmds(runcmds)
- if err != nil {
- fmt.Println("Error encountered running PostDown: " + err.Error())
- }
- }
- return err
- }
|