1
0
Fork 0
This commit is contained in:
Arthur K. 2026-05-12 22:54:32 +03:00
commit ca4bd7d7c7
Signed by: wzray
GPG key ID: B97F30FDC4636357
10 changed files with 944 additions and 0 deletions

183
internal/resolv/resolv.go Normal file
View file

@ -0,0 +1,183 @@
package resolv
import (
"bufio"
"errors"
"fmt"
"net"
"os"
"slices"
"strings"
"sync"
)
const DefaultPath = "/run/dns/resolv.conf"
var (
ErrInvalidAddress = errors.New("invalid nameserver address")
ErrAlreadyExists = errors.New("nameserver already exists")
ErrNotFound = errors.New("nameserver not found")
)
type Manager struct {
path string
mu sync.Mutex
}
func New(path string) *Manager {
if path == "" {
path = DefaultPath
}
return &Manager{path: path}
}
func (m *Manager) List() ([]string, error) {
m.mu.Lock()
defer m.mu.Unlock()
_, servers, err := m.read()
if err != nil {
return nil, err
}
return servers, nil
}
func (m *Manager) Add(addr string) error {
addr = strings.TrimSpace(addr)
if !isValidIP(addr) {
return fmt.Errorf("%w: %q", ErrInvalidAddress, addr)
}
m.mu.Lock()
defer m.mu.Unlock()
lines, servers, err := m.read()
if err != nil {
return err
}
if slices.Contains(servers, addr) {
return fmt.Errorf("%w: %s", ErrAlreadyExists, addr)
}
lines = append(lines, "nameserver "+addr)
return m.write(lines)
}
func (m *Manager) Remove(addr string) error {
addr = strings.TrimSpace(addr)
if !isValidIP(addr) {
return fmt.Errorf("%w: %q", ErrInvalidAddress, addr)
}
m.mu.Lock()
defer m.mu.Unlock()
lines, _, err := m.read()
if err != nil {
return err
}
out := make([]string, 0, len(lines))
removed := false
for _, line := range lines {
if !removed && parseNameserver(line) == addr {
removed = true
continue
}
out = append(out, line)
}
if !removed {
return fmt.Errorf("%w: %s", ErrNotFound, addr)
}
return m.write(out)
}
func (m *Manager) read() ([]string, []string, error) {
f, err := os.Open(m.path)
if err != nil {
if os.IsNotExist(err) {
return nil, nil, nil
}
return nil, nil, err
}
defer f.Close()
var lines, servers []string
sc := bufio.NewScanner(f)
for sc.Scan() {
line := sc.Text()
lines = append(lines, line)
if ns := parseNameserver(line); ns != "" {
servers = append(servers, ns)
}
}
if err := sc.Err(); err != nil {
return nil, nil, err
}
return lines, servers, nil
}
func (m *Manager) write(lines []string) error {
dir := dirOf(m.path)
tmp, err := os.CreateTemp(dir, ".resolv.conf.tmp.*")
if err != nil {
return err
}
tmpName := tmp.Name()
cleanup := func() { _ = os.Remove(tmpName) }
w := bufio.NewWriter(tmp)
for _, line := range lines {
if _, err := w.WriteString(line); err != nil {
tmp.Close()
cleanup()
return err
}
if _, err := w.WriteString("\n"); err != nil {
tmp.Close()
cleanup()
return err
}
}
if err := w.Flush(); err != nil {
tmp.Close()
cleanup()
return err
}
if err := tmp.Sync(); err != nil {
tmp.Close()
cleanup()
return err
}
if err := tmp.Close(); err != nil {
cleanup()
return err
}
if err := os.Rename(tmpName, m.path); err != nil {
cleanup()
return err
}
return nil
}
func parseNameserver(line string) string {
s := strings.TrimSpace(line)
if s == "" || strings.HasPrefix(s, "#") || strings.HasPrefix(s, ";") {
return ""
}
fields := strings.Fields(s)
if len(fields) < 2 || fields[0] != "nameserver" {
return ""
}
return fields[1]
}
func isValidIP(s string) bool {
return s != "" && net.ParseIP(s) != nil
}
func dirOf(path string) string {
if i := strings.LastIndex(path, "/"); i >= 0 {
return path[:i]
}
return "."
}

View file

