fs: add Options registry and rework rc to use it

This commit is contained in:
Nick Craig-Wood 2024-07-03 17:22:47 +01:00
parent 8e10fe71f7
commit e79273f9c9
5 changed files with 125 additions and 40 deletions

View file

@ -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)

View file

@ -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
if configName != "" {
config.AddGetter(configEnvVars(configName), configmap.PriorityNormal) 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)
} }

View file

@ -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)
} }

View file

@ -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) {

View file

@ -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)