checkpoint
This commit is contained in:
parent
25642eff64
commit
9eb32a3174
|
@ -5,7 +5,10 @@ go_library(
|
||||||
srcs = ["main.go"],
|
srcs = ["main.go"],
|
||||||
importpath = "git.wntrmute.dev/kyle/kdhcp/cmd/kdhcpd",
|
importpath = "git.wntrmute.dev/kyle/kdhcp/cmd/kdhcpd",
|
||||||
visibility = ["//visibility:private"],
|
visibility = ["//visibility:private"],
|
||||||
deps = ["//server"],
|
deps = [
|
||||||
|
"//log",
|
||||||
|
"//server",
|
||||||
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
go_binary(
|
go_binary(
|
||||||
|
|
|
@ -2,18 +2,33 @@ package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"flag"
|
"flag"
|
||||||
"log"
|
|
||||||
|
|
||||||
|
"git.wntrmute.dev/kyle/kdhcp/log"
|
||||||
"git.wntrmute.dev/kyle/kdhcp/server"
|
"git.wntrmute.dev/kyle/kdhcp/server"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
cfg := &server.Config{}
|
cfg := server.DefaultConfig()
|
||||||
flag.StringVar(&cfg.Device, "i", "eth0", "network `interface` to listen on")
|
var level, tag string
|
||||||
|
flag.StringVar(&level, "l", "DEBUG", "log level") // TODO(kyle): change this warning later
|
||||||
|
flag.IntVar(&cfg.Port, "p", cfg.Port, "port to listen on")
|
||||||
|
flag.StringVar(&tag, "t", "kdhcpd", "logging tag")
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
_, err := server.NewServer(cfg)
|
log.Setup(level, tag)
|
||||||
|
|
||||||
|
srv, err := server.NewServer(cfg)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
packet, err := srv.ReadFrom()
|
||||||
|
if err != nil {
|
||||||
|
log.Warning(err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf("receive %d byte packet from %s", len(packet.Data), packet.Addr)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,5 +5,5 @@ go_library(
|
||||||
srcs = ["logger.go"],
|
srcs = ["logger.go"],
|
||||||
importpath = "git.wntrmute.dev/kyle/kdhcp/log",
|
importpath = "git.wntrmute.dev/kyle/kdhcp/log",
|
||||||
visibility = ["//visibility:public"],
|
visibility = ["//visibility:public"],
|
||||||
deps = ["@com_github_hashicorp_go_syslog//:go-syslog"],
|
deps = ["//bazel-kdhcp/external/com_github_hashicorp_go_syslog:go-syslog"],
|
||||||
)
|
)
|
||||||
|
|
|
@ -2,6 +2,8 @@ package log
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
gsyslog "github.com/hashicorp/go-syslog"
|
gsyslog "github.com/hashicorp/go-syslog"
|
||||||
|
@ -13,6 +15,10 @@ type logger struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (log *logger) printf(p gsyslog.Priority, format string, args ...interface{}) {
|
func (log *logger) printf(p gsyslog.Priority, format string, args ...interface{}) {
|
||||||
|
if !strings.HasSuffix(format, "\n") {
|
||||||
|
format += "\n"
|
||||||
|
}
|
||||||
|
|
||||||
if p <= log.p {
|
if p <= log.p {
|
||||||
fmt.Printf("%s [%s] ", prioritiev[p], timestamp())
|
fmt.Printf("%s [%s] ", prioritiev[p], timestamp())
|
||||||
fmt.Printf(format, args...)
|
fmt.Printf(format, args...)
|
||||||
|
@ -158,34 +164,45 @@ func Emergln(args ...interface{}) {
|
||||||
log.println(gsyslog.LOG_EMERG, args...)
|
log.println(gsyslog.LOG_EMERG, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Debugf(args ...interface{}) {
|
func Debugf(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_DEBUG, args...)
|
log.printf(gsyslog.LOG_DEBUG, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Infof(args ...interface{}) {
|
func Infof(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_INFO, args...)
|
log.printf(gsyslog.LOG_INFO, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Noticef(args ...interface{}) {
|
func Noticef(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_NOTICE, args...)
|
log.printf(gsyslog.LOG_NOTICE, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Warningf(args ...interface{}) {
|
func Warningf(format string, args ...interface{}) {
|
||||||
log.print(gsyslog.LOG_WARNING, args...)
|
log.printf(gsyslog.LOG_WARNING, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Errf(args ...interface{}) {
|
func Errf(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_ERR, args...)
|
log.printf(gsyslog.LOG_ERR, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Critf(args ...interface{}) {
|
func Critf(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_CRIT, args...)
|
log.printf(gsyslog.LOG_CRIT, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Alertf(args ...interface{}) {
|
func Alertf(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_ALERT, args...)
|
log.printf(gsyslog.LOG_ALERT, format, args...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func Emergf(args ...interface{}) {
|
func Emergf(format string, args ...interface{}) {
|
||||||
log.printf(gsyslog.LOG_EMERG, args...)
|
log.printf(gsyslog.LOG_EMERG, format, args...)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Fatal(args ...interface{}) {
|
||||||
|
log.println(gsyslog.LOG_ERR, args...)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func Fatalf(format string, args ...interface{}) {
|
||||||
|
log.printf(gsyslog.LOG_ERR, format, args...)
|
||||||
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,8 +2,16 @@ load("@io_bazel_rules_go//go:def.bzl", "go_library")
|
||||||
|
|
||||||
go_library(
|
go_library(
|
||||||
name = "server",
|
name = "server",
|
||||||
srcs = ["server.go"],
|
srcs = [
|
||||||
|
"addr.go",
|
||||||
|
"config.go",
|
||||||
|
"packet.go",
|
||||||
|
"server.go",
|
||||||
|
],
|
||||||
importpath = "git.wntrmute.dev/kyle/kdhcp/server",
|
importpath = "git.wntrmute.dev/kyle/kdhcp/server",
|
||||||
visibility = ["//visibility:public"],
|
visibility = ["//visibility:public"],
|
||||||
deps = ["@com_github_davecgh_go_spew//spew"],
|
deps = [
|
||||||
|
"//bazel-kdhcp/external/com_github_davecgh_go_spew/spew",
|
||||||
|
"//log",
|
||||||
|
],
|
||||||
)
|
)
|
||||||
|
|
|
@ -0,0 +1,61 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net"
|
||||||
|
|
||||||
|
"git.wntrmute.dev/kyle/kdhcp/bazel-kdhcp/external/com_github_davecgh_go_spew/spew"
|
||||||
|
"git.wntrmute.dev/kyle/kdhcp/log"
|
||||||
|
)
|
||||||
|
|
||||||
|
type addr struct {
|
||||||
|
ifi *net.Interface
|
||||||
|
IP net.IP
|
||||||
|
ipn *net.IPNet
|
||||||
|
}
|
||||||
|
|
||||||
|
func addrsForDevice(dev string) ([]addr, error) {
|
||||||
|
netInterface, err := net.InterfaceByName(dev)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("while selecting interface %s: %w", dev, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
spew.Dump(netInterface)
|
||||||
|
|
||||||
|
var addrs []addr
|
||||||
|
devAddrs, err := netInterface.Addrs()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, devAddr := range devAddrs {
|
||||||
|
log.Debugf("consider %s", devAddr.String())
|
||||||
|
ip, ipn, err := net.ParseCIDR(devAddr.String())
|
||||||
|
if err != nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if ip == nil {
|
||||||
|
continue // address isn't an IP address
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf("found IP: %s", ip)
|
||||||
|
ip = ip.To4()
|
||||||
|
|
||||||
|
// DHCP should only listen on private addresses.
|
||||||
|
if !ip.IsPrivate() {
|
||||||
|
log.Debugln("skipping non-private")
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// only support IPv4 for now
|
||||||
|
if len(ip) != 4 {
|
||||||
|
log.Debugf("%d IP, only supporting v4 right now", len(ip))
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
addrs = append(addrs, addr{netInterface, ip, ipn})
|
||||||
|
}
|
||||||
|
|
||||||
|
return addrs, nil
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
const (
|
||||||
|
DefaultPort = 67
|
||||||
|
DefaultMaxPacketSize = 512
|
||||||
|
DefaultNetwork = "udp4"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Config struct {
|
||||||
|
Port int
|
||||||
|
MaxPacketSize int
|
||||||
|
}
|
||||||
|
|
||||||
|
func DefaultConfig() *Config {
|
||||||
|
return &Config{
|
||||||
|
Port: DefaultPort,
|
||||||
|
MaxPacketSize: DefaultMaxPacketSize,
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
package server
|
||||||
|
|
||||||
|
import "net"
|
||||||
|
|
||||||
|
type Packet struct {
|
||||||
|
Data []byte
|
||||||
|
Addr net.Addr
|
||||||
|
}
|
|
@ -1,82 +1,62 @@
|
||||||
package server
|
package server
|
||||||
|
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"fmt"
|
||||||
"log"
|
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
"github.com/davecgh/go-spew/spew"
|
"git.wntrmute.dev/kyle/kdhcp/log"
|
||||||
)
|
)
|
||||||
|
|
||||||
// github.com/insomniacslk/dhcp
|
// github.com/insomniacslk/dhcp
|
||||||
|
|
||||||
type addr struct {
|
|
||||||
IP net.IP
|
|
||||||
ipn *net.IPNet
|
|
||||||
}
|
|
||||||
|
|
||||||
type Config struct {
|
|
||||||
Device string `yaml:"device"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Server struct {
|
type Server struct {
|
||||||
addrs []net.IP
|
cfg *Config
|
||||||
l net.Listener
|
conn net.PacketConn
|
||||||
}
|
}
|
||||||
|
|
||||||
func addrsForDevice(dev string) ([]addr, error) {
|
func (srv *Server) Listen() (err error) {
|
||||||
netInterface, err := net.InterfaceByName(dev)
|
if srv.conn != nil {
|
||||||
|
srv.conn.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf("attempting to set up packet listener on %s 0.0.0.0:%d", DefaultNetwork, srv.cfg.Port)
|
||||||
|
srv.conn, err = net.ListenPacket(DefaultNetwork, fmt.Sprintf(":%d", srv.cfg.Port))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (srv *Server) Close() error {
|
||||||
|
if srv.conn != nil {
|
||||||
|
return srv.conn.Close()
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (srv *Server) ReadFrom() (*Packet, error) {
|
||||||
|
b := make([]byte, srv.cfg.MaxPacketSize)
|
||||||
|
n, addr, err := srv.conn.ReadFrom(b)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
spew.Dump(netInterface)
|
return &Packet{
|
||||||
|
Data: b[:n],
|
||||||
var addrs []addr
|
Addr: addr,
|
||||||
devAddrs, err := netInterface.Addrs()
|
}, nil
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, devAddr := range devAddrs {
|
|
||||||
log.Printf("consider %s", devAddr.String())
|
|
||||||
ip, ipn, err := net.ParseCIDR(devAddr.String())
|
|
||||||
if err != nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if ip == nil {
|
|
||||||
continue // address isn't an IP address
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Printf("found IP: %s", ip)
|
|
||||||
ip = ip.To4()
|
|
||||||
|
|
||||||
// DHCP should only listen on private addresses.
|
|
||||||
if !ip.IsPrivate() {
|
|
||||||
log.Println("skipping non-private")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
// only support IPv4 for now
|
|
||||||
if len(ip) != 4 {
|
|
||||||
log.Printf("%d IP, only supporting v4 right now", len(ip))
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
addrs = append(addrs, addr{ip, ipn})
|
|
||||||
}
|
|
||||||
|
|
||||||
return addrs, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewServer(cfg *Config) (*Server, error) {
|
func NewServer(cfg *Config) (*Server, error) {
|
||||||
addrs, err := addrsForDevice(cfg.Device)
|
srv := &Server{
|
||||||
|
cfg: cfg,
|
||||||
|
}
|
||||||
|
|
||||||
|
err := srv.Listen()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Println("server IP list: ", addrs)
|
return srv, nil
|
||||||
return nil, errors.New("not implemented")
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue