Compare commits
	
		
			28 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | dcbdbe331a | ||
|   | 275eb18ca8 | ||
|   | 39b23f4e91 | ||
|   | 9cc7c03eb3 | ||
|   | 8da27edd39 | ||
|   | c750bc1d92 | ||
|   | cc4cd1c010 | ||
|   | 4b1b189d5b | ||
|   | 09f1134d1e | ||
|   | 31cf03c10a | ||
|   | 1885e741fc | ||
|   | 03d0cc5453 | ||
|   | a27171ea62 | ||
|   | c4a57651f2 | ||
|   | da9adb6f08 | ||
|   | a454f57fb0 | ||
|   | 9d76c89aca | ||
|   | f7664f3263 | ||
|   | 6418a305aa | ||
|   | 4e25bc9b36 | ||
|   | b1caae2d87 | ||
|   | 48531ebdc2 | ||
|   | 663a66b2a2 | ||
|   | be276aef0b | ||
|   | afc2118b98 | ||
|   | 92c169dc9c | ||
|   | fa103eb1a1 | ||
|   | d1baa9fe1a | 
							
								
								
									
										5
									
								
								.github/workflows/build.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/build.sh
									
									
									
									
										vendored
									
									
								
							| @@ -1,12 +1,13 @@ | |||||||
