154 lines
3.4 KiB
Go
154 lines
3.4 KiB
Go
package gost
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
"github.com/go-log/log"
|
|
"github.com/songgao/water"
|
|
)
|
|
|
|
func createTun(cfg TunConfig) (conn net.Conn, itf *net.Interface, err error) {
|
|
ip, ipNet, err := net.ParseCIDR(cfg.Addr)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
ifce, err := water.New(water.Config{
|
|
DeviceType: water.TUN,
|
|
PlatformSpecificParams: water.PlatformSpecificParams{
|
|
ComponentID: "tap0901",
|
|
InterfaceName: cfg.Name,
|
|
Network: cfg.Addr,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
cmd := fmt.Sprintf("netsh interface ip set address name=\"%s\" "+
|
|
"source=static addr=%s mask=%s gateway=none",
|
|
ifce.Name(), ip.String(), ipMask(ipNet.Mask))
|
|
log.Log("[tun]", cmd)
|
|
args := strings.Split(cmd, " ")
|
|
if er := exec.Command(args[0], args[1:]...).Run(); er != nil {
|
|
err = fmt.Errorf("%s: %v", cmd, er)
|
|
return
|
|
}
|
|
|
|
if err = addTunRoutes(ifce.Name(), cfg.Gateway, cfg.Routes...); err != nil {
|
|
return
|
|
}
|
|
|
|
itf, err = net.InterfaceByName(ifce.Name())
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
conn = &tunTapConn{
|
|
ifce: ifce,
|
|
addr: &net.IPAddr{IP: ip},
|
|
}
|
|
return
|
|
}
|
|
|
|
func createTap(cfg TapConfig) (conn net.Conn, itf *net.Interface, err error) {
|
|
ip, ipNet, _ := net.ParseCIDR(cfg.Addr)
|
|
|
|
ifce, err := water.New(water.Config{
|
|
DeviceType: water.TAP,
|
|
PlatformSpecificParams: water.PlatformSpecificParams{
|
|
ComponentID: "tap0901",
|
|
InterfaceName: cfg.Name,
|
|
Network: cfg.Addr,
|
|
},
|
|
})
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
if ip != nil && ipNet != nil {
|
|
cmd := fmt.Sprintf("netsh interface ip set address name=\"%s\" "+
|
|
"source=static addr=%s mask=%s gateway=none",
|
|
ifce.Name(), ip.String(), ipMask(ipNet.Mask))
|
|
log.Log("[tap]", cmd)
|
|
args := strings.Split(cmd, " ")
|
|
if er := exec.Command(args[0], args[1:]...).Run(); er != nil {
|
|
err = fmt.Errorf("%s: %v", cmd, er)
|
|
return
|
|
}
|
|
}
|
|
|
|
if err = addTapRoutes(ifce.Name(), cfg.Gateway, cfg.Routes...); err != nil {
|
|
return
|
|
}
|
|
|
|
itf, err = net.InterfaceByName(ifce.Name())
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
conn = &tunTapConn{
|
|
ifce: ifce,
|
|
addr: &net.IPAddr{IP: ip},
|
|
}
|
|
return
|
|
}
|
|
|
|
func addTunRoutes(ifName string, gw string, routes ...IPRoute) error {
|
|
for _, route := range routes {
|
|
if route.Dest == nil {
|
|
continue
|
|
}
|
|
|
|
deleteRoute(ifName, route.Dest.String())
|
|
|
|
cmd := fmt.Sprintf("netsh interface ip add route prefix=%s interface=\"%s\" store=active",
|
|
route.Dest.String(), ifName)
|
|
if gw != "" {
|
|
cmd += " nexthop=" + gw
|
|
}
|
|
log.Logf("[tun] %s", cmd)
|
|
args := strings.Split(cmd, " ")
|
|
if er := exec.Command(args[0], args[1:]...).Run(); er != nil {
|
|
return fmt.Errorf("%s: %v", cmd, er)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func addTapRoutes(ifName string, gw string, routes ...string) error {
|
|
for _, route := range routes {
|
|
if route == "" {
|
|
continue
|
|
}
|
|
|
|
deleteRoute(ifName, route)
|
|
|
|
cmd := fmt.Sprintf("netsh interface ip add route prefix=%s interface=\"%s\" store=active",
|
|
route, ifName)
|
|
if gw != "" {
|
|
cmd += " nexthop=" + gw
|
|
}
|
|
log.Logf("[tap] %s", cmd)
|
|
args := strings.Split(cmd, " ")
|
|
if er := exec.Command(args[0], args[1:]...).Run(); er != nil {
|
|
return fmt.Errorf("%s: %v", cmd, er)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func deleteRoute(ifName string, route string) error {
|
|
cmd := fmt.Sprintf("netsh interface ip delete route prefix=%s interface=\"%s\" store=active",
|
|
route, ifName)
|
|
args := strings.Split(cmd, " ")
|
|
return exec.Command(args[0], args[1:]...).Run()
|
|
}
|
|
|
|
func ipMask(mask net.IPMask) string {
|
|
return fmt.Sprintf("%d.%d.%d.%d", mask[0], mask[1], mask[2], mask[3])
|
|
}
|