@ -0,0 +1,158 @@
package resolv
import (
"errors"
"os"
"path/filepath"
"reflect"
"sync"
"testing"
)
func newTestManager(t *testing.T, initial string) (*Manager, string) {
t.Helper()
dir := t.TempDir()
path := filepath.Join(dir, "resolv.conf")
if initial != "" {
if err := os.WriteFile(path, []byte(initial), 0o644); err != nil {
t.Fatalf("seed: %v", err)
}
}
return New(path), path
}
func readFile(t *testing.T, path string) string {
t.Helper()
b, err := os.ReadFile(path)
if err != nil {
t.Fatalf("read: %v", err)
}
return string(b)
}
func TestList_EmptyMissingFile(t *testing.T) {
m, _ := newTestManager(t, "")
got, err := m.List()
if err != nil {
t.Fatalf("List: %v", err)
}
if len(got) != 0 {
t.Fatalf("want empty, got %v", got)
}
}
func TestList_ParsesNameserversIgnoresOther(t *testing.T) {
const initial = `# comment
; another comment
search example.com
nameserver 1.1.1.1
options rotate
nameserver 8.8.8.8
nameserver 2001:4860:4860::8888
`
m, _ := newTestManager(t, initial)
got, err := m.List()
if err != nil {
t.Fatalf("List: %v", err)
}
want := []string{"1.1.1.1", "8.8.8.8", "2001:4860:4860::8888"}
if !reflect.DeepEqual(got, want) {
t.Fatalf("want %v, got %v", want, got)
}
}
func TestAdd_AppendsAndPersists(t *testing.T) {
m, path := newTestManager(t, "search example.com\nnameserver 1.1.1.1\n")
if err := m.Add("8.8.8.8"); err != nil {
t.Fatalf("Add: %v", err)
}
got, _ := m.List()
want := []string{"1.1.1.1", "8.8.8.8"}
if !reflect.DeepEqual(got, want) {
t.Fatalf("list want %v, got %v", want, got)
}
content := readFile(t, path)
if want := "search example.com\nnameserver 1.1.1.1\nnameserver 8.8.8.8\n"; content != want {
t.Fatalf("file mismatch:\nwant: %q\ngot: %q", want, content)
}
}
func TestAdd_RejectsInvalid(t *testing.T) {
m, _ := newTestManager(t, "")
for _, addr := range []string{"", "not-an-ip", "999.999.999.999", "1.1.1"} {
if err := m.Add(addr); !errors.Is(err, ErrInvalidAddress) {
t.Fatalf("Add(%q): want ErrInvalidAddress, got %v", addr, err)
}
}
}
func TestAdd_DuplicateRejected(t *testing.T) {
m, _ := newTestManager(t, "nameserver 1.1.1.1\n")
if err := m.Add("1.1.1.1"); !errors.Is(err, ErrAlreadyExists) {
t.Fatalf("want ErrAlreadyExists, got %v", err)
}
}
func TestRemove_DeletesEntry(t *testing.T) {
const initial = "search example.com\nnameserver 1.1.1.1\nnameserver 8.8.8.8\n"
m, path := newTestManager(t, initial)
if err := m.Remove("1.1.1.1"); err != nil {
t.Fatalf("Remove: %v", err)
}
got, _ := m.List()
if want := []string{"8.8.8.8"}; !reflect.DeepEqual(got, want) {
t.Fatalf("want %v, got %v", want, got)
}
content := readFile(t, path)
if want := "search example.com\nnameserver 8.8.8.8\n"; content != want {
t.Fatalf("file mismatch:\nwant: %q\ngot: %q", want, content)
}
}
func TestRemove_NotFound(t *testing.T) {
m, _ := newTestManager(t, "nameserver 1.1.1.1\n")
if err := m.Remove("8.8.8.8"); !errors.Is(err, ErrNotFound) {
t.Fatalf("want ErrNotFound, got %v", err)
}
}
func TestRemove_InvalidAddress(t *testing.T) {
m, _ := newTestManager(t, "")
if err := m.Remove("nope"); !errors.Is(err, ErrInvalidAddress) {
t.Fatalf("want ErrInvalidAddress, got %v", err)
}
}
func TestConcurrentAddsConsistent(t *testing.T) {
m, _ := newTestManager(t, "")
addrs := []string{
"1.1.1.1", "8.8.8.8", "9.9.9.9", "8.8.4.4",
"1.0.0.1", "208.67.222.222", "208.67.220.220", "64.6.64.6",
}
var wg sync.WaitGroup
for _, a := range addrs {
wg.Add(1)
go func(a string) {
defer wg.Done()
if err := m.Add(a); err != nil {
t.Errorf("Add(%s): %v", a, err)
}
}(a)
}
wg.Wait()
got, err := m.List()
if err != nil {
t.Fatalf("List: %v", err)
}
if len(got) != len(addrs) {
t.Fatalf("want %d entries, got %d (%v)", len(addrs), len(got), got)
}
seen := make(map[string]bool, len(got))
for _, s := range got {
if seen[s] {
t.Fatalf("duplicate %s in %v", s, got)
}
seen[s] = true
}
}