| #!/bin/bash | #!/bin/bash | ||||||
|  |  | ||||||
| platforms=("linux/amd64" "linux/arm" "linux/arm64" "darwin/amd64" "darwin/arm64") | platforms=("linux/amd64" "linux/arm" "linux/arm64" "darwin/amd64" "darwin/arm64" "windows/amd64" "windows/386") | ||||||
|  |  | ||||||
| for platform in "${platforms[@]}" | for platform in "${platforms[@]}" | ||||||
| do | do | ||||||
|     platform_split=(${platform//\// }) |     platform_split=(${platform//\// }) | ||||||
|     GOOS=${platform_split[0]} |     GOOS=${platform_split[0]} | ||||||
|     GOARCH=${platform_split[1]} |     GOARCH=${platform_split[1]} | ||||||
|     env GOOS=$GOOS GOARCH=$GOARCH CGO_ENABLED=0 go build -ldflags "-s -w -X github.com/hyprspace/hyprspace/cli.appVersion=$1" -o hyprspace-$1-${GOOS}-${GOARCH} . |     [ $GOOS == "windows" ] && EXT=".exe" | ||||||
|  |     env GOOS=$GOOS GOARCH=$GOARCH CGO_ENABLED=0 go build -ldflags "-s -w -X github.com/hyprspace/hyprspace/cli.appVersion=$1" -o hyprspace-$1-${GOOS}-${GOARCH}${EXT} . | ||||||
|  |  | ||||||
| done | done | ||||||
							
								
								
									
										15
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| <img src="hyprspace.png" width="300" height="300"> | <img src="hyprspace.png" width="250"> | ||||||
|  |  | ||||||
| # Hyprspace | # Hyprspace | ||||||
| [](https://goreportcard.com/report/github.com/hyprspace/hyprspace) | [](https://goreportcard.com/report/github.com/hyprspace/hyprspace) | ||||||
| @@ -22,12 +22,12 @@ https://user-images.githubusercontent.com/19558067/121469777-f42cdb80-c971-11eb- | |||||||
| - [Tutorial](#tutorial) | - [Tutorial](#tutorial) | ||||||
|  |  | ||||||
| ## A Bit of Backstory | ## A Bit of Backstory | ||||||
| [Libp2p](libp2p.io) is a networking library created by [Protocol Labs](https://protocol.ai) that allows nodes to discover each other using a Distributed Hash Table. Paired with [NAT hole punching](https://en.wikipedia.org/wiki/Hole_punching_(networking)) this allows Hyprspace to create a direct encrypted tunnel between two nodes even if they're both behind firewalls. | [Libp2p](https://libp2p.io) is a networking library created by [Protocol Labs](https://protocol.ai) that allows nodes to discover each other using a Distributed Hash Table. Paired with [NAT hole punching](https://en.wikipedia.org/wiki/Hole_punching_(networking)) this allows Hyprspace to create a direct encrypted tunnel between two nodes even if they're both behind firewalls. | ||||||
|  |  | ||||||
| **Moreover! Each node doesn't even need to know the other's ip address prior to starting up the connection.** This makes Hyprspace perfect for devices that frequently migrate between locations but still require a constant virtual ip address. | **Moreover! Each node doesn't even need to know the other's ip address prior to starting up the connection.** This makes Hyprspace perfect for devices that frequently migrate between locations but still require a constant virtual ip address. | ||||||
|  |  | ||||||
| ### So How Does Hyprspace Compare to Something Like Wireguard? | ### So How Does Hyprspace Compare to Something Like Wireguard? | ||||||
| [Wireguard](https://wireguard.com) is an amazing VPN written by Jason A. Donenfeld. If you haven't already, definitely go check it out! Wireguard actually inspired me to write Hyprspace. That said, although Wireguard is in a class of it's own as a great VPN, it requires at least one of your nodes to have a public IP address. In this mode, as long as one of your nodes is publicly accessible, it can be used as a cental relay to reach the other nodes in the network. However, this means that all of the traffic for your entire system is going through that one system which can slow down your network and make it fragile in the case that node goes down and you loose the whole network. So instead say that you want each node to be able to directly connect to each other as they do in Hyprspace. Unfortunately through Wireguard this would require every node to be publicly addressable which means manual port forwarding and no travelling nodes. | [Wireguard](https://wireguard.com) is an amazing VPN written by Jason A. Donenfeld. If you haven't already, definitely go check it out! Wireguard actually inspired me to write Hyprspace. That said, although Wireguard is in a class of its own as a great VPN, it requires at least one of your nodes to have a public IP address. In this mode, as long as one of your nodes is publicly accessible, it can be used as a central relay to reach the other nodes in the network. However, this means that all of the traffic for your entire system is going through that one system which can slow down your network and make it fragile in the case that node goes down and you lose the whole network. So instead say that you want each node to be able to directly connect to each other as they do in Hyprspace. Unfortunately through Wireguard this would require every node to be publicly addressable which means manual port forwarding and no travelling nodes. | ||||||
|  |  | ||||||
| By contrast Hyprspace allows all of your nodes to connect directly to each other creating a strong reliable network even if they're all behind their own firewalls. No manual port forwarding required!  | By contrast Hyprspace allows all of your nodes to connect directly to each other creating a strong reliable network even if they're all behind their own firewalls. No manual port forwarding required!  | ||||||
|  |  | ||||||
| @@ -45,10 +45,7 @@ If anyone else has some use cases please add them! Pull requests welcome! | |||||||
|  |  | ||||||
| ## Getting Started | ## Getting Started | ||||||
| ### Prerequisites | ### Prerequisites | ||||||
| If you're running Hyprspace on a Mac you'll need to install `iproute2mac`. If you're using the `brew` package manager that's as simple as, | If you're running Hyprspace on Windows you'll need to install [tap-windows](http://build.openvpn.net/downloads/releases/). | ||||||
| ```bash |  | ||||||
| brew install iproute2mac |  | ||||||
| ``` |  | ||||||
|  |  | ||||||
| ### Installation | ### Installation | ||||||
|  |  | ||||||
| @@ -88,6 +85,8 @@ but yours could be anything you'd like. | |||||||
|  |  | ||||||
| (Note: if you're using a Mac you'll have to use the interface name `utun[0-9]`. Check which interfaces are already in use by running `ip a` once you've got `iproute2mac` installed.) | (Note: if you're using a Mac you'll have to use the interface name `utun[0-9]`. Check which interfaces are already in use by running `ip a` once you've got `iproute2mac` installed.) | ||||||
|  |  | ||||||
|  | (Note: if you're using Windows you'll have to use the interface name as seen in Control Panel. IP address will be set automatically only if you run Hyprspace as Administrator.) | ||||||
|  |  | ||||||
| ###### Local Machine | ###### Local Machine | ||||||
| ```bash | ```bash | ||||||
| sudo hyprspace init hs0 | sudo hyprspace init hs0 | ||||||
| @@ -178,7 +177,7 @@ sudo hyprspace down hs1 | |||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
| Copyright 2021 Alec Scott <hi@alecbcs.com> | Copyright 2021-2022 Alec Scott <hi@alecbcs.com> | ||||||
|  |  | ||||||
| Licensed under the Apache License, Version 2.0 (the "License"); | Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
| you may not use this file except in compliance with the License. | you may not use this file except in compliance with the License. | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								cli/down.go
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								cli/down.go
									
									
									
									
									
								
							| @@ -2,6 +2,9 @@ package cli | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"os" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"strconv" | ||||||
|  |  | ||||||
| 	"github.com/DataDrake/cli-ng/v2/cmd" | 	"github.com/DataDrake/cli-ng/v2/cmd" | ||||||
| 	"github.com/hyprspace/hyprspace/tun" | 	"github.com/hyprspace/hyprspace/tun" | ||||||
| @@ -26,7 +29,34 @@ func DownRun(r *cmd.Root, c *cmd.Sub) { | |||||||
| 	// Parse Command Args | 	// Parse Command Args | ||||||
| 	args := c.Args.(*DownArgs) | 	args := c.Args.(*DownArgs) | ||||||
|  |  | ||||||
| 	fmt.Println("[+] ip link delete dev " + args.InterfaceName) | 	// Parse Global Config Flag for Custom Config Path | ||||||
| 	err := tun.Delete(args.InterfaceName) | 	configPath := r.Flags.(*GlobalFlags).Config | ||||||
|  | 	if configPath == "" { | ||||||
|  | 		configPath = "/etc/hyprspace/" + args.InterfaceName + ".yaml" | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Read lock from file system to stop process. | ||||||
|  | 	lockPath := filepath.Join(filepath.Dir(configPath), args.InterfaceName+".lock") | ||||||
|  | 	out, err := os.ReadFile(lockPath) | ||||||
| 	checkErr(err) | 	checkErr(err) | ||||||
|  |  | ||||||
|  | 	pid, err := strconv.Atoi(string(out)) | ||||||
|  | 	checkErr(err) | ||||||
|  |  | ||||||
|  | 	process, err := os.FindProcess(pid) | ||||||
|  | 	checkErr(err) | ||||||
|  |  | ||||||
|  | 	err0 := process.Signal(os.Interrupt) | ||||||
|  |  | ||||||
|  | 	err1 := tun.Delete(args.InterfaceName) | ||||||
|  |  | ||||||
|  | 	// Different types of systems may need the tun devices destroyed first or | ||||||
|  | 	// the process to exit first don't worry as long as one of these two has | ||||||
|  | 	// suceeded. | ||||||
|  | 	if err0 != nil && err1 != nil { | ||||||
|  | 		checkErr(err0) | ||||||
|  | 		checkErr(err1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	fmt.Println("[+] deleted hyprspace " + args.InterfaceName + " daemon") | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										173
									
								
								cli/up.go
									
									
									
									
									
								
							
							
						
						
									
										173
									
								
								cli/up.go
									
									
									
									
									
								
							| @@ -5,11 +5,12 @@ import ( | |||||||
| 	"context" | 	"context" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io" |  | ||||||
| 	"log" | 	"log" | ||||||
| 	"net" | 	"net" | ||||||
| 	"os" | 	"os" | ||||||
| 	"os/signal" | 	"os/signal" | ||||||
|  | 	"path/filepath" | ||||||
|  | 	"runtime" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"syscall" | 	"syscall" | ||||||
| @@ -22,17 +23,17 @@ import ( | |||||||
| 	"github.com/libp2p/go-libp2p-core/host" | 	"github.com/libp2p/go-libp2p-core/host" | ||||||
| 	"github.com/libp2p/go-libp2p-core/network" | 	"github.com/libp2p/go-libp2p-core/network" | ||||||
| 	"github.com/libp2p/go-libp2p-core/peer" | 	"github.com/libp2p/go-libp2p-core/peer" | ||||||
| 	"github.com/songgao/water" |  | ||||||
| 	"golang.org/x/net/ipv4" |  | ||||||
| ) | ) | ||||||
|  |  | ||||||
| var ( | var ( | ||||||
| 	// iface is the tun device used to pass packets between | 	// iface is the tun device used to pass packets between | ||||||
| 	// Hyprspace and the user's machine. | 	// Hyprspace and the user's machine. | ||||||
| 	iface *water.Interface | 	tunDev *tun.TUN | ||||||
| 	// RevLookup allow quick lookups of an incoming stream | 	// RevLookup allow quick lookups of an incoming stream | ||||||
| 	// for security before accepting or responding to any data. | 	// for security before accepting or responding to any data. | ||||||
| 	RevLookup map[string]bool | 	RevLookup map[string]string | ||||||
|  | 	// activeStreams is a map of active streams to a peer | ||||||
|  | 	activeStreams map[string]network.Stream | ||||||
| ) | ) | ||||||
|  |  | ||||||
| // Up creates and brings up a Hyprspace Interface. | // Up creates and brings up a Hyprspace Interface. | ||||||
| @@ -91,21 +92,42 @@ func UpRun(r *cmd.Root, c *cmd.Sub) { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Setup reverse lookup hash map for authentication. | 	// Setup reverse lookup hash map for authentication. | ||||||
| 	RevLookup = make(map[string]bool, len(cfg.Peers)) | 	RevLookup = make(map[string]string, len(cfg.Peers)) | ||||||
| 	for _, id := range cfg.Peers { | 	for ip, id := range cfg.Peers { | ||||||
| 		RevLookup[id.ID] = true | 		RevLookup[id.ID] = ip | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	fmt.Println("[+] Creating TUN Device") | 	fmt.Println("[+] Creating TUN Device") | ||||||
| 	// Create new TUN device |  | ||||||
| 	iface, err = tun.New(cfg.Interface.Name) | 	if runtime.GOOS == "darwin" { | ||||||
| 	if err != nil { | 		if len(cfg.Peers) > 1 { | ||||||
| 		checkErr(errors.New("interface already in use")) | 			checkErr(errors.New("cannot create interface macos does not support more than one peer")) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Grab ip address of only peer in config | ||||||
|  | 		var destPeer string | ||||||
|  | 		for ip := range cfg.Peers { | ||||||
|  | 			destPeer = ip | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// Create new TUN device | ||||||
|  | 		tunDev, err = tun.New( | ||||||
|  | 			cfg.Interface.Name, | ||||||
|  | 			tun.Address(cfg.Interface.Address), | ||||||
|  | 			tun.DestAddress(destPeer), | ||||||
|  | 			tun.MTU(1420), | ||||||
|  | 		) | ||||||
|  | 	} else { | ||||||
|  | 		// Create new TUN device | ||||||
|  | 		tunDev, err = tun.New( | ||||||
|  | 			cfg.Interface.Name, | ||||||
|  | 			tun.Address(cfg.Interface.Address), | ||||||
|  | 			tun.MTU(1420), | ||||||
|  | 		) | ||||||
|  | 	} | ||||||
|  | 	if err != nil { | ||||||
|  | 		checkErr(err) | ||||||
| 	} | 	} | ||||||
| 	// Set TUN MTU |  | ||||||
| 	tun.SetMTU(cfg.Interface.Name, 1420) |  | ||||||
| 	// Add Address to Interface |  | ||||||
| 	tun.SetAddress(cfg.Interface.Name, cfg.Interface.Address) |  | ||||||
|  |  | ||||||
| 	// Setup System Context | 	// Setup System Context | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
| @@ -113,34 +135,16 @@ func UpRun(r *cmd.Root, c *cmd.Sub) { | |||||||
| 	fmt.Println("[+] Creating LibP2P Node") | 	fmt.Println("[+] Creating LibP2P Node") | ||||||
|  |  | ||||||
| 	// Check that the listener port is available. | 	// Check that the listener port is available. | ||||||
| 	var ln net.Listener | 	port, err := verifyPort(cfg.Interface.ListenPort) | ||||||
| 	port := cfg.Interface.ListenPort | 	checkErr(err) | ||||||
| 	if port != 8001 { |  | ||||||
| 		ln, err = net.Listen("tcp", ":"+strconv.Itoa(port)) |  | ||||||
| 		if err != nil { |  | ||||||
| 			checkErr(errors.New("could not create node, listen port already in use by something else")) |  | ||||||
| 		} |  | ||||||
| 	} else { |  | ||||||
| 		for { |  | ||||||
| 			ln, err = net.Listen("tcp", ":"+strconv.Itoa(port)) |  | ||||||
| 			if err == nil { |  | ||||||
| 				break |  | ||||||
| 			} |  | ||||||
| 			if port >= 65535 { |  | ||||||
| 				checkErr(errors.New("failed to find open port")) |  | ||||||
| 			} |  | ||||||
| 			port++ |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	if ln != nil { |  | ||||||
| 		ln.Close() |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	// Create P2P Node | 	// Create P2P Node | ||||||
| 	host, dht, err := p2p.CreateNode(ctx, | 	host, dht, err := p2p.CreateNode( | ||||||
|  | 		ctx, | ||||||
| 		cfg.Interface.PrivateKey, | 		cfg.Interface.PrivateKey, | ||||||
| 		port, | 		port, | ||||||
| 		streamHandler) | 		streamHandler, | ||||||
|  | 	) | ||||||
| 	checkErr(err) | 	checkErr(err) | ||||||
|  |  | ||||||
| 	// Setup Peer Table for Quick Packet --> Dest ID lookup | 	// Setup Peer Table for Quick Packet --> Dest ID lookup | ||||||
| @@ -155,6 +159,9 @@ func UpRun(r *cmd.Root, c *cmd.Sub) { | |||||||
| 	go p2p.Discover(ctx, host, dht, peerTable) | 	go p2p.Discover(ctx, host, dht, peerTable) | ||||||
| 	go prettyDiscovery(ctx, host, peerTable) | 	go prettyDiscovery(ctx, host, peerTable) | ||||||
|  |  | ||||||
|  | 	// Configure path for lock | ||||||
|  | 	lockPath := filepath.Join(filepath.Dir(cfg.Path), cfg.Interface.Name+".lock") | ||||||
|  |  | ||||||
| 	go func() { | 	go func() { | ||||||
| 		// Wait for a SIGINT or SIGTERM signal | 		// Wait for a SIGINT or SIGTERM signal | ||||||
| 		ch := make(chan os.Signal, 1) | 		ch := make(chan os.Signal, 1) | ||||||
| @@ -166,36 +173,59 @@ func UpRun(r *cmd.Root, c *cmd.Sub) { | |||||||
| 		if err := host.Close(); err != nil { | 		if err := host.Close(); err != nil { | ||||||
| 			panic(err) | 			panic(err) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		// Remove daemon lock from file system. | ||||||
|  | 		err = os.Remove(lockPath) | ||||||
|  | 		checkErr(err) | ||||||
|  |  | ||||||
|  | 		// Exit the application. | ||||||
| 		os.Exit(0) | 		os.Exit(0) | ||||||
| 	}() | 	}() | ||||||
|  |  | ||||||
|  | 	// Write lock to filesystem to indicate an existing running daemon. | ||||||
|  | 	err = os.WriteFile(lockPath, []byte(fmt.Sprint(os.Getpid())), os.ModePerm) | ||||||
|  | 	checkErr(err) | ||||||
|  |  | ||||||
| 	// Bring Up TUN Device | 	// Bring Up TUN Device | ||||||
| 	tun.Up(cfg.Interface.Name) | 	err = tunDev.Up() | ||||||
|  | 	if err != nil { | ||||||
|  | 		checkErr(errors.New("unable to bring up tun device")) | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	fmt.Println("[+] Network Setup Complete...Waiting on Node Discovery") | 	fmt.Println("[+] Network Setup Complete...Waiting on Node Discovery") | ||||||
| 	// Listen For New Packets on TUN Interface | 	// Listen For New Packets on TUN Interface | ||||||
| 	packet := make([]byte, 1420) | 	activeStreams = make(map[string]network.Stream) | ||||||
| 	var stream network.Stream | 	var packet = make([]byte, 1420) | ||||||
| 	var header *ipv4.Header |  | ||||||
| 	var plen int |  | ||||||
| 	for { | 	for { | ||||||
| 		plen, err = iface.Read(packet) | 		plen, err := tunDev.Iface.Read(packet) | ||||||
| 		checkErr(err) | 		if err != nil { | ||||||
| 		header, _ = ipv4.ParseHeader(packet) | 			log.Println(err) | ||||||
| 		_, ok := cfg.Peers[header.Dst.String()] | 			continue | ||||||
|  | 		} | ||||||
|  | 		dst := net.IPv4(packet[16], packet[17], packet[18], packet[19]).String() | ||||||
|  | 		stream, ok := activeStreams[dst] | ||||||
| 		if ok { | 		if ok { | ||||||
| 			stream, err = host.NewStream(ctx, peerTable[header.Dst.String()], p2p.Protocol) | 			_, err = stream.Write(packet[:plen]) | ||||||
|  | 			if err == nil { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			stream.Close() | ||||||
|  | 			delete(activeStreams, dst) | ||||||
|  | 			ok = false | ||||||
|  | 		} | ||||||
|  | 		if peer, ok := peerTable[dst]; ok { | ||||||
|  | 			stream, err = host.NewStream(ctx, peer, p2p.Protocol) | ||||||
| 			if err != nil { | 			if err != nil { | ||||||
| 				log.Println(err) | 				log.Println(err) | ||||||
| 				continue | 				continue | ||||||
| 			} | 			} | ||||||
| 			stream.Write(packet[:plen]) | 			stream.Write(packet[:plen]) | ||||||
| 			stream.Close() | 			activeStreams[dst] = stream | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func createDaemon(cfg config.Config, out chan<- error) { | func createDaemon(cfg *config.Config, out chan<- error) { | ||||||
| 	path, err := os.Executable() | 	path, err := os.Executable() | ||||||
| 	checkErr(err) | 	checkErr(err) | ||||||
| 	// Create Pipe to monitor for daemon output. | 	// Create Pipe to monitor for daemon output. | ||||||
| @@ -218,6 +248,8 @@ func createDaemon(cfg config.Config, out chan<- error) { | |||||||
| 			count++ | 			count++ | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// Release the created daemon | ||||||
| 	err = process.Release() | 	err = process.Release() | ||||||
| 	checkErr(err) | 	checkErr(err) | ||||||
| 	if count < len(cfg.Peers) { | 	if count < len(cfg.Peers) { | ||||||
| @@ -230,9 +262,18 @@ func streamHandler(stream network.Stream) { | |||||||
| 	// If the remote node ID isn't in the list of known nodes don't respond. | 	// If the remote node ID isn't in the list of known nodes don't respond. | ||||||
| 	if _, ok := RevLookup[stream.Conn().RemotePeer().Pretty()]; !ok { | 	if _, ok := RevLookup[stream.Conn().RemotePeer().Pretty()]; !ok { | ||||||
| 		stream.Reset() | 		stream.Reset() | ||||||
|  | 		return | ||||||
| 	} | 	} | ||||||
| 	io.Copy(iface.ReadWriteCloser, stream) | 	var packet = make([]byte, 1420) | ||||||
|  | 	for { | ||||||
|  | 		plen, err := stream.Read(packet) | ||||||
|  | 		if err != nil { | ||||||
| 			stream.Close() | 			stream.Close() | ||||||
|  | 			delete(activeStreams, RevLookup[stream.Conn().RemotePeer().Pretty()]) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		tunDev.Iface.Write(packet[:plen]) | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| func prettyDiscovery(ctx context.Context, node host.Host, peerTable map[string]peer.ID) { | func prettyDiscovery(ctx context.Context, node host.Host, peerTable map[string]peer.ID) { | ||||||
| @@ -256,3 +297,29 @@ func prettyDiscovery(ctx context.Context, node host.Host, peerTable map[string]p | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func verifyPort(port int) (int, error) { | ||||||
|  | 	var ln net.Listener | ||||||
|  | 	var err error | ||||||
|  | 	if port != 8001 { | ||||||
|  | 		ln, err = net.Listen("tcp", ":"+strconv.Itoa(port)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return port, errors.New("could not create node, listen port already in use by something else") | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		for { | ||||||
|  | 			ln, err = net.Listen("tcp", ":"+strconv.Itoa(port)) | ||||||
|  | 			if err == nil { | ||||||
|  | 				break | ||||||
|  | 			} | ||||||
|  | 			if port >= 65535 { | ||||||
|  | 				return port, errors.New("failed to find open port") | ||||||
|  | 			} | ||||||
|  | 			port++ | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if ln != nil { | ||||||
|  | 		ln.Close() | ||||||
|  | 	} | ||||||
|  | 	return port, nil | ||||||
|  | } | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ import ( | |||||||
|  |  | ||||||
| // Config is the main Configuration Struct for Hyprspace. | // Config is the main Configuration Struct for Hyprspace. | ||||||
| type Config struct { | type Config struct { | ||||||
|  | 	Path      string          `yaml:"path,omitempty"` | ||||||
| 	Interface Interface       `yaml:"interface"` | 	Interface Interface       `yaml:"interface"` | ||||||
| 	Peers     map[string]Peer `yaml:"peers"` | 	Peers     map[string]Peer `yaml:"peers"` | ||||||
| } | } | ||||||
| @@ -27,12 +28,12 @@ type Peer struct { | |||||||
| } | } | ||||||
|  |  | ||||||
| // Read initializes a config from a file. | // Read initializes a config from a file. | ||||||
| func Read(path string) (result Config, err error) { | func Read(path string) (*Config, error) { | ||||||
| 	in, err := os.ReadFile(path) | 	in, err := os.ReadFile(path) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return | 		return nil, err | ||||||
| 	} | 	} | ||||||
| 	result = Config{ | 	result := Config{ | ||||||
| 		Interface: Interface{ | 		Interface: Interface{ | ||||||
| 			Name:       "hs0", | 			Name:       "hs0", | ||||||
| 			ListenPort: 8001, | 			ListenPort: 8001, | ||||||
| @@ -41,6 +42,14 @@ func Read(path string) (result Config, err error) { | |||||||
| 			PrivateKey: "", | 			PrivateKey: "", | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// Read in config settings from file. | ||||||
| 	err = yaml.Unmarshal(in, &result) | 	err = yaml.Unmarshal(in, &result) | ||||||
| 	return | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Overwrite path of config to input. | ||||||
|  | 	result.Path = path | ||||||
|  | 	return &result, nil | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								go.mod
									
									
									
									
									
								
							| @@ -16,7 +16,7 @@ require ( | |||||||
| 	github.com/multiformats/go-multiaddr v0.3.3 | 	github.com/multiformats/go-multiaddr v0.3.3 | ||||||
| 	github.com/songgao/water v0.0.0-20200317203138-2b4b6d7c09d8 | 	github.com/songgao/water v0.0.0-20200317203138-2b4b6d7c09d8 | ||||||
| 	github.com/tcnksm/go-latest v0.0.0-20170313132115-e3007ae9052e | 	github.com/tcnksm/go-latest v0.0.0-20170313132115-e3007ae9052e | ||||||
| 	golang.org/x/net v0.0.0-20210428140749-89ef3d95e781 | 	github.com/vishvananda/netlink v1.1.0 | ||||||
| 	golang.org/x/sys v0.0.0-20210603125802-9665404d3644 // indirect | 	golang.org/x/sys v0.0.0-20210603125802-9665404d3644 // indirect | ||||||
| 	gopkg.in/yaml.v2 v2.4.0 | 	gopkg.in/yaml.v2 v2.4.0 | ||||||
| 	honnef.co/go/tools v0.0.1-2020.1.4 // indirect | 	honnef.co/go/tools v0.0.1-2020.1.4 // indirect | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								go.sum
									
									
									
									
									
								
							| @@ -879,6 +879,10 @@ github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijb | |||||||
| github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= | ||||||
| github.com/viant/assertly v0.4.8/go.mod h1:aGifi++jvCrUaklKEKT0BU95igDNaqkvz+49uaYMPRU= | github.com/viant/assertly v0.4.8/go.mod h1:aGifi++jvCrUaklKEKT0BU95igDNaqkvz+49uaYMPRU= | ||||||
| github.com/viant/toolbox v0.24.0/go.mod h1:OxMCG57V0PXuIP2HNQrtJf2CjqdmbrOx5EkMILuUhzM= | github.com/viant/toolbox v0.24.0/go.mod h1:OxMCG57V0PXuIP2HNQrtJf2CjqdmbrOx5EkMILuUhzM= | ||||||
|  | github.com/vishvananda/netlink v1.1.0 h1:1iyaYNBLmP6L0220aDnYQpo1QEV4t4hJ+xEEhhJH8j0= | ||||||
|  | github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE= | ||||||
|  | github.com/vishvananda/netns v0.0.0-20191106174202-0a2b9b5464df h1:OviZH7qLw/7ZovXvuNyL3XQl8UFofeikI1NW1Gypu7k= | ||||||
|  | github.com/vishvananda/netns v0.0.0-20191106174202-0a2b9b5464df/go.mod h1:JP3t17pCcGlemwknint6hfoeCVQrEMVwxRLRjXpq+BU= | ||||||
| github.com/wangjia184/sortedset v0.0.0-20160527075905-f5d03557ba30/go.mod h1:YkocrP2K2tcw938x9gCOmT5G5eCD6jsTz0SZuyAqwIE= | github.com/wangjia184/sortedset v0.0.0-20160527075905-f5d03557ba30/go.mod h1:YkocrP2K2tcw938x9gCOmT5G5eCD6jsTz0SZuyAqwIE= | ||||||
| github.com/whyrusleeping/go-keyspace v0.0.0-20160322163242-5b898ac5add1 h1:EKhdznlJHPMoKr0XTrX+IlJs1LH3lyx2nfr1dOlZ79k= | github.com/whyrusleeping/go-keyspace v0.0.0-20160322163242-5b898ac5add1 h1:EKhdznlJHPMoKr0XTrX+IlJs1LH3lyx2nfr1dOlZ79k= | ||||||
| github.com/whyrusleeping/go-keyspace v0.0.0-20160322163242-5b898ac5add1/go.mod h1:8UvriyWtv5Q5EOgjHaSseUEdkQfvwFv1I/In/O2M9gc= | github.com/whyrusleeping/go-keyspace v0.0.0-20160322163242-5b898ac5add1/go.mod h1:8UvriyWtv5Q5EOgjHaSseUEdkQfvwFv1I/In/O2M9gc= | ||||||
| @@ -1042,6 +1046,7 @@ golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7w | |||||||
| golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20190526052359-791d8a0f4d09/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20190526052359-791d8a0f4d09/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
|  | golang.org/x/sys v0.0.0-20190606203320-7fc4e5ec1444/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20190626221950-04f50cda93cb/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20190626221950-04f50cda93cb/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
| golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								hyprspace.png
									
									
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								hyprspace.png
									
									
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 145 KiB After Width: | Height: | Size: 604 KiB | 
							
								
								
									
										21
									
								
								tun/options.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								tun/options.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | package tun | ||||||
|  |  | ||||||
|  | type Option func(tun *TUN) error | ||||||
|  |  | ||||||
|  | func Address(address string) Option { | ||||||
|  | 	return func(tun *TUN) error { | ||||||
|  | 		return tun.setAddress(address) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func MTU(mtu int) Option { | ||||||
|  | 	return func(tun *TUN) error { | ||||||
|  | 		return tun.setMTU(mtu) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func DestAddress(address string) Option { | ||||||
|  | 	return func(tun *TUN) error { | ||||||
|  | 		return tun.setDestAddress(address) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										62
									
								
								tun/tun.go
									
									
									
									
									
								
							
							
						
						
									
										62
									
								
								tun/tun.go
									
									
									
									
									
								
							| @@ -1,52 +1,22 @@ | |||||||
| package tun | package tun | ||||||
|  |  | ||||||
| import ( | import "github.com/songgao/water" | ||||||
| 	"fmt" |  | ||||||
| 	"os/exec" |  | ||||||
|  |  | ||||||
| 	"github.com/songgao/water" | type TUN struct { | ||||||
| ) | 	Iface *water.Interface | ||||||
|  | 	MTU   int | ||||||
|  | 	Src   string | ||||||
|  | 	Dst   string | ||||||
|  | } | ||||||
|  |  | ||||||
| // New creates and returns a new TUN interface for the application. | func (t *TUN) Apply(opts ...Option) error { | ||||||
| func New(name string) (result *water.Interface, err error) { | 	for _, opt := range opts { | ||||||
| 	// Setup TUN Config | 		if opt == nil { | ||||||
| 	cfg := water.Config{ | 			continue | ||||||
| 		DeviceType: water.TUN, |  | ||||||
| 		} | 		} | ||||||
| 	cfg.Name = name | 		if err := opt(t); err != nil { | ||||||
| 	// Create TUN Interface | 			return err | ||||||
| 	result, err = water.New(cfg) | 		} | ||||||
| 	return | 	} | ||||||
| } | 	return nil | ||||||
|  |  | ||||||
| // SetMTU sets the Maximum Tansmission Unit Size for a |  | ||||||
| // Packet on the interface. |  | ||||||
| func SetMTU(name string, mtu int) (err error) { |  | ||||||
| 	return ip("link", "set", "dev", name, "mtu", fmt.Sprintf("%d", mtu)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // SetAddress sets the interface's known address and subnet. |  | ||||||
| func SetAddress(name string, address string) (err error) { |  | ||||||
| 	return ip("addr", "add", address, "dev", name) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Up brings up an interface to allow it to start accepting connections. |  | ||||||
| func Up(name string) (err error) { |  | ||||||
| 	return ip("link", "set", "dev", name, "up") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Down brings down an interface stopping active connections. |  | ||||||
| func Down(name string) (err error) { |  | ||||||
| 	return ip("link", "set", "dev", name, "down") |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // Delete removes a TUN device from the host. |  | ||||||
| func Delete(name string) (err error) { |  | ||||||
| 	return ip("link", "delete", "dev", name) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func ip(args ...string) (err error) { |  | ||||||
| 	cmd := exec.Command("ip", args...) |  | ||||||
| 	err = cmd.Run() |  | ||||||
| 	return |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										72
									
								
								tun/tun_darwin.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								tun/tun_darwin.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,72 @@ | |||||||
|  | //go:build darwin | ||||||
|  | // +build darwin | ||||||
|  |  | ||||||
|  | package tun | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"os/exec" | ||||||
|  |  | ||||||
|  | 	"github.com/songgao/water" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // New creates and returns a new TUN interface for the application. | ||||||
|  | func New(name string, opts ...Option) (*TUN, error) { | ||||||
|  | 	// Setup TUN Config | ||||||
|  | 	cfg := water.Config{ | ||||||
|  | 		DeviceType: water.TUN, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Create Water Interface | ||||||
|  | 	iface, err := water.New(cfg) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Create TUN result struct | ||||||
|  | 	result := TUN{ | ||||||
|  | 		Iface: iface, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Apply options to set TUN config values | ||||||
|  | 	err = result.Apply(opts...) | ||||||
|  | 	return &result, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetMTU sets the Maximum Tansmission Unit Size for a | ||||||
|  | // Packet on the interface. | ||||||
|  | func (t *TUN) setMTU(mtu int) error { | ||||||
|  | 	return ifconfig(t.Iface.Name(), "mtu", fmt.Sprintf("%d", mtu)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetDestAddress sets the interface's address. | ||||||
|  | func (t *TUN) setAddress(address string) error { | ||||||
|  | 	t.Src = address | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetDestAddress sets the interface's address. | ||||||
|  | func (t *TUN) setDestAddress(address string) error { | ||||||
|  | 	t.Dst = address | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Up brings up an interface to allow it to start accepting connections. | ||||||
|  | func (t *TUN) Up() error { | ||||||
|  | 	return ifconfig(t.Iface.Name(), "inet", t.Src, t.Dst, "up") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Down brings down an interface stopping active connections. | ||||||
|  | func (t *TUN) Down() error { | ||||||
|  | 	return ifconfig(t.Iface.Name(), "down") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Delete removes a TUN device from the host. | ||||||
|  | func Delete(name string) error { | ||||||
|  | 	return fmt.Errorf("removing an interface is unsupported under mac") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ifconfig(args ...string) error { | ||||||
|  | 	cmd := exec.Command("ifconfig", args...) | ||||||
|  | 	return cmd.Run() | ||||||
|  | } | ||||||
							
								
								
									
										92
									
								
								tun/tun_linux.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								tun/tun_linux.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | |||||||
|  | //go:build linux | ||||||
|  | // +build linux | ||||||
|  |  | ||||||
|  | package tun | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"errors" | ||||||
|  |  | ||||||
|  | 	"github.com/songgao/water" | ||||||
|  | 	"github.com/vishvananda/netlink" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // New creates and returns a new TUN interface for the application. | ||||||
|  | func New(name string, opts ...Option) (*TUN, error) { | ||||||
|  | 	// Setup TUN Config | ||||||
|  | 	cfg := water.Config{ | ||||||
|  | 		DeviceType: water.TUN, | ||||||
|  | 	} | ||||||
|  | 	cfg.Name = name | ||||||
|  |  | ||||||
|  | 	// Create Water Interface | ||||||
|  | 	iface, err := water.New(cfg) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Create TUN result struct | ||||||
|  | 	result := TUN{ | ||||||
|  | 		Iface: iface, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Apply options to set TUN config values | ||||||
|  | 	err = result.Apply(opts...) | ||||||
|  | 	return &result, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setMTU sets the Maximum Tansmission Unit Size for a | ||||||
|  | // Packet on the interface. | ||||||
|  | func (t *TUN) setMTU(mtu int) error { | ||||||
|  | 	link, err := netlink.LinkByName(t.Iface.Name()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return netlink.LinkSetMTU(link, mtu) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setDestAddress sets the interface's destination address and subnet. | ||||||
|  | func (t *TUN) setAddress(address string) error { | ||||||
|  | 	addr, err := netlink.ParseAddr(address) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	link, err := netlink.LinkByName(t.Iface.Name()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return netlink.AddrAdd(link, addr) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetDestAddress isn't supported under Linux. | ||||||
|  | // You should instead use set address to set the interface to handle | ||||||
|  | // all addresses within a subnet. | ||||||
|  | func (t *TUN) setDestAddress(address string) error { | ||||||
|  | 	return errors.New("destination addresses are not supported under linux") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Up brings up an interface to allow it to start accepting connections. | ||||||
|  | func (t *TUN) Up() error { | ||||||
|  | 	link, err := netlink.LinkByName(t.Iface.Name()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return netlink.LinkSetUp(link) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Down brings down an interface stopping active connections. | ||||||
|  | func (t *TUN) Down() error { | ||||||
|  | 	link, err := netlink.LinkByName(t.Iface.Name()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return netlink.LinkSetDown(link) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Delete removes a TUN device from the host. | ||||||
|  | func Delete(name string) error { | ||||||
|  | 	link, err := netlink.LinkByName(name) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return netlink.LinkDel(link) | ||||||
|  | } | ||||||
							
								
								
									
										127
									
								
								tun/tun_windows.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										127
									
								
								tun/tun_windows.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,127 @@ | |||||||
|  | //go:build windows | ||||||
|  | // +build windows | ||||||
|  |  | ||||||
|  | package tun | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"net" | ||||||
|  | 	"os/exec" | ||||||
|  |  | ||||||
|  | 	"github.com/songgao/water" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // New creates and returns a new TUN interface for the application. | ||||||
|  | func New(name string, opts ...Option) (*TUN, error) { | ||||||
|  | 	result := TUN{} | ||||||
|  |  | ||||||
|  | 	// Apply options early to set struct values for interface creation. | ||||||
|  | 	err := result.Apply(opts...) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TUN on Windows requires address and network to be set on device creation stage | ||||||
|  | 	// We also set network to 0.0.0.0/0 so we able to reach networks behind the node | ||||||
|  | 	// https://github.com/songgao/water/blob/master/params_windows.go | ||||||
|  | 	// https://gitlab.com/openconnect/openconnect/-/blob/master/tun-win32.c | ||||||
|  | 	ip, _, err := net.ParseCIDR(result.Src) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	network := net.IPNet{ | ||||||
|  | 		IP:   ip, | ||||||
|  | 		Mask: net.IPv4Mask(0, 0, 0, 0), | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Setup TUN Config | ||||||
|  | 	cfg := water.Config{ | ||||||
|  | 		DeviceType: water.TUN, | ||||||
|  | 		PlatformSpecificParams: water.PlatformSpecificParams{ | ||||||
|  | 			ComponentID:   "tap0901", | ||||||
|  | 			InterfaceName: name, | ||||||
|  | 			Network:       network.String(), | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Interface should be enabled before creation of water interface | ||||||
|  | 	// Otherwise there will be an error "The system cannot find the file specified." | ||||||
|  | 	netsh("interface", "set", "interface", "name=", name, "enable") | ||||||
|  |  | ||||||
|  | 	// Create Water Interface | ||||||
|  | 	iface, err := water.New(cfg) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Set TUN interface to newly created interface | ||||||
|  | 	result.Iface = iface | ||||||
|  |  | ||||||
|  | 	// Apply options to setup TUN interface configuration | ||||||
|  | 	// Setup interface address | ||||||
|  | 	err = result.setupAddress(result.Src) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Setup interface mtu size | ||||||
|  | 	err = result.setupMTU(result.MTU) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return &result, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setMTU configures the interface's MTU. | ||||||
|  | func (t *TUN) setMTU(mtu int) error { | ||||||
|  | 	t.MTU = mtu | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setAddress configures the interface's address. | ||||||
|  | func (t *TUN) setAddress(address string) error { | ||||||
|  | 	t.Src = address | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setupMTU sets the Maximum Tansmission Unit Size for a | ||||||
|  | // Packet on the interface. | ||||||
|  | func (t *TUN) setupMTU(mtu int) error { | ||||||
|  | 	return netsh("interface", "ipv4", "set", "subinterface", t.Iface.Name(), "mtu=", fmt.Sprintf("%d", mtu)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setupAddress sets the interface's destination address and subnet. | ||||||
|  | func (t *TUN) setupAddress(address string) error { | ||||||
|  | 	return netsh("interface", "ip", "set", "address", "name=", t.Iface.Name(), "static", address) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetDestAddress isn't supported under Windows. | ||||||
|  | // You should instead use set address to set the interface to handle | ||||||
|  | // all addresses within a subnet. | ||||||
|  | func (t *TUN) setDestAddress(address string) error { | ||||||
|  | 	return errors.New("destination addresses are not supported under windows") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Up brings up an interface to allow it to start accepting connections. | ||||||
|  | func (t *TUN) Up() error { | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Down brings down an interface stopping active connections. | ||||||
|  | func (t *TUN) Down() error { | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Delete removes a TUN device from the host. | ||||||
|  | func Delete(name string) error { | ||||||
|  | 	return netsh("interface", "set", "interface", "name=", name, "disable") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func netsh(args ...string) (err error) { | ||||||
|  | 	cmd := exec.Command("netsh", args...) | ||||||
|  | 	err = cmd.Run() | ||||||
|  | 	return | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user