SOCKS5 proxy

This commit is contained in:
Neil Alexander 2022-02-13 22:20:44 +00:00 committed by Vasyl Gello
parent b613994f40
commit 8ac5443047
2 changed files with 60 additions and 7 deletions

View file

@ -1,6 +1,7 @@
package main package main
import ( import (
"context"
"crypto/ed25519" "crypto/ed25519"
"encoding/hex" "encoding/hex"
"encoding/json" "encoding/json"
@ -15,6 +16,7 @@ import (
"github.com/gologme/log" "github.com/gologme/log"
gsyslog "github.com/hashicorp/go-syslog" gsyslog "github.com/hashicorp/go-syslog"
"github.com/hjson/hjson-go" "github.com/hjson/hjson-go"
"github.com/things-go/go-socks5"
"github.com/yggdrasil-network/yggdrasil-go/contrib/netstack" "github.com/yggdrasil-network/yggdrasil-go/contrib/netstack"
"github.com/yggdrasil-network/yggdrasil-go/src/address" "github.com/yggdrasil-network/yggdrasil-go/src/address"
@ -27,8 +29,19 @@ import (
_ "net/http/pprof" _ "net/http/pprof"
) )
type nameResolver struct{}
func (r *nameResolver) Resolve(ctx context.Context, name string) (context.Context, net.IP, error) {
ip := net.ParseIP(name)
if ip == nil {
return nil, nil, fmt.Errorf("not a valid IP address")
}
return ctx, ip, nil
}
// The main function is responsible for configuring and starting Yggdrasil. // The main function is responsible for configuring and starting Yggdrasil.
func main() { func main() {
socks := flag.String("socks", "", "address to listen on for SOCKS, i.e. :1080")
args := setup.ParseArguments() args := setup.ParseArguments()
// Create a new logger that logs output to stdout. // Create a new logger that logs output to stdout.
@ -148,12 +161,21 @@ func main() {
logger.Fatalln(err) logger.Fatalln(err)
} }
if *socks != "" {
resolver := &nameResolver{}
server := socks5.NewServer(
socks5.WithDial(s.DialContext),
socks5.WithResolver(resolver),
)
go server.ListenAndServe("tcp", *socks) // nolint:errcheck
}
listener, err := s.ListenTCP(&net.TCPAddr{Port: 80}) listener, err := s.ListenTCP(&net.TCPAddr{Port: 80})
if err != nil { if err != nil {
log.Panicln(err) log.Panicln(err)
} }
http.HandleFunc("/", func(writer http.ResponseWriter, request *http.Request) { http.HandleFunc("/", func(writer http.ResponseWriter, request *http.Request) {
_, _ = io.WriteString(writer, "Hello from userspace TCP "+request.RemoteAddr) _, _ = io.WriteString(writer, "I am Yggstack!")
}) })
httpServer := &http.Server{} httpServer := &http.Server{}
go httpServer.Serve(listener) // nolint:errcheck go httpServer.Serve(listener) // nolint:errcheck

View file

@ -1,8 +1,10 @@
package netstack package netstack
import ( import (
"context"
"fmt" "fmt"
"net" "net"
"strconv"
"github.com/yggdrasil-network/yggdrasil-go/src/core" "github.com/yggdrasil-network/yggdrasil-go/src/core"
@ -38,30 +40,59 @@ func CreateYggdrasilNetstack(ygg *core.Core) (*YggdrasilNetstack, error) {
return s, nil return s, nil
} }
func convertToFullAddr(ip net.IP, port int) (tcpip.FullAddress, tcpip.NetworkProtocolNumber) { func convertToFullAddr(ip net.IP, port int) (tcpip.FullAddress, tcpip.NetworkProtocolNumber, error) {
return tcpip.FullAddress{ return tcpip.FullAddress{
NIC: 1, NIC: 1,
Addr: tcpip.Address(ip), Addr: tcpip.Address(ip),
Port: uint16(port), Port: uint16(port),
}, ipv6.ProtocolNumber }, ipv6.ProtocolNumber, nil
}
func convertToFullAddrFromString(endpoint string) (tcpip.FullAddress, tcpip.NetworkProtocolNumber, error) {
host, port, err := net.SplitHostPort(endpoint)
if err != nil {
return tcpip.FullAddress{}, 0, fmt.Errorf("net.SplitHostPort: %w", err)
}
pn := 80
if port != "" {
if pn, err = strconv.Atoi(port); err != nil {
return tcpip.FullAddress{}, 0, fmt.Errorf("strconv.Atoi: %w", err)
}
}
return convertToFullAddr(net.ParseIP(host), pn)
}
func (s *YggdrasilNetstack) DialContext(ctx context.Context, network, address string) (net.Conn, error) {
fa, pn, err := convertToFullAddrFromString(address)
if err != nil {
return nil, fmt.Errorf("convertToFullAddrFromString: %w", err)
}
switch network {
case "tcp", "tcp6":
return gonet.DialContextTCP(ctx, s.stack, fa, pn)
case "udp", "udp6":
return gonet.DialUDP(s.stack, nil, &fa, pn)
default:
return nil, fmt.Errorf("not supported")
}
} }
func (s *YggdrasilNetstack) DialTCP(addr *net.TCPAddr) (net.Conn, error) { func (s *YggdrasilNetstack) DialTCP(addr *net.TCPAddr) (net.Conn, error) {
fa, pn := convertToFullAddr(addr.IP, addr.Port) fa, pn, _ := convertToFullAddr(addr.IP, addr.Port)
return gonet.DialTCP(s.stack, fa, pn) return gonet.DialTCP(s.stack, fa, pn)
} }
func (s *YggdrasilNetstack) DialUDP(addr *net.UDPAddr) (net.PacketConn, error) { func (s *YggdrasilNetstack) DialUDP(addr *net.UDPAddr) (net.PacketConn, error) {
fa, pn := convertToFullAddr(addr.IP, addr.Port) fa, pn, _ := convertToFullAddr(addr.IP, addr.Port)
return gonet.DialUDP(s.stack, nil, &fa, pn) return gonet.DialUDP(s.stack, nil, &fa, pn)
} }
func (s *YggdrasilNetstack) ListenTCP(addr *net.TCPAddr) (net.Listener, error) { func (s *YggdrasilNetstack) ListenTCP(addr *net.TCPAddr) (net.Listener, error) {
fa, pn := convertToFullAddr(addr.IP, addr.Port) fa, pn, _ := convertToFullAddr(addr.IP, addr.Port)
return gonet.ListenTCP(s.stack, fa, pn) return gonet.ListenTCP(s.stack, fa, pn)
} }
func (s *YggdrasilNetstack) ListenUDP(addr *net.UDPAddr) (net.PacketConn, error) { func (s *YggdrasilNetstack) ListenUDP(addr *net.UDPAddr) (net.PacketConn, error) {
fa, pn := convertToFullAddr(addr.IP, addr.Port) fa, pn, _ := convertToFullAddr(addr.IP, addr.Port)
return gonet.DialUDP(s.stack, &fa, nil, pn) return gonet.DialUDP(s.stack, &fa, nil, pn)
} }