config: load config file only on first access (closes #1659, closes #2096) (#2147)

This commit is contained in:
Stefan 2018-03-17 12:36:30 +01:00 committed by GitHub
parent e62fe06763
commit 86892467d9
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 98 additions and 52 deletions

View File

@ -22,7 +22,6 @@ import (
"github.com/ncw/rclone/fs" "github.com/ncw/rclone/fs"
"github.com/ncw/rclone/fs/accounting" "github.com/ncw/rclone/fs/accounting"
"github.com/ncw/rclone/fs/config"
"github.com/ncw/rclone/fs/config/configflags" "github.com/ncw/rclone/fs/config/configflags"
"github.com/ncw/rclone/fs/config/flags" "github.com/ncw/rclone/fs/config/flags"
"github.com/ncw/rclone/fs/filter" "github.com/ncw/rclone/fs/filter"
@ -373,9 +372,6 @@ func initConfig() {
// Finish parsing any command line flags // Finish parsing any command line flags
configflags.SetFlags() configflags.SetFlags()
// Load the rest of the config now we have started the logger
config.LoadConfig()
// Load filters // Load filters
var err error var err error
filter.Active, err = filter.NewFilter(&filterflags.Opt) filter.Active, err = filter.NewFilter(&filterflags.Opt)

View File

@ -6,10 +6,10 @@ import (
) )
func init() { func init() {
cmd.Root.AddCommand(commandDefintion) cmd.Root.AddCommand(commandDefinition)
} }
var commandDefintion = &cobra.Command{ var commandDefinition = &cobra.Command{
Use: "version", Use: "version",
Short: `Show the version number.`, Short: `Show the version number.`,
Run: func(command *cobra.Command, args []string) { Run: func(command *cobra.Command, args []string) {

View File

@ -0,0 +1,43 @@
package version
import (
"io/ioutil"
"os"
"testing"
"github.com/ncw/rclone/cmd"
"github.com/ncw/rclone/fs/config"
"github.com/stretchr/testify/assert"
)
func TestVersionWorksWithoutAccessibleConfigFile(t *testing.T) {
// create temp config file
tempFile, err := ioutil.TempFile("", "unreadable_config.conf")
assert.NoError(t, err)
path := tempFile.Name()
defer func() {
err := os.Remove(path)
assert.NoError(t, err)
}()
assert.NoError(t, tempFile.Close())
assert.NoError(t, os.Chmod(path, 0000))
// re-wire
oldOsStdout := os.Stdout
oldConfigPath := config.ConfigPath
config.ConfigPath = path
os.Stdout = nil
defer func() {
os.Stdout = oldOsStdout
config.ConfigPath = oldConfigPath
}()
cmd.Root.SetArgs([]string{"version"})
assert.NotPanics(t, func() {
assert.NoError(t, cmd.Root.Execute())
})
cmd.Root.SetArgs([]string{"--version"})
assert.NotPanics(t, func() {
assert.NoError(t, cmd.Root.Execute())
})
}

View File

@ -60,8 +60,8 @@ const (
// Global // Global
var ( var (
// configData is the config file data structure // configFile is the global config data structure. Don't read it directly, use getConfigData()
configData *goconfig.ConfigFile configFile *goconfig.ConfigFile
// ConfigPath points to the config file // ConfigPath points to the config file
ConfigPath = makeConfigPath() ConfigPath = makeConfigPath()
@ -81,6 +81,13 @@ func init() {
fs.ConfigFileGet = FileGet fs.ConfigFileGet = FileGet
} }
func getConfigData() *goconfig.ConfigFile {
if configFile == nil {
LoadConfig()
}
return configFile
}
// Return the path to the configuration file // Return the path to the configuration file
func makeConfigPath() string { func makeConfigPath() string {
// Find user's home directory // Find user's home directory
@ -153,10 +160,10 @@ func makeConfigPath() string {
func LoadConfig() { func LoadConfig() {
// Load configuration file. // Load configuration file.
var err error var err error
configData, err = loadConfigFile() configFile, err = loadConfigFile()
if err == errorConfigFileNotFound { if err == errorConfigFileNotFound {
fs.Logf(nil, "Config file %q not found - using defaults", ConfigPath) fs.Logf(nil, "Config file %q not found - using defaults", ConfigPath)
configData, _ = goconfig.LoadFromReader(&bytes.Buffer{}) configFile, _ = goconfig.LoadFromReader(&bytes.Buffer{})
} else if err != nil { } else if err != nil {
log.Fatalf("Failed to load config file %q: %v", ConfigPath, err) log.Fatalf("Failed to load config file %q: %v", ConfigPath, err)
} else { } else {
@ -365,7 +372,7 @@ func saveConfig() error {
}() }()
var buf bytes.Buffer var buf bytes.Buffer
err = goconfig.SaveConfigData(configData, &buf) err = goconfig.SaveConfigData(getConfigData(), &buf)
if err != nil { if err != nil {
return errors.Errorf("Failed to save config file: %v", err) return errors.Errorf("Failed to save config file: %v", err)
} }
@ -456,7 +463,7 @@ func SaveConfig() {
// disk first and overwrites the given value only. // disk first and overwrites the given value only.
func SetValueAndSave(name, key, value string) (err error) { func SetValueAndSave(name, key, value string) (err error) {
// Set the value in config in case we fail to reload it // Set the value in config in case we fail to reload it
configData.SetValue(name, key, value) getConfigData().SetValue(name, key, value)
// Reload the config file // Reload the config file
reloadedConfigFile, err := loadConfigFile() reloadedConfigFile, err := loadConfigFile()
if err == errorConfigFileNotFound { if err == errorConfigFileNotFound {
@ -471,7 +478,7 @@ func SetValueAndSave(name, key, value string) (err error) {
return err return err
} }
// Update the config file with the reloaded version // Update the config file with the reloaded version
configData = reloadedConfigFile configFile = reloadedConfigFile
// Set the value in the reloaded version // Set the value in the reloaded version
reloadedConfigFile.SetValue(name, key, value) reloadedConfigFile.SetValue(name, key, value)
// Save it again // Save it again
@ -481,7 +488,7 @@ func SetValueAndSave(name, key, value string) (err error) {
// ShowRemotes shows an overview of the config file // ShowRemotes shows an overview of the config file
func ShowRemotes() { func ShowRemotes() {
remotes := configData.GetSectionList() remotes := getConfigData().GetSectionList()
if len(remotes) == 0 { if len(remotes) == 0 {
return return
} }
@ -495,7 +502,7 @@ func ShowRemotes() {
// ChooseRemote chooses a remote name // ChooseRemote chooses a remote name
func ChooseRemote() string { func ChooseRemote() string {
remotes := configData.GetSectionList() remotes := getConfigData().GetSectionList()
sort.Strings(remotes) sort.Strings(remotes)
return Choose("remote", remotes, nil, false) return Choose("remote", remotes, nil, false)
} }
@ -622,7 +629,7 @@ func ShowRemote(name string) {
fmt.Printf("--------------------\n") fmt.Printf("--------------------\n")
fmt.Printf("[%s]\n", name) fmt.Printf("[%s]\n", name)
fs := MustFindByName(name) fs := MustFindByName(name)
for _, key := range configData.GetKeyList(name) { for _, key := range getConfigData().GetKeyList(name) {
isPassword := false isPassword := false
for _, option := range fs.Options { for _, option := range fs.Options {
if option.Name == key && option.IsPassword { if option.Name == key && option.IsPassword {
@ -649,7 +656,7 @@ func OkRemote(name string) bool {
case 'e': case 'e':
return false return false
case 'd': case 'd':
configData.DeleteSection(name) getConfigData().DeleteSection(name)
return true return true
default: default:
fs.Errorf(nil, "Bad choice %c", i) fs.Errorf(nil, "Bad choice %c", i)
@ -736,7 +743,7 @@ func UpdateRemote(name string, keyValues []string) error {
} }
// Set the config // Set the config
for i := 0; i < len(keyValues); i += 2 { for i := 0; i < len(keyValues); i += 2 {
configData.SetValue(name, keyValues[i], keyValues[i+1]) getConfigData().SetValue(name, keyValues[i], keyValues[i+1])
} }
RemoteConfig(name) RemoteConfig(name)
ShowRemote(name) ShowRemote(name)
@ -751,11 +758,11 @@ func CreateRemote(name string, provider string, keyValues []string) error {
// Suppress Confirm // Suppress Confirm
fs.Config.AutoConfirm = true fs.Config.AutoConfirm = true
// Delete the old config if it exists // Delete the old config if it exists
configData.DeleteSection(name) getConfigData().DeleteSection(name)
// Set the type // Set the type
configData.SetValue(name, "type", provider) getConfigData().SetValue(name, "type", provider)
// Show this is automatically configured // Show this is automatically configured
configData.SetValue(name, ConfigAutomatic, "yes") getConfigData().SetValue(name, ConfigAutomatic, "yes")
// Set the remaining values // Set the remaining values
return UpdateRemote(name, keyValues) return UpdateRemote(name, keyValues)
} }
@ -770,7 +777,7 @@ func PasswordRemote(name string, keyValues []string) error {
fs.Config.AutoConfirm = true fs.Config.AutoConfirm = true
passwd := obscure.MustObscure(keyValues[1]) passwd := obscure.MustObscure(keyValues[1])
if passwd != "" { if passwd != "" {
configData.SetValue(name, keyValues[0], passwd) getConfigData().SetValue(name, keyValues[0], passwd)
RemoteConfig(name) RemoteConfig(name)
ShowRemote(name) ShowRemote(name)
SaveConfig() SaveConfig()
@ -830,10 +837,10 @@ func NewRemoteName() (name string) {
// NewRemote make a new remote from its name // NewRemote make a new remote from its name
func NewRemote(name string) { func NewRemote(name string) {
newType := ChooseOption(fsOption()) newType := ChooseOption(fsOption())
configData.SetValue(name, "type", newType) getConfigData().SetValue(name, "type", newType)
fs := fs.MustFind(newType) fs := fs.MustFind(newType)
for _, option := range fs.Options { for _, option := range fs.Options {
configData.SetValue(name, option.Name, ChooseOption(&option)) getConfigData().SetValue(name, option.Name, ChooseOption(&option))
} }
RemoteConfig(name) RemoteConfig(name)
if OkRemote(name) { if OkRemote(name) {
@ -855,7 +862,7 @@ func EditRemote(fs *fs.RegInfo, name string) {
fmt.Printf("Edit? (y/n)>\n") fmt.Printf("Edit? (y/n)>\n")
if Confirm() { if Confirm() {
newValue := ChooseOption(&option) newValue := ChooseOption(&option)
configData.SetValue(name, key, newValue) getConfigData().SetValue(name, key, newValue)
} }
} }
if OkRemote(name) { if OkRemote(name) {
@ -868,7 +875,7 @@ func EditRemote(fs *fs.RegInfo, name string) {
// DeleteRemote gets the user to delete a remote // DeleteRemote gets the user to delete a remote
func DeleteRemote(name string) { func DeleteRemote(name string) {
configData.DeleteSection(name) getConfigData().DeleteSection(name)
SaveConfig() SaveConfig()
} }
@ -877,9 +884,9 @@ func DeleteRemote(name string) {
func copyRemote(name string) string { func copyRemote(name string) string {
newName := NewRemoteName() newName := NewRemoteName()
// Copy the keys // Copy the keys
for _, key := range configData.GetKeyList(name) { for _, key := range getConfigData().GetKeyList(name) {
value := configData.MustValue(name, key, "") value := getConfigData().MustValue(name, key, "")
configData.SetValue(newName, key, value) getConfigData().SetValue(newName, key, value)
} }
return newName return newName
} }
@ -889,7 +896,7 @@ func RenameRemote(name string) {
fmt.Printf("Enter new name for %q remote.\n", name) fmt.Printf("Enter new name for %q remote.\n", name)
newName := copyRemote(name) newName := copyRemote(name)
if name != newName { if name != newName {
configData.DeleteSection(name) getConfigData().DeleteSection(name)
SaveConfig() SaveConfig()
} }
} }
@ -914,7 +921,7 @@ func ShowConfigLocation() {
// ShowConfig prints the (unencrypted) config options // ShowConfig prints the (unencrypted) config options
func ShowConfig() { func ShowConfig() {
var buf bytes.Buffer var buf bytes.Buffer
if err := goconfig.SaveConfigData(configData, &buf); err != nil { if err := goconfig.SaveConfigData(getConfigData(), &buf); err != nil {
log.Fatalf("Failed to serialize config: %v", err) log.Fatalf("Failed to serialize config: %v", err)
} }
str := buf.String() str := buf.String()
@ -927,7 +934,7 @@ func ShowConfig() {
// EditConfig edits the config file interactively // EditConfig edits the config file interactively
func EditConfig() { func EditConfig() {
for { for {
haveRemotes := len(configData.GetSectionList()) != 0 haveRemotes := len(getConfigData().GetSectionList()) != 0
what := []string{"eEdit existing remote", "nNew remote", "dDelete remote", "rRename remote", "cCopy remote", "sSet configuration password", "qQuit config"} what := []string{"eEdit existing remote", "nNew remote", "dDelete remote", "rRename remote", "cCopy remote", "sSet configuration password", "qQuit config"}
if haveRemotes { if haveRemotes {
fmt.Printf("Current remotes:\n\n") fmt.Printf("Current remotes:\n\n")
@ -1023,10 +1030,10 @@ func Authorize(args []string) {
defer DeleteRemote(name) defer DeleteRemote(name)
// Indicate that we want fully automatic configuration. // Indicate that we want fully automatic configuration.
configData.SetValue(name, ConfigAutomatic, "yes") getConfigData().SetValue(name, ConfigAutomatic, "yes")
if len(args) == 3 { if len(args) == 3 {
configData.SetValue(name, ConfigClientID, args[1]) getConfigData().SetValue(name, ConfigClientID, args[1])
configData.SetValue(name, ConfigClientSecret, args[2]) getConfigData().SetValue(name, ConfigClientSecret, args[2])
} }
fs.Config(name) fs.Config(name)
} }
@ -1048,7 +1055,7 @@ func FileGet(section, key string, defaultVal ...string) string {
if found { if found {
defaultVal = []string{newValue} defaultVal = []string{newValue}
} }
return configData.MustValue(section, key, defaultVal...) return getConfigData().MustValue(section, key, defaultVal...)
} }
// FileGetBool gets the config key under section returning the // FileGetBool gets the config key under section returning the
@ -1066,7 +1073,7 @@ func FileGetBool(section, key string, defaultVal ...bool) bool {
defaultVal = []bool{newBool} defaultVal = []bool{newBool}
} }
} }
return configData.MustBool(section, key, defaultVal...) return getConfigData().MustBool(section, key, defaultVal...)
} }
// FileGetInt gets the config key under section returning the // FileGetInt gets the config key under section returning the
@ -1084,20 +1091,20 @@ func FileGetInt(section, key string, defaultVal ...int) int {
defaultVal = []int{newInt} defaultVal = []int{newInt}
} }
} }
return configData.MustInt(section, key, defaultVal...) return getConfigData().MustInt(section, key, defaultVal...)
} }
// FileSet sets the key in section to value. It doesn't save // FileSet sets the key in section to value. It doesn't save
// the config file. // the config file.
func FileSet(section, key, value string) { func FileSet(section, key, value string) {
configData.SetValue(section, key, value) getConfigData().SetValue(section, key, value)
} }
// FileDeleteKey deletes the config key in the config file. // FileDeleteKey deletes the config key in the config file.
// It returns true if the key was deleted, // It returns true if the key was deleted,
// or returns false if the section or key didn't exist. // or returns false if the section or key didn't exist.
func FileDeleteKey(section, key string) bool { func FileDeleteKey(section, key string) bool {
return configData.DeleteKey(section, key) return getConfigData().DeleteKey(section, key)
} }
var matchEnv = regexp.MustCompile(`^RCLONE_CONFIG_(.*?)_TYPE=.*$`) var matchEnv = regexp.MustCompile(`^RCLONE_CONFIG_(.*?)_TYPE=.*$`)
@ -1105,7 +1112,7 @@ var matchEnv = regexp.MustCompile(`^RCLONE_CONFIG_(.*?)_TYPE=.*$`)
// FileSections returns the sections in the config file // FileSections returns the sections in the config file
// including any defined by environment variables. // including any defined by environment variables.
func FileSections() []string { func FileSections() []string {
sections := configData.GetSectionList() sections := getConfigData().GetSectionList()
for _, item := range os.Environ() { for _, item := range os.Environ() {
matches := matchEnv.FindStringSubmatch(item) matches := matchEnv.FindStringSubmatch(item)
if len(matches) == 2 { if len(matches) == 2 {
@ -1118,9 +1125,9 @@ func FileSections() []string {
// Dump dumps all the config as a JSON file // Dump dumps all the config as a JSON file
func Dump() error { func Dump() error {
dump := make(map[string]map[string]string) dump := make(map[string]map[string]string)
for _, name := range configData.GetSectionList() { for _, name := range getConfigData().GetSectionList() {
params := make(map[string]string) params := make(map[string]string)
for _, key := range configData.GetKeyList(name) { for _, key := range getConfigData().GetKeyList(name) {
params[key] = FileGet(name, key) params[key] = FileGet(name, key)
} }
dump[name] = params dump[name] = params

View File

@ -27,22 +27,22 @@ func TestCRUD(t *testing.T) {
oldOsStdout := os.Stdout oldOsStdout := os.Stdout
oldConfigPath := ConfigPath oldConfigPath := ConfigPath
oldConfig := fs.Config oldConfig := fs.Config
oldConfigData := configData oldConfigFile := configFile
oldReadLine := ReadLine oldReadLine := ReadLine
os.Stdout = nil os.Stdout = nil
ConfigPath = path ConfigPath = path
fs.Config = &fs.ConfigInfo{} fs.Config = &fs.ConfigInfo{}
configData = nil configFile = nil
defer func() { defer func() {
os.Stdout = oldOsStdout os.Stdout = oldOsStdout
ConfigPath = oldConfigPath ConfigPath = oldConfigPath
ReadLine = oldReadLine ReadLine = oldReadLine
fs.Config = oldConfig fs.Config = oldConfig
configData = oldConfigData configFile = oldConfigFile
}() }()
LoadConfig() LoadConfig()
assert.Equal(t, []string{}, configData.GetSectionList()) assert.Equal(t, []string{}, getConfigData().GetSectionList())
// Fake a remote // Fake a remote
fs.Register(&fs.RegInfo{Name: "config_test_remote"}) fs.Register(&fs.RegInfo{Name: "config_test_remote"})
@ -59,25 +59,25 @@ func TestCRUD(t *testing.T) {
} }
NewRemote("test") NewRemote("test")
assert.Equal(t, []string{"test"}, configData.GetSectionList()) assert.Equal(t, []string{"test"}, configFile.GetSectionList())
// Reload the config file to workaround this bug // Reload the config file to workaround this bug
// https://github.com/Unknwon/goconfig/issues/39 // https://github.com/Unknwon/goconfig/issues/39
configData, err = loadConfigFile() configFile, err = loadConfigFile()
require.NoError(t, err) require.NoError(t, err)
// normal rename, test → asdf // normal rename, test → asdf
ReadLine = func() string { return "asdf" } ReadLine = func() string { return "asdf" }
RenameRemote("test") RenameRemote("test")
assert.Equal(t, []string{"asdf"}, configData.GetSectionList()) assert.Equal(t, []string{"asdf"}, configFile.GetSectionList())
// no-op rename, asdf → asdf // no-op rename, asdf → asdf
RenameRemote("asdf") RenameRemote("asdf")
assert.Equal(t, []string{"asdf"}, configData.GetSectionList()) assert.Equal(t, []string{"asdf"}, configFile.GetSectionList())
// delete remote // delete remote
DeleteRemote("asdf") DeleteRemote("asdf")
assert.Equal(t, []string{}, configData.GetSectionList()) assert.Equal(t, []string{}, configFile.GetSectionList())
} }
// Test some error cases // Test some error cases