mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-11 02:36:17 +00:00
75d4d2df8b
Doesn't actually work very well with the batched approach to needed files, not documented, not exposed in UI. I'll be happy to reintegrate if this is solved.
395 lines
10 KiB
Go
395 lines
10 KiB
Go
// Copyright (C) 2014 Jakob Borg and Contributors (see the CONTRIBUTORS file).
|
|
// All rights reserved. Use of this source code is governed by an MIT-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// Package config implements reading and writing of the syncthing configuration file.
|
|
package config
|
|
|
|
import (
|
|
"encoding/xml"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"reflect"
|
|
"sort"
|
|
"strconv"
|
|
|
|
"code.google.com/p/go.crypto/bcrypt"
|
|
"github.com/calmh/syncthing/logger"
|
|
"github.com/calmh/syncthing/protocol"
|
|
)
|
|
|
|
var l = logger.DefaultLogger
|
|
|
|
type Configuration struct {
|
|
Version int `xml:"version,attr" default:"2"`
|
|
Repositories []RepositoryConfiguration `xml:"repository"`
|
|
Nodes []NodeConfiguration `xml:"node"`
|
|
GUI GUIConfiguration `xml:"gui"`
|
|
Options OptionsConfiguration `xml:"options"`
|
|
XMLName xml.Name `xml:"configuration" json:"-"`
|
|
}
|
|
|
|
type RepositoryConfiguration struct {
|
|
ID string `xml:"id,attr"`
|
|
Directory string `xml:"directory,attr"`
|
|
Nodes []NodeConfiguration `xml:"node"`
|
|
ReadOnly bool `xml:"ro,attr"`
|
|
IgnorePerms bool `xml:"ignorePerms,attr"`
|
|
Invalid string `xml:"-"` // Set at runtime when there is an error, not saved
|
|
Versioning VersioningConfiguration `xml:"versioning"`
|
|
|
|
nodeIDs []protocol.NodeID
|
|
}
|
|
|
|
type VersioningConfiguration struct {
|
|
Type string `xml:"type,attr"`
|
|
Params map[string]string
|
|
}
|
|
|
|
type InternalVersioningConfiguration struct {
|
|
Type string `xml:"type,attr,omitempty"`
|
|
Params []InternalParam `xml:"param"`
|
|
}
|
|
|
|
type InternalParam struct {
|
|
Key string `xml:"key,attr"`
|
|
Val string `xml:"val,attr"`
|
|
}
|
|
|
|
func (c *VersioningConfiguration) MarshalXML(e *xml.Encoder, start xml.StartElement) error {
|
|
var tmp InternalVersioningConfiguration
|
|
tmp.Type = c.Type
|
|
for k, v := range c.Params {
|
|
tmp.Params = append(tmp.Params, InternalParam{k, v})
|
|
}
|
|
|
|
return e.EncodeElement(tmp, start)
|
|
|
|
}
|
|
|
|
func (c *VersioningConfiguration) UnmarshalXML(d *xml.Decoder, start xml.StartElement) error {
|
|
var tmp InternalVersioningConfiguration
|
|
err := d.DecodeElement(&tmp, &start)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.Type = tmp.Type
|
|
c.Params = make(map[string]string, len(tmp.Params))
|
|
for _, p := range tmp.Params {
|
|
c.Params[p.Key] = p.Val
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (r *RepositoryConfiguration) NodeIDs() []protocol.NodeID {
|
|
if r.nodeIDs == nil {
|
|
for _, n := range r.Nodes {
|
|
r.nodeIDs = append(r.nodeIDs, n.NodeID)
|
|
}
|
|
}
|
|
return r.nodeIDs
|
|
}
|
|
|
|
type NodeConfiguration struct {
|
|
NodeID protocol.NodeID `xml:"id,attr"`
|
|
Name string `xml:"name,attr,omitempty"`
|
|
Addresses []string `xml:"address,omitempty"`
|
|
}
|
|
|
|
type OptionsConfiguration struct {
|
|
ListenAddress []string `xml:"listenAddress" default:"0.0.0.0:22000"`
|
|
GlobalAnnServer string `xml:"globalAnnounceServer" default:"announce.syncthing.net:22026"`
|
|
GlobalAnnEnabled bool `xml:"globalAnnounceEnabled" default:"true"`
|
|
LocalAnnEnabled bool `xml:"localAnnounceEnabled" default:"true"`
|
|
LocalAnnPort int `xml:"localAnnouncePort" default:"21025"`
|
|
ParallelRequests int `xml:"parallelRequests" default:"16"`
|
|
MaxSendKbps int `xml:"maxSendKbps"`
|
|
RescanIntervalS int `xml:"rescanIntervalS" default:"60"`
|
|
ReconnectIntervalS int `xml:"reconnectionIntervalS" default:"60"`
|
|
MaxChangeKbps int `xml:"maxChangeKbps" default:"10000"`
|
|
StartBrowser bool `xml:"startBrowser" default:"true"`
|
|
UPnPEnabled bool `xml:"upnpEnabled" default:"true"`
|
|
URAccepted int `xml:"urAccepted"` // Accepted usage reporting version; 0 for off (undecided), -1 for off (permanently)
|
|
|
|
Deprecated_UREnabled bool `xml:"urEnabled,omitempty" json:"-"`
|
|
Deprecated_URDeclined bool `xml:"urDeclined,omitempty" json:"-"`
|
|
Deprecated_ReadOnly bool `xml:"readOnly,omitempty" json:"-"`
|
|
Deprecated_GUIEnabled bool `xml:"guiEnabled,omitempty" json:"-"`
|
|
Deprecated_GUIAddress string `xml:"guiAddress,omitempty" json:"-"`
|
|
}
|
|
|
|
type GUIConfiguration struct {
|
|
Enabled bool `xml:"enabled,attr" default:"true"`
|
|
Address string `xml:"address" default:"127.0.0.1:8080"`
|
|
User string `xml:"user,omitempty"`
|
|
Password string `xml:"password,omitempty"`
|
|
UseTLS bool `xml:"tls,attr"`
|
|
APIKey string `xml:"apikey,omitempty"`
|
|
}
|
|
|
|
func (cfg *Configuration) NodeMap() map[protocol.NodeID]NodeConfiguration {
|
|
m := make(map[protocol.NodeID]NodeConfiguration, len(cfg.Nodes))
|
|
for _, n := range cfg.Nodes {
|
|
m[n.NodeID] = n
|
|
}
|
|
return m
|
|
}
|
|
|
|
func (cfg *Configuration) RepoMap() map[string]RepositoryConfiguration {
|
|
m := make(map[string]RepositoryConfiguration, len(cfg.Repositories))
|
|
for _, r := range cfg.Repositories {
|
|
m[r.ID] = r
|
|
}
|
|
return m
|
|
}
|
|
|
|
func setDefaults(data interface{}) error {
|
|
s := reflect.ValueOf(data).Elem()
|
|
t := s.Type()
|
|
|
|
for i := 0; i < s.NumField(); i++ {
|
|
f := s.Field(i)
|
|
tag := t.Field(i).Tag
|
|
|
|
v := tag.Get("default")
|
|
if len(v) > 0 {
|
|
switch f.Interface().(type) {
|
|
case string:
|
|
f.SetString(v)
|
|
|
|
case int:
|
|
i, err := strconv.ParseInt(v, 10, 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
f.SetInt(i)
|
|
|
|
case bool:
|
|
f.SetBool(v == "true")
|
|
|
|
case []string:
|
|
// We don't do anything with string slices here. Any default
|
|
// we set will be appended to by the XML decoder, so we fill
|
|
// those after decoding.
|
|
|
|
default:
|
|
panic(f.Type())
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// fillNilSlices sets default value on slices that are still nil.
|
|
func fillNilSlices(data interface{}) error {
|
|
s := reflect.ValueOf(data).Elem()
|
|
t := s.Type()
|
|
|
|
for i := 0; i < s.NumField(); i++ {
|
|
f := s.Field(i)
|
|
tag := t.Field(i).Tag
|
|
|
|
v := tag.Get("default")
|
|
if len(v) > 0 {
|
|
switch f.Interface().(type) {
|
|
case []string:
|
|
if f.IsNil() {
|
|
rv := reflect.MakeSlice(reflect.TypeOf([]string{}), 1, 1)
|
|
rv.Index(0).SetString(v)
|
|
f.Set(rv)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func Save(wr io.Writer, cfg Configuration) error {
|
|
e := xml.NewEncoder(wr)
|
|
e.Indent("", " ")
|
|
err := e.Encode(cfg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = wr.Write([]byte("\n"))
|
|
return err
|
|
}
|
|
|
|
func uniqueStrings(ss []string) []string {
|
|
var m = make(map[string]bool, len(ss))
|
|
for _, s := range ss {
|
|
m[s] = true
|
|
}
|
|
|
|
var us = make([]string, 0, len(m))
|
|
for k := range m {
|
|
us = append(us, k)
|
|
}
|
|
|
|
return us
|
|
}
|
|
|
|
func Load(rd io.Reader, myID protocol.NodeID) (Configuration, error) {
|
|
var cfg Configuration
|
|
|
|
setDefaults(&cfg)
|
|
setDefaults(&cfg.Options)
|
|
setDefaults(&cfg.GUI)
|
|
|
|
var err error
|
|
if rd != nil {
|
|
err = xml.NewDecoder(rd).Decode(&cfg)
|
|
}
|
|
|
|
fillNilSlices(&cfg.Options)
|
|
|
|
cfg.Options.ListenAddress = uniqueStrings(cfg.Options.ListenAddress)
|
|
|
|
// Initialize an empty slice for repositories if the config has none
|
|
if cfg.Repositories == nil {
|
|
cfg.Repositories = []RepositoryConfiguration{}
|
|
}
|
|
|
|
// Check for missing, bad or duplicate repository ID:s
|
|
var seenRepos = map[string]*RepositoryConfiguration{}
|
|
var uniqueCounter int
|
|
for i := range cfg.Repositories {
|
|
repo := &cfg.Repositories[i]
|
|
|
|
if len(repo.Directory) == 0 {
|
|
repo.Invalid = "no directory configured"
|
|
continue
|
|
}
|
|
|
|
if repo.ID == "" {
|
|
repo.ID = "default"
|
|
}
|
|
|
|
if seen, ok := seenRepos[repo.ID]; ok {
|
|
l.Warnf("Multiple repositories with ID %q; disabling", repo.ID)
|
|
|
|
seen.Invalid = "duplicate repository ID"
|
|
if seen.ID == repo.ID {
|
|
uniqueCounter++
|
|
seen.ID = fmt.Sprintf("%s~%d", repo.ID, uniqueCounter)
|
|
}
|
|
repo.Invalid = "duplicate repository ID"
|
|
uniqueCounter++
|
|
repo.ID = fmt.Sprintf("%s~%d", repo.ID, uniqueCounter)
|
|
} else {
|
|
seenRepos[repo.ID] = repo
|
|
}
|
|
}
|
|
|
|
if cfg.Options.Deprecated_URDeclined {
|
|
cfg.Options.URAccepted = -1
|
|
}
|
|
cfg.Options.Deprecated_URDeclined = false
|
|
cfg.Options.Deprecated_UREnabled = false
|
|
|
|
// Upgrade to v2 configuration if appropriate
|
|
if cfg.Version == 1 {
|
|
convertV1V2(&cfg)
|
|
}
|
|
|
|
// Hash old cleartext passwords
|
|
if len(cfg.GUI.Password) > 0 && cfg.GUI.Password[0] != '$' {
|
|
hash, err := bcrypt.GenerateFromPassword([]byte(cfg.GUI.Password), 0)
|
|
if err != nil {
|
|
l.Warnln(err)
|
|
} else {
|
|
cfg.GUI.Password = string(hash)
|
|
}
|
|
}
|
|
|
|
// Ensure this node is present in all relevant places
|
|
cfg.Nodes = ensureNodePresent(cfg.Nodes, myID)
|
|
for i := range cfg.Repositories {
|
|
cfg.Repositories[i].Nodes = ensureNodePresent(cfg.Repositories[i].Nodes, myID)
|
|
}
|
|
|
|
// An empty address list is equivalent to a single "dynamic" entry
|
|
for i := range cfg.Nodes {
|
|
n := &cfg.Nodes[i]
|
|
if len(n.Addresses) == 0 || len(n.Addresses) == 1 && n.Addresses[0] == "" {
|
|
n.Addresses = []string{"dynamic"}
|
|
}
|
|
}
|
|
|
|
// The global discovery format and port number changed in v0.9. Having the
|
|
// default announce server but old port number is guaranteed to be legacy.
|
|
if cfg.Options.GlobalAnnServer == "announce.syncthing.net:22025" {
|
|
cfg.Options.GlobalAnnServer = "announce.syncthing.net:22026"
|
|
}
|
|
|
|
return cfg, err
|
|
}
|
|
|
|
func convertV1V2(cfg *Configuration) {
|
|
// Collect the list of nodes.
|
|
// Replace node configs inside repositories with only a reference to the nide ID.
|
|
// Set all repositories to read only if the global read only flag is set.
|
|
var nodes = map[string]NodeConfiguration{}
|
|
for i, repo := range cfg.Repositories {
|
|
cfg.Repositories[i].ReadOnly = cfg.Options.Deprecated_ReadOnly
|
|
for j, node := range repo.Nodes {
|
|
id := node.NodeID.String()
|
|
if _, ok := nodes[id]; !ok {
|
|
nodes[id] = node
|
|
}
|
|
cfg.Repositories[i].Nodes[j] = NodeConfiguration{NodeID: node.NodeID}
|
|
}
|
|
}
|
|
cfg.Options.Deprecated_ReadOnly = false
|
|
|
|
// Set and sort the list of nodes.
|
|
for _, node := range nodes {
|
|
cfg.Nodes = append(cfg.Nodes, node)
|
|
}
|
|
sort.Sort(NodeConfigurationList(cfg.Nodes))
|
|
|
|
// GUI
|
|
cfg.GUI.Address = cfg.Options.Deprecated_GUIAddress
|
|
cfg.GUI.Enabled = cfg.Options.Deprecated_GUIEnabled
|
|
cfg.Options.Deprecated_GUIEnabled = false
|
|
cfg.Options.Deprecated_GUIAddress = ""
|
|
|
|
cfg.Version = 2
|
|
}
|
|
|
|
type NodeConfigurationList []NodeConfiguration
|
|
|
|
func (l NodeConfigurationList) Less(a, b int) bool {
|
|
return l[a].NodeID.Compare(l[b].NodeID) == -1
|
|
}
|
|
func (l NodeConfigurationList) Swap(a, b int) {
|
|
l[a], l[b] = l[b], l[a]
|
|
}
|
|
func (l NodeConfigurationList) Len() int {
|
|
return len(l)
|
|
}
|
|
|
|
func ensureNodePresent(nodes []NodeConfiguration, myID protocol.NodeID) []NodeConfiguration {
|
|
var myIDExists bool
|
|
for _, node := range nodes {
|
|
if node.NodeID.Equals(myID) {
|
|
myIDExists = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !myIDExists {
|
|
name, _ := os.Hostname()
|
|
nodes = append(nodes, NodeConfiguration{
|
|
NodeID: myID,
|
|
Name: name,
|
|
})
|
|
}
|
|
|
|
sort.Sort(NodeConfigurationList(nodes))
|
|
|
|
return nodes
|
|
}
|