2023-10-11 21:55:57 +01:00
|
|
|
// Copyright (c) Tailscale Inc & AUTHORS
|
|
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
|
|
|
|
// Package conffile contains code to load, manipulate, and access config file
|
|
|
|
// settings.
|
|
|
|
package conffile
|
|
|
|
|
|
|
|
import (
|
2023-10-16 20:15:03 +01:00
|
|
|
"bytes"
|
2023-10-11 21:55:57 +01:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/tailscale/hujson"
|
|
|
|
"tailscale.com/ipn"
|
|
|
|
)
|
|
|
|
|
2023-10-16 20:15:03 +01:00
|
|
|
// Config describes a config file.
|
2023-10-11 21:55:57 +01:00
|
|
|
type Config struct {
|
|
|
|
Path string // disk path of HuJSON
|
|
|
|
Raw []byte // raw bytes from disk, in HuJSON form
|
|
|
|
Std []byte // standardized JSON form
|
|
|
|
Version string // "alpha0" for now
|
|
|
|
|
|
|
|
// Parsed is the parsed config, converted from its on-disk version to the
|
|
|
|
// latest known format.
|
|
|
|
//
|
|
|
|
// As of 2023-10-15 there is exactly one format ("alpha0") so this is both
|
|
|
|
// the on-disk format and the in-memory upgraded format.
|
|
|
|
Parsed ipn.ConfigVAlpha
|
|
|
|
}
|
|
|
|
|
2023-10-16 20:15:03 +01:00
|
|
|
// WantRunning reports whether c is non-nil and it's configured to be running.
|
|
|
|
func (c *Config) WantRunning() bool {
|
|
|
|
return c != nil && !c.Parsed.Enabled.EqualBool(false)
|
|
|
|
}
|
|
|
|
|
2023-10-11 21:55:57 +01:00
|
|
|
// Load reads and parses the config file at the provided path on disk.
|
|
|
|
func Load(path string) (*Config, error) {
|
|
|
|
var c Config
|
|
|
|
c.Path = path
|
|
|
|
|
|
|
|
var err error
|
|
|
|
c.Raw, err = os.ReadFile(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
c.Std, err = hujson.Standardize(c.Raw)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error parsing config file %s HuJSON/JSON: %w", path, err)
|
|
|
|
}
|
|
|
|
var ver struct {
|
|
|
|
Version string `json:"version"`
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(c.Std, &ver); err != nil {
|
|
|
|
return nil, fmt.Errorf("error parsing config file %s: %w", path, err)
|
|
|
|
}
|
|
|
|
switch ver.Version {
|
|
|
|
case "":
|
|
|
|
return nil, fmt.Errorf("error parsing config file %s: no \"version\" field defined", path)
|
|
|
|
case "alpha0":
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("error parsing config file %s: unsupported \"version\" value %q; want \"alpha0\" for now", path, ver.Version)
|
|
|
|
}
|
|
|
|
c.Version = ver.Version
|
|
|
|
|
2023-10-16 20:15:03 +01:00
|
|
|
jd := json.NewDecoder(bytes.NewReader(c.Std))
|
|
|
|
jd.DisallowUnknownFields()
|
|
|
|
err = jd.Decode(&c.Parsed)
|
2023-10-11 21:55:57 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error parsing config file %s: %w", path, err)
|
|
|
|
}
|
2023-10-16 20:15:03 +01:00
|
|
|
if jd.More() {
|
|
|
|
return nil, fmt.Errorf("error parsing config file %s: trailing data after JSON object", path)
|
|
|
|
}
|
2023-10-11 21:55:57 +01:00
|
|
|
return &c, nil
|
|
|
|
}
|