router7/cmd/dhcp4/dhcp4.go

88 lines
2.1 KiB
Go
Raw Normal View History

2018-05-28 09:53:54 +02:00
// Binary dhcp4 obtains a DHCPv4 lease, persists it to
2018-05-27 17:30:42 +02:00
// /perm/dhcp4/wire/lease.json and notifies netconfigd.
package main
import (
"encoding/json"
"flag"
"fmt"
2018-05-27 17:30:42 +02:00
"io/ioutil"
"net"
"os"
2018-06-01 22:16:51 +02:00
"os/signal"
2018-05-27 17:30:42 +02:00
"path/filepath"
"syscall"
"time"
"router7/internal/dhcp4"
"router7/internal/notify"
2018-06-01 22:16:51 +02:00
"router7/internal/teelogger"
2018-05-27 17:30:42 +02:00
)
2018-06-01 22:16:51 +02:00
var log = teelogger.NewConsole()
2018-05-27 17:30:42 +02:00
func logic() error {
const leasePath = "/perm/dhcp4/wire/lease.json"
if err := os.MkdirAll(filepath.Dir(leasePath), 0755); err != nil {
2018-05-27 17:30:42 +02:00
return err
}
iface, err := net.InterfaceByName("uplink0")
if err != nil {
return err
}
const ackFn = "/perm/dhcp4/wire/ack"
ack, err := ioutil.ReadFile(ackFn)
if err != nil && !os.IsNotExist(err) {
log.Printf("Loading previous DHCPACK packet from %s: %v", ackFn, err)
}
2018-05-27 17:30:42 +02:00
c := dhcp4.Client{
Interface: iface,
Ack: ack,
2018-05-27 17:30:42 +02:00
}
2018-06-01 22:17:24 +02:00
usr2 := make(chan os.Signal, 1)
signal.Notify(usr2, syscall.SIGUSR2)
2018-05-27 17:30:42 +02:00
for c.ObtainOrRenew() {
if err := c.Err(); err != nil {
log.Printf("Temporary error: %v", err)
2018-06-02 10:15:59 +02:00
time.Sleep(1 * time.Second)
2018-05-27 17:30:42 +02:00
continue
}
log.Printf("lease: %+v", c.Config())
b, err := json.Marshal(c.Config())
if err != nil {
return err
}
if err := ioutil.WriteFile(leasePath, b, 0644); err != nil {
return fmt.Errorf("persisting lease to %s: %v", leasePath, err)
}
if err := ioutil.WriteFile(ackFn, c.Ack, 0644); err != nil {
return fmt.Errorf("persisting DHCPACK to %s: %v", ackFn, err)
2018-05-27 17:30:42 +02:00
}
if err := notify.Process("/user/netconfi", syscall.SIGUSR1); err != nil {
log.Printf("notifying netconfig: %v", err)
}
2018-06-17 19:30:30 +02:00
if err := notify.Process("/user/dyndns", syscall.SIGUSR1); err != nil {
log.Printf("notifying dyndns: %v", err)
}
2018-06-01 22:17:24 +02:00
select {
case <-time.After(time.Until(c.Config().RenewAfter)):
// fallthrough and renew the DHCP lease
case <-usr2:
2018-06-02 10:16:16 +02:00
log.Printf("SIGUSR2 received, sending DHCPRELEASE")
2018-06-01 22:17:24 +02:00
if err := c.Release(); err != nil {
return err
}
os.Exit(125) // quit supervision by gokrazy
}
2018-05-27 17:30:42 +02:00
}
return c.Err() // permanent error
}
func main() {
// TODO: drop privileges, run as separate uid?
flag.Parse()
if err := logic(); err != nil {
log.Fatal(err)
}
}