fs: add Options registry and rework rc to use it
This commit is contained in:
parent
8e10fe71f7
commit
e79273f9c9
5 changed files with 125 additions and 40 deletions
13
cmd/cmd.go
13
cmd/cmd.go
|
@ -29,7 +29,6 @@ import (
|
||||||
"github.com/rclone/rclone/fs/config/configflags"
|
"github.com/rclone/rclone/fs/config/configflags"
|
||||||
"github.com/rclone/rclone/fs/config/flags"
|
"github.com/rclone/rclone/fs/config/flags"
|
||||||
"github.com/rclone/rclone/fs/filter"
|
"github.com/rclone/rclone/fs/filter"
|
||||||
"github.com/rclone/rclone/fs/filter/filterflags"
|
|
||||||
"github.com/rclone/rclone/fs/fserrors"
|
"github.com/rclone/rclone/fs/fserrors"
|
||||||
"github.com/rclone/rclone/fs/fspath"
|
"github.com/rclone/rclone/fs/fspath"
|
||||||
fslog "github.com/rclone/rclone/fs/log"
|
fslog "github.com/rclone/rclone/fs/log"
|
||||||
|
@ -383,6 +382,12 @@ func StartStats() func() {
|
||||||
|
|
||||||
// initConfig is run by cobra after initialising the flags
|
// initConfig is run by cobra after initialising the flags
|
||||||
func initConfig() {
|
func initConfig() {
|
||||||
|
// Set the global options from the flags
|
||||||
|
err := fs.GlobalOptionsInit()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("Failed to initialise global options: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
ci := fs.GetConfig(ctx)
|
ci := fs.GetConfig(ctx)
|
||||||
|
|
||||||
|
@ -409,12 +414,6 @@ func initConfig() {
|
||||||
terminal.EnableColorsStdout()
|
terminal.EnableColorsStdout()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Load filters
|
|
||||||
err := filterflags.Reload(ctx)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("Failed to load filters: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write the args for debug purposes
|
// Write the args for debug purposes
|
||||||
fs.Debugf("rclone", "Version %q starting with parameters %q", fs.Version, os.Args)
|
fs.Debugf("rclone", "Version %q starting with parameters %q", fs.Version, os.Args)
|
||||||
|
|
||||||
|
|
|
@ -33,10 +33,15 @@ func (oev optionEnvVars) Get(key string) (value string, ok bool) {
|
||||||
if opt == nil {
|
if opt == nil {
|
||||||
return "", false
|
return "", false
|
||||||
}
|
}
|
||||||
envKey := OptionToEnv(oev.prefix + "-" + key)
|
var envKey string
|
||||||
|
if oev.prefix == "" {
|
||||||
|
envKey = OptionToEnv(key)
|
||||||
|
} else {
|
||||||
|
envKey = OptionToEnv(oev.prefix + "-" + key)
|
||||||
|
}
|
||||||
value, ok = os.LookupEnv(envKey)
|
value, ok = os.LookupEnv(envKey)
|
||||||
if ok {
|
if ok {
|
||||||
Debugf(nil, "Setting %s_%s=%q from environment variable %s", oev.prefix, key, value, envKey)
|
Debugf(nil, "Setting %s %s=%q from environment variable %s", oev.prefix, key, value, envKey)
|
||||||
} else if opt.NoPrefix {
|
} else if opt.NoPrefix {
|
||||||
// For options with NoPrefix set, check without prefix too
|
// For options with NoPrefix set, check without prefix too
|
||||||
envKey := OptionToEnv(key)
|
envKey := OptionToEnv(key)
|
||||||
|
@ -115,10 +120,12 @@ func ConfigMap(prefix string, options Options, configName string, connectionStri
|
||||||
}
|
}
|
||||||
|
|
||||||
// remote specific environment vars
|
// remote specific environment vars
|
||||||
config.AddGetter(configEnvVars(configName), configmap.PriorityNormal)
|
if configName != "" {
|
||||||
|
config.AddGetter(configEnvVars(configName), configmap.PriorityNormal)
|
||||||
|
}
|
||||||
|
|
||||||
// backend specific environment vars
|
// backend specific environment vars
|
||||||
if options != nil && prefix != "" {
|
if options != nil {
|
||||||
config.AddGetter(optionEnvVars{prefix: prefix, options: options}, configmap.PriorityNormal)
|
config.AddGetter(optionEnvVars{prefix: prefix, options: options}, configmap.PriorityNormal)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,21 +12,17 @@ import (
|
||||||
"github.com/rclone/rclone/fs/filter"
|
"github.com/rclone/rclone/fs/filter"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
|
||||||
optionBlock = map[string]interface{}{}
|
|
||||||
optionReload = map[string]func(context.Context) error{}
|
|
||||||
)
|
|
||||||
|
|
||||||
// AddOption adds an option set
|
// AddOption adds an option set
|
||||||
func AddOption(name string, option interface{}) {
|
func AddOption(name string, option interface{}) {
|
||||||
optionBlock[name] = option
|
// FIXME remove this function when conversion to options is complete
|
||||||
|
fs.RegisterGlobalOptions(fs.OptionsInfo{Name: name, Opt: option})
|
||||||
}
|
}
|
||||||
|
|
||||||
// AddOptionReload adds an option set with a reload function to be
|
// AddOptionReload adds an option set with a reload function to be
|
||||||
// called when options are changed
|
// called when options are changed
|
||||||
func AddOptionReload(name string, option interface{}, reload func(context.Context) error) {
|
func AddOptionReload(name string, option interface{}, reload func(context.Context) error) {
|
||||||
optionBlock[name] = option
|
// FIXME remove this function when conversion to options is complete
|
||||||
optionReload[name] = reload
|
fs.RegisterGlobalOptions(fs.OptionsInfo{Name: name, Opt: option, Reload: reload})
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
func init() {
|
||||||
|
@ -42,8 +38,8 @@ func init() {
|
||||||
// Show the list of all the option blocks
|
// Show the list of all the option blocks
|
||||||
func rcOptionsBlocks(ctx context.Context, in Params) (out Params, err error) {
|
func rcOptionsBlocks(ctx context.Context, in Params) (out Params, err error) {
|
||||||
options := []string{}
|
options := []string{}
|
||||||
for name := range optionBlock {
|
for _, opt := range fs.OptionsRegistry {
|
||||||
options = append(options, name)
|
options = append(options, opt.Name)
|
||||||
}
|
}
|
||||||
out = make(Params)
|
out = make(Params)
|
||||||
out["options"] = options
|
out["options"] = options
|
||||||
|
@ -71,8 +67,8 @@ map to the external options very easily with a few exceptions.
|
||||||
// Show the list of all the option blocks
|
// Show the list of all the option blocks
|
||||||
func rcOptionsGet(ctx context.Context, in Params) (out Params, err error) {
|
func rcOptionsGet(ctx context.Context, in Params) (out Params, err error) {
|
||||||
out = make(Params)
|
out = make(Params)
|
||||||
for name, options := range optionBlock {
|
for _, opt := range fs.OptionsRegistry {
|
||||||
out[name] = options
|
out[opt.Name] = opt.Opt
|
||||||
}
|
}
|
||||||
return out, nil
|
return out, nil
|
||||||
}
|
}
|
||||||
|
@ -144,16 +140,16 @@ And this sets NOTICE level logs (normal without -v)
|
||||||
// Set an option in an option block
|
// Set an option in an option block
|
||||||
func rcOptionsSet(ctx context.Context, in Params) (out Params, err error) {
|
func rcOptionsSet(ctx context.Context, in Params) (out Params, err error) {
|
||||||
for name, options := range in {
|
for name, options := range in {
|
||||||
current := optionBlock[name]
|
opt, ok := fs.OptionsRegistry[name]
|
||||||
if current == nil {
|
if !ok {
|
||||||
return nil, fmt.Errorf("unknown option block %q", name)
|
return nil, fmt.Errorf("unknown option block %q", name)
|
||||||
}
|
}
|
||||||
err := Reshape(current, options)
|
err := Reshape(opt.Opt, options)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to write options from block %q: %w", name, err)
|
return nil, fmt.Errorf("failed to write options from block %q: %w", name, err)
|
||||||
}
|
}
|
||||||
if reload := optionReload[name]; reload != nil {
|
if opt.Reload != nil {
|
||||||
err = reload(ctx)
|
err = opt.Reload(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, fmt.Errorf("failed to reload options from block %q: %w", name, err)
|
return nil, fmt.Errorf("failed to reload options from block %q: %w", name, err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,10 +13,10 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func clearOptionBlock() func() {
|
func clearOptionBlock() func() {
|
||||||
oldOptionBlock := optionBlock
|
oldOptionBlock := fs.OptionsRegistry
|
||||||
optionBlock = map[string]interface{}{}
|
fs.OptionsRegistry = map[string]fs.OptionsInfo{}
|
||||||
return func() {
|
return func() {
|
||||||
optionBlock = oldOptionBlock
|
fs.OptionsRegistry = oldOptionBlock
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -30,22 +30,20 @@ var testOptions = struct {
|
||||||
|
|
||||||
func TestAddOption(t *testing.T) {
|
func TestAddOption(t *testing.T) {
|
||||||
defer clearOptionBlock()()
|
defer clearOptionBlock()()
|
||||||
assert.Equal(t, len(optionBlock), 0)
|
assert.Equal(t, len(fs.OptionsRegistry), 0)
|
||||||
AddOption("potato", &testOptions)
|
AddOption("potato", &testOptions)
|
||||||
assert.Equal(t, len(optionBlock), 1)
|
assert.Equal(t, len(fs.OptionsRegistry), 1)
|
||||||
assert.Equal(t, len(optionReload), 0)
|
assert.Equal(t, &testOptions, fs.OptionsRegistry["potato"].Opt)
|
||||||
assert.Equal(t, &testOptions, optionBlock["potato"])
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAddOptionReload(t *testing.T) {
|
func TestAddOptionReload(t *testing.T) {
|
||||||
defer clearOptionBlock()()
|
defer clearOptionBlock()()
|
||||||
assert.Equal(t, len(optionBlock), 0)
|
assert.Equal(t, len(fs.OptionsRegistry), 0)
|
||||||
reload := func(ctx context.Context) error { return nil }
|
reload := func(ctx context.Context) error { return nil }
|
||||||
AddOptionReload("potato", &testOptions, reload)
|
AddOptionReload("potato", &testOptions, reload)
|
||||||
assert.Equal(t, len(optionBlock), 1)
|
assert.Equal(t, len(fs.OptionsRegistry), 1)
|
||||||
assert.Equal(t, len(optionReload), 1)
|
assert.Equal(t, &testOptions, fs.OptionsRegistry["potato"].Opt)
|
||||||
assert.Equal(t, &testOptions, optionBlock["potato"])
|
assert.Equal(t, fmt.Sprintf("%p", reload), fmt.Sprintf("%p", fs.OptionsRegistry["potato"].Reload))
|
||||||
assert.Equal(t, fmt.Sprintf("%p", reload), fmt.Sprintf("%p", optionReload["potato"]))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestOptionsBlocks(t *testing.T) {
|
func TestOptionsBlocks(t *testing.T) {
|
||||||
|
|
|
@ -61,6 +61,22 @@ func (ri *RegInfo) FileName() string {
|
||||||
// Options is a slice of configuration Option for a backend
|
// Options is a slice of configuration Option for a backend
|
||||||
type Options []Option
|
type Options []Option
|
||||||
|
|
||||||
|
// Add more options returning a new options slice
|
||||||
|
func (os Options) Add(newOptions Options) Options {
|
||||||
|
return append(os, newOptions...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddPrefix adds more options with a prefix returning a new options slice
|
||||||
|
func (os Options) AddPrefix(newOptions Options, prefix string, groups string) Options {
|
||||||
|
for _, opt := range newOptions {
|
||||||
|
// opt is a copy so can modify
|
||||||
|
opt.Name = prefix + "_" + opt.Name
|
||||||
|
opt.Groups = groups
|
||||||
|
os = append(os, opt)
|
||||||
|
}
|
||||||
|
return os
|
||||||
|
}
|
||||||
|
|
||||||
// Set the default values for the options
|
// Set the default values for the options
|
||||||
func (os Options) setValues() {
|
func (os Options) setValues() {
|
||||||
for i := range os {
|
for i := range os {
|
||||||
|
@ -91,6 +107,19 @@ func (os Options) Get(name string) *Option {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetDefault sets the default for the Option corresponding to name
|
||||||
|
//
|
||||||
|
// Writes an ERROR level log if the option is not found
|
||||||
|
func (os Options) SetDefault(name string, def any) Options {
|
||||||
|
opt := os.Get(name)
|
||||||
|
if opt == nil {
|
||||||
|
Errorf(nil, "Couldn't find option %q to SetDefault on", name)
|
||||||
|
} else {
|
||||||
|
opt.Default = def
|
||||||
|
}
|
||||||
|
return os
|
||||||
|
}
|
||||||
|
|
||||||
// Overridden discovers which config items have been overridden in the
|
// Overridden discovers which config items have been overridden in the
|
||||||
// configmap passed in, either by the config string, command line
|
// configmap passed in, either by the config string, command line
|
||||||
// flags or environment variables
|
// flags or environment variables
|
||||||
|
@ -370,6 +399,62 @@ func MustFind(name string) *RegInfo {
|
||||||
return fs
|
return fs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// OptionsInfo holds info about an block of options
|
||||||
|
type OptionsInfo struct {
|
||||||
|
Name string // name of this options block for the rc
|
||||||
|
Opt interface{} // pointer to a struct to set the options in
|
||||||
|
Options Options // description of the options
|
||||||
|
Reload func(context.Context) error // if not nil, call when options changed and on init
|
||||||
|
}
|
||||||
|
|
||||||
|
// OptionsRegistry is a registry of global options
|
||||||
|
var OptionsRegistry = map[string]OptionsInfo{}
|
||||||
|
|
||||||
|
// RegisterGlobalOptions registers global options to be made into
|
||||||
|
// command line options, rc options and environment variable reading.
|
||||||
|
//
|
||||||
|
// Packages which need global options should use this in an init() function
|
||||||
|
func RegisterGlobalOptions(oi OptionsInfo) {
|
||||||
|
OptionsRegistry[oi.Name] = oi
|
||||||
|
}
|
||||||
|
|
||||||
|
// load the defaults from the options
|
||||||
|
//
|
||||||
|
// Reload the options if required
|
||||||
|
func (oi *OptionsInfo) load() error {
|
||||||
|
if oi.Options == nil {
|
||||||
|
Errorf(nil, "No options defined for config block %q", oi.Name)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
m := ConfigMap("", oi.Options, "", nil)
|
||||||
|
err := configstruct.Set(m, oi.Opt)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to initialise %q options: %w", oi.Name, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if oi.Reload != nil {
|
||||||
|
err = oi.Reload(context.Background())
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to reload %q options: %w", oi.Name, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GlobalOptionsInit initialises the defaults of global options to
|
||||||
|
// their values read from the options, environment variables and
|
||||||
|
// command line parameters.
|
||||||
|
func GlobalOptionsInit() error {
|
||||||
|
for _, opt := range OptionsRegistry {
|
||||||
|
err := opt.load()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// Type returns a textual string to identify the type of the remote
|
// Type returns a textual string to identify the type of the remote
|
||||||
func Type(f Fs) string {
|
func Type(f Fs) string {
|
||||||
typeName := fmt.Sprintf("%T", f)
|
typeName := fmt.Sprintf("%T", f)
|
||||||
|
|
Loading…
Reference in a new issue