2015-01-01 18:02:25 +01:00
|
|
|
// Copyright 2015 The Chihaya Authors. All rights reserved.
|
2013-06-22 03:43:11 +02:00
|
|
|
// Use of this source code is governed by the BSD 2-Clause license,
|
|
|
|
// which can be found in the LICENSE file.
|
|
|
|
|
2013-06-24 04:34:13 +02:00
|
|
|
// Package config implements the configuration for a BitTorrent tracker
|
2013-06-22 01:31:32 +02:00
|
|
|
package config
|
|
|
|
|
|
|
|
import (
|
2013-11-24 06:49:20 +01:00
|
|
|
"encoding/json"
|
2014-07-15 02:36:49 +02:00
|
|
|
"errors"
|
2013-11-24 06:49:20 +01:00
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"time"
|
2013-06-22 01:31:32 +02:00
|
|
|
)
|
|
|
|
|
2014-07-15 05:13:35 +02:00
|
|
|
// ErrMissingRequiredParam is used by drivers to indicate that an entry required
|
|
|
|
// to be within the DriverConfig.Params map is not present.
|
2014-07-15 02:36:49 +02:00
|
|
|
var ErrMissingRequiredParam = errors.New("A parameter that was required by a driver is not present")
|
|
|
|
|
2014-05-01 06:30:46 +02:00
|
|
|
// Duration wraps a time.Duration and adds JSON marshalling.
|
2014-07-03 23:37:13 +02:00
|
|
|
type Duration struct{ time.Duration }
|
2013-06-22 01:31:32 +02:00
|
|
|
|
2014-05-01 06:30:46 +02:00
|
|
|
// MarshalJSON transforms a duration into JSON.
|
2013-06-23 12:21:59 +02:00
|
|
|
func (d *Duration) MarshalJSON() ([]byte, error) {
|
2013-11-24 06:49:20 +01:00
|
|
|
return json.Marshal(d.String())
|
2013-06-23 12:21:59 +02:00
|
|
|
}
|
2013-06-22 01:31:32 +02:00
|
|
|
|
2014-05-01 06:30:46 +02:00
|
|
|
// UnmarshalJSON transform JSON into a Duration.
|
2013-06-23 12:21:59 +02:00
|
|
|
func (d *Duration) UnmarshalJSON(b []byte) error {
|
2013-11-24 06:49:20 +01:00
|
|
|
var str string
|
|
|
|
err := json.Unmarshal(b, &str)
|
|
|
|
d.Duration, err = time.ParseDuration(str)
|
|
|
|
return err
|
2013-06-22 01:31:32 +02:00
|
|
|
}
|
|
|
|
|
2014-06-24 09:59:30 +02:00
|
|
|
// DriverConfig is the configuration used to connect to a tracker.Driver or
|
|
|
|
// a backend.Driver.
|
|
|
|
type DriverConfig struct {
|
2014-07-03 23:37:13 +02:00
|
|
|
Name string `json:"driver"`
|
|
|
|
Params map[string]string `json:"params,omitempty"`
|
2013-06-22 01:31:32 +02:00
|
|
|
}
|
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
// SubnetConfig is the configuration used to specify if local peers should be
|
|
|
|
// given a preference when responding to an announce.
|
2014-09-24 05:00:50 +02:00
|
|
|
type SubnetConfig struct {
|
|
|
|
PreferredSubnet bool `json:"preferred_subnet,omitempty"`
|
|
|
|
PreferredIPv4Subnet int `json:"preferred_ipv4_subnet,omitempty"`
|
|
|
|
PreferredIPv6Subnet int `json:"preferred_ipv6_subnet,omitempty"`
|
|
|
|
}
|
|
|
|
|
2014-07-23 20:00:17 +02:00
|
|
|
// NetConfig is the configuration used to tune networking behaviour.
|
|
|
|
type NetConfig struct {
|
|
|
|
AllowIPSpoofing bool `json:"allow_ip_spoofing"`
|
|
|
|
DualStackedPeers bool `json:"dual_stacked_peers"`
|
|
|
|
RealIPHeader string `json:"real_ip_header"`
|
2014-08-21 01:24:33 +02:00
|
|
|
RespectAF bool `json:"respect_af"`
|
2014-09-24 05:00:50 +02:00
|
|
|
SubnetConfig
|
2014-07-23 20:00:17 +02:00
|
|
|
}
|
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
// StatsConfig is the configuration used to record runtime statistics.
|
2014-07-23 23:25:01 +02:00
|
|
|
type StatsConfig struct {
|
|
|
|
BufferSize int `json:"stats_buffer_size"`
|
|
|
|
IncludeMem bool `json:"include_mem_stats"`
|
|
|
|
VerboseMem bool `json:"verbose_mem_stats"`
|
|
|
|
|
|
|
|
MemUpdateInterval Duration `json:"mem_stats_interval"`
|
|
|
|
}
|
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
// WhitelistConfig is the configuration used enable and store a whitelist of
|
|
|
|
// acceptable torrent client peer ID prefixes.
|
|
|
|
type WhitelistConfig struct {
|
|
|
|
ClientWhitelistEnabled bool `json:"client_whitelist_enabled"`
|
|
|
|
ClientWhitelist []string `json:"client_whitelist,omitempty"`
|
2014-09-24 18:53:35 +02:00
|
|
|
}
|
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
// TrackerConfig is the configuration for tracker functionality.
|
|
|
|
type TrackerConfig struct {
|
|
|
|
PrivateEnabled bool `json:"private_enabled"`
|
|
|
|
FreeleechEnabled bool `json:"freeleech_enabled"`
|
|
|
|
PurgeInactiveTorrents bool `json:"purge_inactive_torrents"`
|
|
|
|
Announce Duration `json:"announce"`
|
|
|
|
MinAnnounce Duration `json:"min_announce"`
|
|
|
|
NumWantFallback int `json:"default_num_want"`
|
|
|
|
TorrentMapShards int `json:"torrent_map_shards"`
|
2013-06-22 01:31:32 +02:00
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
NetConfig
|
|
|
|
WhitelistConfig
|
|
|
|
}
|
2014-07-17 01:38:51 +02:00
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
// HTTPConfig is the configuration for HTTP functionality.
|
|
|
|
type HTTPConfig struct {
|
2015-02-20 05:46:28 +01:00
|
|
|
HTTPListenAddr string `json:"http_listen_addr"`
|
|
|
|
HTTPRequestTimeout Duration `json:"http_request_timeout"`
|
|
|
|
HTTPReadTimeout Duration `json:"http_read_timeout"`
|
|
|
|
HTTPWriteTimeout Duration `json:"http_write_timeout"`
|
|
|
|
HTTPListenLimit int `json:"http_listen_limit"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// UDPConfig is the configuration for HTTP functionality.
|
|
|
|
type UDPConfig struct {
|
|
|
|
UDPListenAddr string `json:"udp_listen_addr"`
|
|
|
|
UDPReadBufferSize int `json:"udp_read_buffer_size"`
|
2014-11-02 01:12:40 +01:00
|
|
|
}
|
2014-07-25 22:58:26 +02:00
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
// Config is the global configuration for an instance of Chihaya.
|
|
|
|
type Config struct {
|
|
|
|
TrackerConfig
|
|
|
|
HTTPConfig
|
2015-02-20 05:46:28 +01:00
|
|
|
UDPConfig
|
2014-11-02 01:12:40 +01:00
|
|
|
DriverConfig
|
2014-07-23 23:25:01 +02:00
|
|
|
StatsConfig
|
2014-06-24 09:59:30 +02:00
|
|
|
}
|
|
|
|
|
2014-07-15 05:13:35 +02:00
|
|
|
// DefaultConfig is a configuration that can be used as a fallback value.
|
2014-07-02 03:40:29 +02:00
|
|
|
var DefaultConfig = Config{
|
2014-11-02 01:12:40 +01:00
|
|
|
TrackerConfig: TrackerConfig{
|
|
|
|
PrivateEnabled: false,
|
|
|
|
FreeleechEnabled: false,
|
|
|
|
PurgeInactiveTorrents: true,
|
|
|
|
Announce: Duration{30 * time.Minute},
|
|
|
|
MinAnnounce: Duration{15 * time.Minute},
|
|
|
|
NumWantFallback: 50,
|
|
|
|
TorrentMapShards: 1,
|
|
|
|
|
|
|
|
NetConfig: NetConfig{
|
|
|
|
AllowIPSpoofing: true,
|
|
|
|
DualStackedPeers: true,
|
|
|
|
RespectAF: false,
|
|
|
|
},
|
|
|
|
|
|
|
|
WhitelistConfig: WhitelistConfig{
|
|
|
|
ClientWhitelistEnabled: false,
|
|
|
|
},
|
|
|
|
},
|
2014-07-17 01:38:51 +02:00
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
HTTPConfig: HTTPConfig{
|
2015-02-20 07:12:47 +01:00
|
|
|
HTTPListenAddr: "",
|
2015-02-20 05:46:28 +01:00
|
|
|
HTTPRequestTimeout: Duration{10 * time.Second},
|
|
|
|
HTTPReadTimeout: Duration{10 * time.Second},
|
|
|
|
HTTPWriteTimeout: Duration{10 * time.Second},
|
|
|
|
},
|
|
|
|
|
|
|
|
UDPConfig: UDPConfig{
|
2015-02-20 07:12:47 +01:00
|
|
|
UDPListenAddr: "",
|
2014-07-02 03:40:29 +02:00
|
|
|
},
|
2014-07-17 01:38:51 +02:00
|
|
|
|
2014-11-02 01:12:40 +01:00
|
|
|
DriverConfig: DriverConfig{
|
2014-07-15 06:22:04 +02:00
|
|
|
Name: "noop",
|
2014-07-02 03:40:29 +02:00
|
|
|
},
|
2014-07-17 01:38:51 +02:00
|
|
|
|
2014-07-23 23:25:01 +02:00
|
|
|
StatsConfig: StatsConfig{
|
|
|
|
BufferSize: 0,
|
|
|
|
IncludeMem: true,
|
|
|
|
VerboseMem: false,
|
|
|
|
|
|
|
|
MemUpdateInterval: Duration{5 * time.Second},
|
|
|
|
},
|
2013-06-22 01:31:32 +02:00
|
|
|
}
|
|
|
|
|
2013-07-05 12:50:52 +02:00
|
|
|
// Open is a shortcut to open a file, read it, and generate a Config.
|
2014-07-15 05:13:35 +02:00
|
|
|
// It supports relative and absolute paths. Given "", it returns DefaultConfig.
|
2013-07-05 12:50:52 +02:00
|
|
|
func Open(path string) (*Config, error) {
|
2014-06-24 09:59:30 +02:00
|
|
|
if path == "" {
|
2014-07-02 03:40:29 +02:00
|
|
|
return &DefaultConfig, nil
|
2014-06-24 09:59:30 +02:00
|
|
|
}
|
|
|
|
|
2014-05-01 06:30:46 +02:00
|
|
|
f, err := os.Open(os.ExpandEnv(path))
|
2013-11-24 06:49:20 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
2013-06-22 01:31:32 +02:00
|
|
|
|
2014-06-24 09:59:30 +02:00
|
|
|
conf, err := Decode(f)
|
2013-11-24 06:49:20 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return conf, nil
|
2013-06-22 01:31:32 +02:00
|
|
|
}
|
2013-06-23 12:21:59 +02:00
|
|
|
|
2014-07-15 05:13:35 +02:00
|
|
|
// Decode casts an io.Reader into a JSONDecoder and decodes it into a *Config.
|
2014-06-24 09:59:30 +02:00
|
|
|
func Decode(r io.Reader) (*Config, error) {
|
2014-07-23 22:14:50 +02:00
|
|
|
conf := DefaultConfig
|
|
|
|
err := json.NewDecoder(r).Decode(&conf)
|
|
|
|
return &conf, err
|
2013-06-23 12:21:59 +02:00
|
|
|
}
|