package overlay import ( "errors" "io" "log/slog" "net/netip" "github.com/slackhq/nebula/config" "github.com/slackhq/nebula/overlay/tio" "github.com/slackhq/nebula/routing" ) func NewUserDeviceFromConfig(c *config.C, l *slog.Logger, vpnNetworks []netip.Prefix, routines int) (Device, error) { return NewUserDevice(vpnNetworks) } func NewUserDevice(vpnNetworks []netip.Prefix) (Device, error) { // these pipes guarantee each write/read will match 1:1 or, ow := io.Pipe() ir, iw := io.Pipe() return &UserDevice{ vpnNetworks: vpnNetworks, outboundReader: or, outboundWriter: ow, inboundReader: ir, inboundWriter: iw, }, nil } type UserDevice struct { vpnNetworks []netip.Prefix numReaders int outboundReader *io.PipeReader outboundWriter *io.PipeWriter inboundReader *io.PipeReader inboundWriter *io.PipeWriter readBuf []byte batchRet [1]tio.Packet } func (d *UserDevice) Read() ([]tio.Packet, error) { if d.readBuf == nil { d.readBuf = make([]byte, defaultBatchBufSize) } n, err := d.outboundReader.Read(d.readBuf) if err != nil { return nil, err } d.batchRet[0] = tio.Packet{Bytes: d.readBuf[:n]} return d.batchRet[:], nil } func (d *UserDevice) Activate() error { return nil } func (d *UserDevice) Networks() []netip.Prefix { return d.vpnNetworks } func (d *UserDevice) Name() string { return "faketun0" } func (d *UserDevice) RoutesFor(ip netip.Addr) routing.Gateways { return routing.Gateways{routing.NewGateway(ip, 1)} } func (d *UserDevice) SupportsMultiqueue() bool { return false } func (d *UserDevice) NewMultiQueueReader() error { return errors.New("not implemented") } func (d *UserDevice) Readers() []tio.Queue { return []tio.Queue{d} } func (d *UserDevice) Pipe() (*io.PipeReader, *io.PipeWriter) { return d.inboundReader, d.outboundWriter } func (d *UserDevice) Write(p []byte) (n int, err error) { return d.inboundWriter.Write(p) } func (d *UserDevice) Close() error { d.inboundWriter.Close() d.outboundWriter.Close() return nil }