2021-04-11 02:55:05 +01:00
|
|
|
// Copyright (c) 2021 Tailscale Inc & AUTHORS All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2021-04-29 20:34:34 +01:00
|
|
|
// +build linux freebsd openbsd
|
|
|
|
|
2021-04-11 02:55:05 +01:00
|
|
|
package dns
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"os/exec"
|
2021-04-11 03:37:11 +01:00
|
|
|
"strings"
|
2021-04-11 02:55:05 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// openresolvManager manages DNS configuration using the openresolv
|
|
|
|
// implementation of the `resolvconf` program.
|
|
|
|
type openresolvManager struct{}
|
|
|
|
|
2021-04-12 23:51:37 +01:00
|
|
|
func newOpenresolvManager() (openresolvManager, error) {
|
|
|
|
return openresolvManager{}, nil
|
2021-04-11 02:55:05 +01:00
|
|
|
}
|
|
|
|
|
2021-04-21 00:19:34 +01:00
|
|
|
func (m openresolvManager) deleteTailscaleConfig() error {
|
|
|
|
cmd := exec.Command("resolvconf", "-f", "-d", "tailscale")
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("running %s: %s", cmd, out)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-04-11 02:55:05 +01:00
|
|
|
func (m openresolvManager) SetDNS(config OSConfig) error {
|
2021-04-21 00:19:34 +01:00
|
|
|
if config.IsZero() {
|
|
|
|
return m.deleteTailscaleConfig()
|
|
|
|
}
|
|
|
|
|
2021-04-11 02:55:05 +01:00
|
|
|
var stdin bytes.Buffer
|
|
|
|
writeResolvConf(&stdin, config.Nameservers, config.SearchDomains)
|
|
|
|
|
|
|
|
cmd := exec.Command("resolvconf", "-m", "0", "-x", "-a", "tailscale")
|
|
|
|
cmd.Stdin = &stdin
|
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("running %s: %s", cmd, out)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m openresolvManager) SupportsSplitDNS() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m openresolvManager) GetBaseConfig() (OSConfig, error) {
|
2021-04-11 04:21:05 +01:00
|
|
|
// List the names of all config snippets openresolv is aware
|
|
|
|
// of. Snippets get listed in priority order (most to least),
|
|
|
|
// which we'll exploit later.
|
2021-04-11 03:37:11 +01:00
|
|
|
bs, err := exec.Command("resolvconf", "-i").CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
return OSConfig{}, err
|
|
|
|
}
|
|
|
|
|
2021-04-11 04:21:05 +01:00
|
|
|
// Remove the "tailscale" snippet from the list.
|
2021-04-11 03:37:11 +01:00
|
|
|
args := []string{"-l"}
|
|
|
|
for _, f := range strings.Split(strings.TrimSpace(string(bs)), " ") {
|
|
|
|
if f == "tailscale" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
args = append(args, f)
|
|
|
|
}
|
|
|
|
|
2021-04-11 04:21:05 +01:00
|
|
|
// List all resolvconf snippets except our own, and parse that as
|
|
|
|
// a resolv.conf. This effectively generates a blended config of
|
|
|
|
// "everyone except tailscale", which is what would be in use if
|
|
|
|
// tailscale hadn't set exclusive mode.
|
|
|
|
//
|
|
|
|
// Note that this is not _entirely_ true. To be perfectly correct,
|
|
|
|
// we should be looking for other interfaces marked exclusive that
|
|
|
|
// predated tailscale, and stick to only those. However, in
|
|
|
|
// practice, openresolv uses are generally quite limited, and boil
|
|
|
|
// down to 1-2 DHCP leases, for which the correct outcome is a
|
|
|
|
// blended config like the one we produce here.
|
2021-04-11 03:37:11 +01:00
|
|
|
var buf bytes.Buffer
|
|
|
|
cmd := exec.Command("resolvconf", args...)
|
|
|
|
cmd.Stdout = &buf
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
return OSConfig{}, err
|
|
|
|
}
|
|
|
|
return readResolv(&buf)
|
2021-04-11 02:55:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (m openresolvManager) Close() error {
|
2021-04-21 00:19:34 +01:00
|
|
|
return m.deleteTailscaleConfig()
|
2021-04-11 02:55:05 +01:00
|
|
|
}
|