Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 1 | // Copyright 2020 The Monogon Project Authors. |
| 2 | // |
| 3 | // SPDX-License-Identifier: Apache-2.0 |
| 4 | // |
| 5 | // Licensed under the Apache License, Version 2.0 (the "License"); |
| 6 | // you may not use this file except in compliance with the License. |
| 7 | // You may obtain a copy of the License at |
| 8 | // |
| 9 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 10 | // |
| 11 | // Unless required by applicable law or agreed to in writing, software |
| 12 | // distributed under the License is distributed on an "AS IS" BASIS, |
| 13 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 14 | // See the License for the specific language governing permissions and |
| 15 | // limitations under the License. |
| 16 | |
| 17 | package network |
| 18 | |
| 19 | import ( |
| 20 | "context" |
| 21 | "fmt" |
Lorenz Brun | f042e6f | 2020-06-24 16:46:09 +0200 | [diff] [blame] | 22 | "io/ioutil" |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 23 | "net" |
| 24 | "os" |
| 25 | |
Lorenz Brun | b682ba5 | 2020-07-08 14:51:36 +0200 | [diff] [blame] | 26 | "github.com/google/nftables" |
| 27 | "github.com/google/nftables/expr" |
| 28 | |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 29 | "github.com/vishvananda/netlink" |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 30 | "golang.org/x/sys/unix" |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 31 | |
| 32 | "git.monogon.dev/source/nexantic.git/core/internal/common/supervisor" |
| 33 | "git.monogon.dev/source/nexantic.git/core/internal/network/dhcp" |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 34 | "git.monogon.dev/source/nexantic.git/core/internal/network/dns" |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 35 | "git.monogon.dev/source/nexantic.git/core/pkg/logtree" |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 36 | ) |
| 37 | |
| 38 | const ( |
| 39 | resolvConfPath = "/etc/resolv.conf" |
| 40 | resolvConfSwapPath = "/etc/resolv.conf.new" |
| 41 | ) |
| 42 | |
| 43 | type Service struct { |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 44 | config Config |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 45 | dhcp *dhcp.Client |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 46 | |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 47 | logger logtree.LeveledLogger |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 48 | } |
| 49 | |
| 50 | type Config struct { |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 51 | CorednsRegistrationChan chan *dns.ExtraDirective |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 52 | } |
| 53 | |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 54 | func New(config Config) *Service { |
| 55 | return &Service{ |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 56 | config: config, |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 57 | dhcp: dhcp.New(), |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 58 | } |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 59 | } |
| 60 | |
| 61 | func setResolvconf(nameservers []net.IP, searchDomains []string) error { |
Leopold Schabel | 68c5875 | 2019-11-14 21:00:59 +0100 | [diff] [blame] | 62 | _ = os.Mkdir("/etc", 0755) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 63 | newResolvConf, err := os.Create(resolvConfSwapPath) |
| 64 | if err != nil { |
| 65 | return err |
| 66 | } |
| 67 | defer newResolvConf.Close() |
| 68 | defer os.Remove(resolvConfSwapPath) |
| 69 | for _, ns := range nameservers { |
| 70 | if _, err := newResolvConf.WriteString(fmt.Sprintf("nameserver %v\n", ns)); err != nil { |
| 71 | return err |
| 72 | } |
| 73 | } |
| 74 | for _, searchDomain := range searchDomains { |
| 75 | if _, err := newResolvConf.WriteString(fmt.Sprintf("search %v", searchDomain)); err != nil { |
| 76 | return err |
| 77 | } |
| 78 | } |
| 79 | newResolvConf.Close() |
| 80 | // Atomically swap in new config |
| 81 | return unix.Rename(resolvConfSwapPath, resolvConfPath) |
| 82 | } |
| 83 | |
Serge Bazanski | 1a5a667 | 2020-02-18 10:09:43 +0100 | [diff] [blame] | 84 | func (s *Service) addNetworkRoutes(link netlink.Link, addr net.IPNet, gw net.IP) error { |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 85 | if err := netlink.AddrReplace(link, &netlink.Addr{IPNet: &addr}); err != nil { |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 86 | return fmt.Errorf("failed to add DHCP address to network interface \"%v\": %w", link.Attrs().Name, err) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 87 | } |
Serge Bazanski | 1a5a667 | 2020-02-18 10:09:43 +0100 | [diff] [blame] | 88 | |
| 89 | if gw.IsUnspecified() { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 90 | s.logger.Infof("No default route set, only local network %s will be reachable", addr.String()) |
Serge Bazanski | 1a5a667 | 2020-02-18 10:09:43 +0100 | [diff] [blame] | 91 | return nil |
| 92 | } |
| 93 | |
| 94 | route := &netlink.Route{ |
Lorenz Brun | 45333b6 | 2019-11-11 15:26:27 +0100 | [diff] [blame] | 95 | Dst: &net.IPNet{IP: net.IPv4(0, 0, 0, 0), Mask: net.IPv4Mask(0, 0, 0, 0)}, |
| 96 | Gw: gw, |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 97 | Scope: netlink.SCOPE_UNIVERSE, |
Serge Bazanski | 1a5a667 | 2020-02-18 10:09:43 +0100 | [diff] [blame] | 98 | } |
| 99 | if err := netlink.RouteAdd(route); err != nil { |
| 100 | return fmt.Errorf("could not add default route: netlink.RouteAdd(%+v): %v", route, err) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 101 | } |
| 102 | return nil |
| 103 | } |
| 104 | |
Lorenz Brun | b682ba5 | 2020-07-08 14:51:36 +0200 | [diff] [blame] | 105 | // nfifname converts an interface name into 16 bytes padded with zeroes (for nftables) |
| 106 | func nfifname(n string) []byte { |
| 107 | b := make([]byte, 16) |
| 108 | copy(b, []byte(n+"\x00")) |
| 109 | return b |
| 110 | } |
| 111 | |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 112 | func (s *Service) useInterface(ctx context.Context, iface netlink.Link) error { |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 113 | err := supervisor.Run(ctx, "dhcp", s.dhcp.Run(iface)) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 114 | if err != nil { |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 115 | return err |
| 116 | } |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 117 | status, err := s.dhcp.Status(ctx, true) |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 118 | if err != nil { |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 119 | return fmt.Errorf("could not get DHCP Status: %w", err) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 120 | } |
Lorenz Brun | aa6b734 | 2019-12-12 02:55:02 +0100 | [diff] [blame] | 121 | |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 122 | // We're currently never removing this directive just like we're not removing routes and IPs |
| 123 | s.config.CorednsRegistrationChan <- dns.NewUpstreamDirective(status.DNS) |
Lorenz Brun | aa6b734 | 2019-12-12 02:55:02 +0100 | [diff] [blame] | 124 | |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 125 | if err := s.addNetworkRoutes(iface, status.Address, status.Gateway); err != nil { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 126 | s.logger.Warning("Failed to add routes: %v", err) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 127 | } |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 128 | |
Lorenz Brun | b682ba5 | 2020-07-08 14:51:36 +0200 | [diff] [blame] | 129 | c := nftables.Conn{} |
| 130 | |
| 131 | nat := c.AddTable(&nftables.Table{ |
| 132 | Family: nftables.TableFamilyIPv4, |
| 133 | Name: "nat", |
| 134 | }) |
| 135 | |
| 136 | postrouting := c.AddChain(&nftables.Chain{ |
| 137 | Name: "postrouting", |
| 138 | Hooknum: nftables.ChainHookPostrouting, |
| 139 | Priority: nftables.ChainPriorityNATSource, |
| 140 | Table: nat, |
| 141 | Type: nftables.ChainTypeNAT, |
| 142 | }) |
| 143 | |
| 144 | // Masquerade/SNAT all traffic going out of the external interface |
| 145 | c.AddRule(&nftables.Rule{ |
| 146 | Table: nat, |
| 147 | Chain: postrouting, |
| 148 | Exprs: []expr.Any{ |
| 149 | &expr.Meta{Key: expr.MetaKeyOIFNAME, Register: 1}, |
| 150 | &expr.Cmp{ |
| 151 | Op: expr.CmpOpEq, |
| 152 | Register: 1, |
| 153 | Data: nfifname(iface.Attrs().Name), |
| 154 | }, |
| 155 | &expr.Masq{}, |
| 156 | }, |
| 157 | }) |
| 158 | |
| 159 | if err := c.Flush(); err != nil { |
| 160 | panic(err) |
| 161 | } |
| 162 | |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 163 | return nil |
| 164 | } |
| 165 | |
Lorenz Brun | aa6b734 | 2019-12-12 02:55:02 +0100 | [diff] [blame] | 166 | // GetIP returns the current IP (and optionally waits for one to be assigned) |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 167 | func (s *Service) GetIP(ctx context.Context, wait bool) (*net.IP, error) { |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 168 | status, err := s.dhcp.Status(ctx, wait) |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 169 | if err != nil { |
| 170 | return nil, err |
Lorenz Brun | aa6b734 | 2019-12-12 02:55:02 +0100 | [diff] [blame] | 171 | } |
Lorenz Brun | 52f7f29 | 2020-06-24 16:42:02 +0200 | [diff] [blame] | 172 | return &status.Address.IP, nil |
Lorenz Brun | aa6b734 | 2019-12-12 02:55:02 +0100 | [diff] [blame] | 173 | } |
| 174 | |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 175 | func (s *Service) Run(ctx context.Context) error { |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 176 | logger := supervisor.Logger(ctx) |
| 177 | dnsSvc := dns.New(s.config.CorednsRegistrationChan) |
| 178 | supervisor.Run(ctx, "dns", dnsSvc.Run) |
| 179 | supervisor.Run(ctx, "interfaces", s.runInterfaces) |
| 180 | |
| 181 | if err := ioutil.WriteFile("/proc/sys/net/ipv4/ip_forward", []byte("1\n"), 0644); err != nil { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 182 | logger.Fatalf("Failed to enable IPv4 forwarding: %v", err) |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 183 | } |
| 184 | |
| 185 | // We're handling all DNS requests with CoreDNS, including local ones |
| 186 | if err := setResolvconf([]net.IP{{127, 0, 0, 1}}, []string{}); err != nil { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 187 | logger.Fatalf("Failed to set resolv.conf: %v", err) |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 188 | } |
| 189 | |
| 190 | supervisor.Signal(ctx, supervisor.SignalHealthy) |
| 191 | supervisor.Signal(ctx, supervisor.SignalDone) |
| 192 | return nil |
| 193 | } |
| 194 | |
| 195 | func (s *Service) runInterfaces(ctx context.Context) error { |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 196 | s.logger = supervisor.Logger(ctx) |
Lorenz Brun | fa5c2fc | 2020-09-28 13:32:12 +0200 | [diff] [blame] | 197 | s.logger.Info("Starting network interface management") |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 198 | |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 199 | links, err := netlink.LinkList() |
| 200 | if err != nil { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 201 | s.logger.Fatalf("Failed to list network links: %s", err) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 202 | } |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 203 | |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 204 | var ethernetLinks []netlink.Link |
| 205 | for _, link := range links { |
| 206 | attrs := link.Attrs() |
| 207 | if link.Type() == "device" && len(attrs.HardwareAddr) > 0 { |
| 208 | if len(attrs.HardwareAddr) == 6 { // Ethernet |
| 209 | if attrs.Flags&net.FlagUp != net.FlagUp { |
| 210 | netlink.LinkSetUp(link) // Attempt to take up all ethernet links |
| 211 | } |
| 212 | ethernetLinks = append(ethernetLinks, link) |
| 213 | } else { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 214 | s.logger.Infof("Ignoring non-Ethernet interface %s", attrs.Name) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 215 | } |
Lorenz Brun | 45333b6 | 2019-11-11 15:26:27 +0100 | [diff] [blame] | 216 | } else if link.Attrs().Name == "lo" { |
| 217 | if err := netlink.LinkSetUp(link); err != nil { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 218 | s.logger.Errorf("Failed to bring up loopback interface: %v", err) |
Lorenz Brun | 45333b6 | 2019-11-11 15:26:27 +0100 | [diff] [blame] | 219 | } |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 220 | } |
| 221 | } |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 222 | if len(ethernetLinks) != 1 { |
Serge Bazanski | c735967 | 2020-10-30 16:38:57 +0100 | [diff] [blame^] | 223 | s.logger.Warningf("Network service needs exactly one link, bailing") |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 224 | } else { |
| 225 | link := ethernetLinks[0] |
| 226 | if err := s.useInterface(ctx, link); err != nil { |
| 227 | return fmt.Errorf("failed to bring up link %s: %w", link.Attrs().Name, err) |
| 228 | } |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 229 | } |
Serge Bazanski | cdb8c78 | 2020-02-17 12:34:02 +0100 | [diff] [blame] | 230 | |
Serge Bazanski | b1b742f | 2020-03-24 13:58:19 +0100 | [diff] [blame] | 231 | supervisor.Signal(ctx, supervisor.SignalHealthy) |
| 232 | supervisor.Signal(ctx, supervisor.SignalDone) |
Hendrik Hofstadt | 0d7c91e | 2019-10-23 21:44:47 +0200 | [diff] [blame] | 233 | return nil |
| 234 | } |