mirror of
https://github.com/kubenetworks/kubevpn.git
synced 2025-12-24 11:51:13 +08:00
106 lines
2.2 KiB
Go
106 lines
2.2 KiB
Go
//go:build linux
|
|
|
|
package tun
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"os"
|
|
"syscall"
|
|
|
|
"github.com/containernetworking/cni/pkg/types"
|
|
"github.com/docker/libcontainer/netlink"
|
|
log "github.com/sirupsen/logrus"
|
|
"golang.zx2c4.com/wireguard/tun"
|
|
|
|
"github.com/wencaiwulue/kubevpn/pkg/config"
|
|
)
|
|
|
|
func createTun(cfg Config) (conn net.Conn, itf *net.Interface, err error) {
|
|
ip, ipNet, err := net.ParseCIDR(cfg.Addr)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
mtu := cfg.MTU
|
|
if mtu <= 0 {
|
|
mtu = config.DefaultMTU
|
|
}
|
|
|
|
var device tun.Device
|
|
device, err = tun.CreateTUN("utun", mtu)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
var name string
|
|
name, err = device.Name()
|
|
if err != nil {
|
|
return
|
|
}
|
|
ifc, err := net.InterfaceByName(name)
|
|
if err != nil {
|
|
err = fmt.Errorf("could not find interface name: %s", err)
|
|
return
|
|
}
|
|
|
|
cmd := fmt.Sprintf("ip link set dev %s mtu %d", name, mtu)
|
|
log.Debugf("[tun] %s", cmd)
|
|
if er := netlink.NetworkSetMTU(ifc, mtu); er != nil {
|
|
err = fmt.Errorf("%s: %v", cmd, er)
|
|
return
|
|
}
|
|
|
|
cmd = fmt.Sprintf("ip address add %s dev %s", cfg.Addr, name)
|
|
log.Debugf("[tun] %s", cmd)
|
|
if er := netlink.NetworkLinkAddIp(ifc, ip, ipNet); er != nil {
|
|
err = fmt.Errorf("%s: %v", cmd, er)
|
|
return
|
|
}
|
|
|
|
cmd = fmt.Sprintf("ip link set dev %s up", name)
|
|
log.Debugf("[tun] %s", cmd)
|
|
if er := netlink.NetworkLinkUp(ifc); er != nil {
|
|
err = fmt.Errorf("%s: %v", cmd, er)
|
|
return
|
|
}
|
|
|
|
if err = os.Setenv(config.EnvTunNameOrLUID, name); err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
if err = addTunRoutes(name, cfg.Routes...); err != nil {
|
|
return
|
|
}
|
|
|
|
itf, err = net.InterfaceByName(name)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
conn = &tunConn{
|
|
ifce: device,
|
|
addr: &net.IPAddr{IP: ip},
|
|
}
|
|
return
|
|
}
|
|
|
|
func addTunRoutes(ifName string, routes ...types.Route) error {
|
|
for _, route := range routes {
|
|
if route.Dst.String() == "" {
|
|
continue
|
|
}
|
|
cmd := fmt.Sprintf("ip route add %s dev %s", route.Dst.String(), ifName)
|
|
log.Debugf("[tun] %s", cmd)
|
|
if err := netlink.AddRoute(route.Dst.String(), "", "", ifName); err != nil && !errors.Is(err, syscall.EEXIST) {
|
|
return fmt.Errorf("%s: %v", cmd, err)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func getInterface() (*net.Interface, error) {
|
|
return net.InterfaceByName(os.Getenv(config.EnvTunNameOrLUID))
|
|
}
|