config: add context.Context #3257 #4685

This add config to the Config callback in the backends and the related
config functions.
This commit is contained in:
Nick Craig-Wood 2020-11-05 18:02:26 +00:00
parent 1fb6ad700f
commit f7efce594b
27 changed files with 105 additions and 103 deletions

View file

@ -70,8 +70,8 @@ func init() {
Prefix: "acd", Prefix: "acd",
Description: "Amazon Drive", Description: "Amazon Drive",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
err := oauthutil.Config("amazon cloud drive", name, m, acdConfig, nil) err := oauthutil.Config(ctx, "amazon cloud drive", name, m, acdConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -255,7 +255,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} else { } else {
fs.Debugf(name+":", "Couldn't add request filter - large file downloads will fail") fs.Debugf(name+":", "Couldn't add request filter - large file downloads will fail")
} }
oAuthClient, ts, err := oauthutil.NewClientWithBaseClient(name, m, acdConfig, baseClient) oAuthClient, ts, err := oauthutil.NewClientWithBaseClient(ctx, name, m, acdConfig, baseClient)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure Amazon Drive") return nil, errors.Wrap(err, "failed to configure Amazon Drive")
} }

View file

@ -84,7 +84,7 @@ func init() {
Name: "box", Name: "box",
Description: "Box", Description: "Box",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
jsonFile, ok := m.Get("box_config_file") jsonFile, ok := m.Get("box_config_file")
boxSubType, boxSubTypeOk := m.Get("box_sub_type") boxSubType, boxSubTypeOk := m.Get("box_sub_type")
boxAccessToken, boxAccessTokenOk := m.Get("access_token") boxAccessToken, boxAccessTokenOk := m.Get("access_token")
@ -97,7 +97,7 @@ func init() {
} }
// Else, if not using an access token, use oauth2 // Else, if not using an access token, use oauth2
} else if boxAccessToken == "" || !boxAccessTokenOk { } else if boxAccessToken == "" || !boxAccessTokenOk {
err = oauthutil.Config("box", name, m, oauthConfig, nil) err = oauthutil.Config(ctx, "box", name, m, oauthConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token with oauth authentication: %v", err) log.Fatalf("Failed to configure token with oauth authentication: %v", err)
} }
@ -390,7 +390,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
var ts *oauthutil.TokenSource var ts *oauthutil.TokenSource
// If not using an accessToken, create an oauth client and tokensource // If not using an accessToken, create an oauth client and tokensource
if opt.AccessToken == "" { if opt.AccessToken == "" {
client, ts, err = oauthutil.NewClient(name, m, oauthConfig) client, ts, err = oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure Box") return nil, errors.Wrap(err, "failed to configure Box")
} }

View file

@ -176,8 +176,7 @@ func init() {
Description: "Google Drive", Description: "Google Drive",
NewFs: NewFs, NewFs: NewFs,
CommandHelp: commandHelp, CommandHelp: commandHelp,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
ctx := context.TODO()
// Parse config into Options struct // Parse config into Options struct
opt := new(Options) opt := new(Options)
err := configstruct.Set(m, opt) err := configstruct.Set(m, opt)
@ -194,7 +193,7 @@ func init() {
} }
if opt.ServiceAccountFile == "" { if opt.ServiceAccountFile == "" {
err = oauthutil.Config("drive", name, m, driveConfig, nil) err = oauthutil.Config(ctx, "drive", name, m, driveConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -991,7 +990,7 @@ func getClient(opt *Options) *http.Client {
} }
} }
func getServiceAccountClient(opt *Options, credentialsData []byte) (*http.Client, error) { func getServiceAccountClient(ctx context.Context, opt *Options, credentialsData []byte) (*http.Client, error) {
scopes := driveScopes(opt.Scope) scopes := driveScopes(opt.Scope)
conf, err := google.JWTConfigFromJSON(credentialsData, scopes...) conf, err := google.JWTConfigFromJSON(credentialsData, scopes...)
if err != nil { if err != nil {
@ -1000,11 +999,11 @@ func getServiceAccountClient(opt *Options, credentialsData []byte) (*http.Client
if opt.Impersonate != "" { if opt.Impersonate != "" {
conf.Subject = opt.Impersonate conf.Subject = opt.Impersonate
} }
ctxWithSpecialClient := oauthutil.Context(getClient(opt)) ctxWithSpecialClient := oauthutil.Context(ctx, getClient(opt))
return oauth2.NewClient(ctxWithSpecialClient, conf.TokenSource(ctxWithSpecialClient)), nil return oauth2.NewClient(ctxWithSpecialClient, conf.TokenSource(ctxWithSpecialClient)), nil
} }
func createOAuthClient(opt *Options, name string, m configmap.Mapper) (*http.Client, error) { func createOAuthClient(ctx context.Context, opt *Options, name string, m configmap.Mapper) (*http.Client, error) {
var oAuthClient *http.Client var oAuthClient *http.Client
var err error var err error
@ -1017,12 +1016,12 @@ func createOAuthClient(opt *Options, name string, m configmap.Mapper) (*http.Cli
opt.ServiceAccountCredentials = string(loadedCreds) opt.ServiceAccountCredentials = string(loadedCreds)
} }
if opt.ServiceAccountCredentials != "" { if opt.ServiceAccountCredentials != "" {
oAuthClient, err = getServiceAccountClient(opt, []byte(opt.ServiceAccountCredentials)) oAuthClient, err = getServiceAccountClient(ctx, opt, []byte(opt.ServiceAccountCredentials))
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to create oauth client from service account") return nil, errors.Wrap(err, "failed to create oauth client from service account")
} }
} else { } else {
oAuthClient, _, err = oauthutil.NewClientWithBaseClient(name, m, driveConfig, getClient(opt)) oAuthClient, _, err = oauthutil.NewClientWithBaseClient(ctx, name, m, driveConfig, getClient(opt))
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to create oauth client") return nil, errors.Wrap(err, "failed to create oauth client")
} }
@ -1081,7 +1080,7 @@ func newFs(ctx context.Context, name, path string, m configmap.Mapper) (*Fs, err
return nil, errors.Wrap(err, "drive: chunk size") return nil, errors.Wrap(err, "drive: chunk size")
} }
oAuthClient, err := createOAuthClient(opt, name, m) oAuthClient, err := createOAuthClient(ctx, opt, name, m)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "drive: failed when making oauth client") return nil, errors.Wrap(err, "drive: failed when making oauth client")
} }
@ -2770,7 +2769,7 @@ func (f *Fs) changeChunkSize(chunkSizeString string) (err error) {
return err return err
} }
func (f *Fs) changeServiceAccountFile(file string) (err error) { func (f *Fs) changeServiceAccountFile(ctx context.Context, file string) (err error) {
fs.Debugf(nil, "Changing Service Account File from %s to %s", f.opt.ServiceAccountFile, file) fs.Debugf(nil, "Changing Service Account File from %s to %s", f.opt.ServiceAccountFile, file)
if file == f.opt.ServiceAccountFile { if file == f.opt.ServiceAccountFile {
return nil return nil
@ -2792,7 +2791,7 @@ func (f *Fs) changeServiceAccountFile(file string) (err error) {
}() }()
f.opt.ServiceAccountFile = file f.opt.ServiceAccountFile = file
f.opt.ServiceAccountCredentials = "" f.opt.ServiceAccountCredentials = ""
oAuthClient, err := createOAuthClient(&f.opt, f.name, f.m) oAuthClient, err := createOAuthClient(ctx, &f.opt, f.name, f.m)
if err != nil { if err != nil {
return errors.Wrap(err, "drive: failed when making oauth client") return errors.Wrap(err, "drive: failed when making oauth client")
} }
@ -3160,7 +3159,7 @@ func (f *Fs) Command(ctx context.Context, name string, arg []string, opt map[str
if serviceAccountFile, ok := opt["service_account_file"]; ok { if serviceAccountFile, ok := opt["service_account_file"]; ok {
serviceAccountMap := make(map[string]string) serviceAccountMap := make(map[string]string)
serviceAccountMap["previous"] = f.opt.ServiceAccountFile serviceAccountMap["previous"] = f.opt.ServiceAccountFile
if err = f.changeServiceAccountFile(serviceAccountFile); err != nil { if err = f.changeServiceAccountFile(ctx, serviceAccountFile); err != nil {
return out, err return out, err
} }
f.m.Set("service_account_file", serviceAccountFile) f.m.Set("service_account_file", serviceAccountFile)

View file

@ -116,11 +116,11 @@ func init() {
Name: "dropbox", Name: "dropbox",
Description: "Dropbox", Description: "Dropbox",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
opt := oauthutil.Options{ opt := oauthutil.Options{
NoOffline: true, NoOffline: true,
} }
err := oauthutil.Config("dropbox", name, m, dropboxConfig, &opt) err := oauthutil.Config(ctx, "dropbox", name, m, dropboxConfig, &opt)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -316,7 +316,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} }
} }
oAuthClient, _, err := oauthutil.NewClient(name, m, dropboxConfig) oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, dropboxConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure dropbox") return nil, errors.Wrap(err, "failed to configure dropbox")
} }

View file

@ -35,7 +35,7 @@ func init() {
fs.Register(&fs.RegInfo{ fs.Register(&fs.RegInfo{
Name: "fichier", Name: "fichier",
Description: "1Fichier", Description: "1Fichier",
Config: func(name string, config configmap.Mapper) { Config: func(ctx context.Context, name string, config configmap.Mapper) {
}, },
NewFs: NewFs, NewFs: NewFs,
Options: []fs.Option{{ Options: []fs.Option{{

View file

@ -76,14 +76,14 @@ func init() {
Prefix: "gcs", Prefix: "gcs",
Description: "Google Cloud Storage (this is not Google Drive)", Description: "Google Cloud Storage (this is not Google Drive)",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
saFile, _ := m.Get("service_account_file") saFile, _ := m.Get("service_account_file")
saCreds, _ := m.Get("service_account_credentials") saCreds, _ := m.Get("service_account_credentials")
anonymous, _ := m.Get("anonymous") anonymous, _ := m.Get("anonymous")
if saFile != "" || saCreds != "" || anonymous == "true" { if saFile != "" || saCreds != "" || anonymous == "true" {
return return
} }
err := oauthutil.Config("google cloud storage", name, m, storageConfig, nil) err := oauthutil.Config(ctx, "google cloud storage", name, m, storageConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -370,12 +370,12 @@ func (o *Object) split() (bucket, bucketPath string) {
return o.fs.split(o.remote) return o.fs.split(o.remote)
} }
func getServiceAccountClient(credentialsData []byte) (*http.Client, error) { func getServiceAccountClient(ctx context.Context, credentialsData []byte) (*http.Client, error) {
conf, err := google.JWTConfigFromJSON(credentialsData, storageConfig.Scopes...) conf, err := google.JWTConfigFromJSON(credentialsData, storageConfig.Scopes...)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "error processing credentials") return nil, errors.Wrap(err, "error processing credentials")
} }
ctxWithSpecialClient := oauthutil.Context(fshttp.NewClient(fs.Config)) ctxWithSpecialClient := oauthutil.Context(ctx, fshttp.NewClient(fs.Config))
return oauth2.NewClient(ctxWithSpecialClient, conf.TokenSource(ctxWithSpecialClient)), nil return oauth2.NewClient(ctxWithSpecialClient, conf.TokenSource(ctxWithSpecialClient)), nil
} }
@ -413,12 +413,12 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
if opt.Anonymous { if opt.Anonymous {
oAuthClient = &http.Client{} oAuthClient = &http.Client{}
} else if opt.ServiceAccountCredentials != "" { } else if opt.ServiceAccountCredentials != "" {
oAuthClient, err = getServiceAccountClient([]byte(opt.ServiceAccountCredentials)) oAuthClient, err = getServiceAccountClient(ctx, []byte(opt.ServiceAccountCredentials))
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed configuring Google Cloud Storage Service Account") return nil, errors.Wrap(err, "failed configuring Google Cloud Storage Service Account")
} }
} else { } else {
oAuthClient, _, err = oauthutil.NewClient(name, m, storageConfig) oAuthClient, _, err = oauthutil.NewClient(ctx, name, m, storageConfig)
if err != nil { if err != nil {
ctx := context.Background() ctx := context.Background()
oAuthClient, err = google.DefaultClient(ctx, storage.DevstorageFullControlScope) oAuthClient, err = google.DefaultClient(ctx, storage.DevstorageFullControlScope)

View file

@ -78,7 +78,7 @@ func init() {
Prefix: "gphotos", Prefix: "gphotos",
Description: "Google Photos", Description: "Google Photos",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
// Parse config into Options struct // Parse config into Options struct
opt := new(Options) opt := new(Options)
err := configstruct.Set(m, opt) err := configstruct.Set(m, opt)
@ -95,7 +95,7 @@ func init() {
} }
// Do the oauth // Do the oauth
err = oauthutil.Config("google photos", name, m, oauthConfig, nil) err = oauthutil.Config(ctx, "google photos", name, m, oauthConfig, nil)
if err != nil { if err != nil {
golog.Fatalf("Failed to configure token: %v", err) golog.Fatalf("Failed to configure token: %v", err)
} }
@ -255,7 +255,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} }
baseClient := fshttp.NewClient(fs.Config) baseClient := fshttp.NewClient(fs.Config)
oAuthClient, ts, err := oauthutil.NewClientWithBaseClient(name, m, oauthConfig, baseClient) oAuthClient, ts, err := oauthutil.NewClientWithBaseClient(ctx, name, m, oauthConfig, baseClient)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure Box") return nil, errors.Wrap(err, "failed to configure Box")
} }

View file

@ -56,8 +56,8 @@ func init() {
Name: "hubic", Name: "hubic",
Description: "Hubic", Description: "Hubic",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
err := oauthutil.Config("hubic", name, m, oauthConfig, nil) err := oauthutil.Config(ctx, "hubic", name, m, oauthConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -147,7 +147,7 @@ func (f *Fs) getCredentials(ctx context.Context) (err error) {
// NewFs constructs an Fs from the path, container:path // NewFs constructs an Fs from the path, container:path
func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, error) { func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, error) {
client, _, err := oauthutil.NewClient(name, m, oauthConfig) client, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure Hubic") return nil, errors.Wrap(err, "failed to configure Hubic")
} }

View file

@ -83,9 +83,7 @@ func init() {
Name: "jottacloud", Name: "jottacloud",
Description: "Jottacloud", Description: "Jottacloud",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
ctx := context.TODO()
refresh := false refresh := false
if version, ok := m.Get("configVersion"); ok { if version, ok := m.Get("configVersion"); ok {
ver, err := strconv.Atoi(version) ver, err := strconv.Atoi(version)
@ -275,7 +273,7 @@ func v1config(ctx context.Context, name string, m configmap.Mapper) {
fmt.Printf("\nDo you want to use a non standard device/mountpoint e.g. for accessing files uploaded using the official Jottacloud client?\n\n") fmt.Printf("\nDo you want to use a non standard device/mountpoint e.g. for accessing files uploaded using the official Jottacloud client?\n\n")
if config.Confirm(false) { if config.Confirm(false) {
oAuthClient, _, err := oauthutil.NewClient(name, m, oauthConfig) oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
log.Fatalf("Failed to load oAuthClient: %s", err) log.Fatalf("Failed to load oAuthClient: %s", err)
} }
@ -387,7 +385,7 @@ func v2config(ctx context.Context, name string, m configmap.Mapper) {
fmt.Printf("\nDo you want to use a non standard device/mountpoint e.g. for accessing files uploaded using the official Jottacloud client?\n\n") fmt.Printf("\nDo you want to use a non standard device/mountpoint e.g. for accessing files uploaded using the official Jottacloud client?\n\n")
if config.Confirm(false) { if config.Confirm(false) {
oAuthClient, _, err := oauthutil.NewClient(name, m, oauthConfig) oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
log.Fatalf("Failed to load oAuthClient: %s", err) log.Fatalf("Failed to load oAuthClient: %s", err)
} }
@ -699,7 +697,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} }
// Create OAuth Client // Create OAuth Client
oAuthClient, ts, err := oauthutil.NewClientWithBaseClient(name, m, oauthConfig, baseClient) oAuthClient, ts, err := oauthutil.NewClientWithBaseClient(ctx, name, m, oauthConfig, baseClient)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "Failed to configure Jottacloud oauth client") return nil, errors.Wrap(err, "Failed to configure Jottacloud oauth client")
} }

View file

@ -420,7 +420,7 @@ func (f *Fs) authorize(ctx context.Context, force bool) (err error) {
if err != nil || !tokenIsValid(t) { if err != nil || !tokenIsValid(t) {
fs.Infof(f, "Valid token not found, authorizing.") fs.Infof(f, "Valid token not found, authorizing.")
ctx := oauthutil.Context(f.cli) ctx := oauthutil.Context(ctx, f.cli)
t, err = oauthConfig.PasswordCredentialsToken(ctx, f.opt.Username, f.opt.Password) t, err = oauthConfig.PasswordCredentialsToken(ctx, f.opt.Username, f.opt.Password)
} }
if err == nil && !tokenIsValid(t) { if err == nil && !tokenIsValid(t) {
@ -443,7 +443,7 @@ func (f *Fs) authorize(ctx context.Context, force bool) (err error) {
// crashing with panic `comparing uncomparable type map[string]interface{}` // crashing with panic `comparing uncomparable type map[string]interface{}`
// As a workaround, mimic oauth2.NewClient() wrapping token source in // As a workaround, mimic oauth2.NewClient() wrapping token source in
// oauth2.ReuseTokenSource // oauth2.ReuseTokenSource
_, ts, err := oauthutil.NewClientWithBaseClient(f.name, f.m, oauthConfig, f.cli) _, ts, err := oauthutil.NewClientWithBaseClient(ctx, f.name, f.m, oauthConfig, f.cli)
if err == nil { if err == nil {
f.source = oauth2.ReuseTokenSource(nil, ts) f.source = oauth2.ReuseTokenSource(nil, ts)
} }

View file

@ -80,9 +80,8 @@ func init() {
Name: "onedrive", Name: "onedrive",
Description: "Microsoft OneDrive", Description: "Microsoft OneDrive",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
ctx := context.TODO() err := oauthutil.Config(ctx, "onedrive", name, m, oauthConfig, nil)
err := oauthutil.Config("onedrive", name, m, oauthConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
return return
@ -111,7 +110,7 @@ func init() {
Sites []siteResource `json:"value"` Sites []siteResource `json:"value"`
} }
oAuthClient, _, err := oauthutil.NewClient(name, m, oauthConfig) oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
log.Fatalf("Failed to configure OneDrive: %v", err) log.Fatalf("Failed to configure OneDrive: %v", err)
} }
@ -233,7 +232,7 @@ func init() {
fmt.Printf("Found drive '%s' of type '%s', URL: %s\nIs that okay?\n", rootItem.Name, rootItem.ParentReference.DriveType, rootItem.WebURL) fmt.Printf("Found drive '%s' of type '%s', URL: %s\nIs that okay?\n", rootItem.Name, rootItem.ParentReference.DriveType, rootItem.WebURL)
// This does not work, YET :) // This does not work, YET :)
if !config.ConfirmWithConfig(m, "config_drive_ok", true) { if !config.ConfirmWithConfig(ctx, m, "config_drive_ok", true) {
log.Fatalf("Cancelled by user") log.Fatalf("Cancelled by user")
} }
@ -614,7 +613,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} }
root = parsePath(root) root = parsePath(root)
oAuthClient, ts, err := oauthutil.NewClient(name, m, oauthConfig) oAuthClient, ts, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure OneDrive") return nil, errors.Wrap(err, "failed to configure OneDrive")
} }

View file

@ -72,7 +72,7 @@ func init() {
Name: "pcloud", Name: "pcloud",
Description: "Pcloud", Description: "Pcloud",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
optc := new(Options) optc := new(Options)
err := configstruct.Set(m, optc) err := configstruct.Set(m, optc)
if err != nil { if err != nil {
@ -98,7 +98,7 @@ func init() {
CheckAuth: checkAuth, CheckAuth: checkAuth,
StateBlankOK: true, // pCloud seems to drop the state parameter now - see #4210 StateBlankOK: true, // pCloud seems to drop the state parameter now - see #4210
} }
err = oauthutil.Config("pcloud", name, m, oauthConfig, &opt) err = oauthutil.Config(ctx, "pcloud", name, m, oauthConfig, &opt)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -288,7 +288,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
return nil, err return nil, err
} }
root = parsePath(root) root = parsePath(root)
oAuthClient, ts, err := oauthutil.NewClient(name, m, oauthConfig) oAuthClient, ts, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure Pcloud") return nil, errors.Wrap(err, "failed to configure Pcloud")
} }

View file

@ -78,8 +78,8 @@ func init() {
Name: "premiumizeme", Name: "premiumizeme",
Description: "premiumize.me", Description: "premiumize.me",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
err := oauthutil.Config("premiumizeme", name, m, oauthConfig, nil) err := oauthutil.Config(ctx, "premiumizeme", name, m, oauthConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -247,7 +247,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
var client *http.Client var client *http.Client
var ts *oauthutil.TokenSource var ts *oauthutil.TokenSource
if opt.APIKey == "" { if opt.APIKey == "" {
client, ts, err = oauthutil.NewClient(name, m, oauthConfig) client, ts, err = oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure premiumize.me") return nil, errors.Wrap(err, "failed to configure premiumize.me")
} }

View file

@ -78,7 +78,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (f fs.Fs,
} }
root = parsePath(root) root = parsePath(root)
httpClient := fshttp.NewClient(fs.Config) httpClient := fshttp.NewClient(fs.Config)
oAuthClient, _, err := oauthutil.NewClientWithBaseClient(name, m, putioConfig, httpClient) oAuthClient, _, err := oauthutil.NewClientWithBaseClient(ctx, name, m, putioConfig, httpClient)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure putio") return nil, errors.Wrap(err, "failed to configure putio")
} }

View file

@ -1,6 +1,7 @@
package putio package putio
import ( import (
"context"
"log" "log"
"regexp" "regexp"
"time" "time"
@ -59,11 +60,11 @@ func init() {
Name: "putio", Name: "putio",
Description: "Put.io", Description: "Put.io",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
opt := oauthutil.Options{ opt := oauthutil.Options{
NoOffline: true, NoOffline: true,
} }
err := oauthutil.Config("putio", name, m, putioConfig, &opt) err := oauthutil.Config(ctx, "putio", name, m, putioConfig, &opt)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }

View file

@ -296,7 +296,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} }
// Config callback for 2FA // Config callback for 2FA
func Config(name string, m configmap.Mapper) { func Config(ctx context.Context, name string, m configmap.Mapper) {
serverURL, ok := m.Get(configURL) serverURL, ok := m.Get(configURL)
if !ok || serverURL == "" { if !ok || serverURL == "" {
// If there's no server URL, it means we're trying an operation at the backend level, like a "rclone authorize seafile" // If there's no server URL, it means we're trying an operation at the backend level, like a "rclone authorize seafile"

View file

@ -136,7 +136,7 @@ func init() {
Name: "sharefile", Name: "sharefile",
Description: "Citrix Sharefile", Description: "Citrix Sharefile",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
oauthConfig := newOauthConfig("") oauthConfig := newOauthConfig("")
checkAuth := func(oauthConfig *oauth2.Config, auth *oauthutil.AuthResult) error { checkAuth := func(oauthConfig *oauth2.Config, auth *oauthutil.AuthResult) error {
if auth == nil || auth.Form == nil { if auth == nil || auth.Form == nil {
@ -155,7 +155,7 @@ func init() {
opt := oauthutil.Options{ opt := oauthutil.Options{
CheckAuth: checkAuth, CheckAuth: checkAuth,
} }
err := oauthutil.Config("sharefile", name, m, oauthConfig, &opt) err := oauthutil.Config(ctx, "sharefile", name, m, oauthConfig, &opt)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
} }
@ -436,7 +436,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
oauthConfig := newOauthConfig(opt.Endpoint + tokenPath) oauthConfig := newOauthConfig(opt.Endpoint + tokenPath)
var client *http.Client var client *http.Client
var ts *oauthutil.TokenSource var ts *oauthutil.TokenSource
client, ts, err = oauthutil.NewClient(name, m, oauthConfig) client, ts, err = oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
return nil, errors.Wrap(err, "failed to configure sharefile") return nil, errors.Wrap(err, "failed to configure sharefile")
} }

View file

@ -76,7 +76,7 @@ func init() {
Name: "sugarsync", Name: "sugarsync",
Description: "Sugarsync", Description: "Sugarsync",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
opt := new(Options) opt := new(Options)
err := configstruct.Set(m, opt) err := configstruct.Set(m, opt)
if err != nil { if err != nil {
@ -85,7 +85,7 @@ func init() {
if opt.RefreshToken != "" { if opt.RefreshToken != "" {
fmt.Printf("Already have a token - refresh?\n") fmt.Printf("Already have a token - refresh?\n")
if !config.ConfirmWithConfig(m, "config_refresh_token", true) { if !config.ConfirmWithConfig(ctx, m, "config_refresh_token", true) {
return return
} }
} }

View file

@ -42,7 +42,7 @@ func init() {
Name: "tardigrade", Name: "tardigrade",
Description: "Tardigrade Decentralized Cloud Storage", Description: "Tardigrade Decentralized Cloud Storage",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, configMapper configmap.Mapper) { Config: func(ctx context.Context, name string, configMapper configmap.Mapper) {
provider, _ := configMapper.Get(fs.ConfigProvider) provider, _ := configMapper.Get(fs.ConfigProvider)
config.FileDeleteKey(name, fs.ConfigProvider) config.FileDeleteKey(name, fs.ConfigProvider)

View file

@ -60,8 +60,8 @@ func init() {
Name: "yandex", Name: "yandex",
Description: "Yandex Disk", Description: "Yandex Disk",
NewFs: NewFs, NewFs: NewFs,
Config: func(name string, m configmap.Mapper) { Config: func(ctx context.Context, name string, m configmap.Mapper) {
err := oauthutil.Config("yandex", name, m, oauthConfig, nil) err := oauthutil.Config(ctx, "yandex", name, m, oauthConfig, nil)
if err != nil { if err != nil {
log.Fatalf("Failed to configure token: %v", err) log.Fatalf("Failed to configure token: %v", err)
return return
@ -260,7 +260,7 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
} }
log.Printf("Automatically upgraded OAuth config.") log.Printf("Automatically upgraded OAuth config.")
} }
oAuthClient, _, err := oauthutil.NewClient(name, m, oauthConfig) oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
if err != nil { if err != nil {
log.Fatalf("Failed to configure Yandex: %v", err) log.Fatalf("Failed to configure Yandex: %v", err)
} }

View file

@ -1,6 +1,8 @@
package authorize package authorize
import ( import (
"context"
"github.com/rclone/rclone/cmd" "github.com/rclone/rclone/cmd"
"github.com/rclone/rclone/fs/config" "github.com/rclone/rclone/fs/config"
"github.com/rclone/rclone/fs/config/flags" "github.com/rclone/rclone/fs/config/flags"
@ -29,6 +31,6 @@ Use the --auth-no-open-browser to prevent rclone to open auth
link in default browser automatically.`, link in default browser automatically.`,
Run: func(command *cobra.Command, args []string) { Run: func(command *cobra.Command, args []string) {
cmd.CheckArgs(1, 3, command, args) cmd.CheckArgs(1, 3, command, args)
config.Authorize(args, noAutoBrowser) config.Authorize(context.Background(), args, noAutoBrowser)
}, },
} }

View file

@ -142,7 +142,7 @@ using remote authorization you would do this:
if err != nil { if err != nil {
return err return err
} }
err = config.CreateRemote(args[0], args[1], in, configObscure, configNoObscure) err = config.CreateRemote(context.Background(), args[0], args[1], in, configObscure, configNoObscure)
if err != nil { if err != nil {
return err return err
} }
@ -181,7 +181,7 @@ require this add an extra parameter thus:
if err != nil { if err != nil {
return err return err
} }
err = config.UpdateRemote(args[0], in, configObscure, configNoObscure) err = config.UpdateRemote(context.Background(), args[0], in, configObscure, configNoObscure)
if err != nil { if err != nil {
return err return err
} }
@ -219,7 +219,7 @@ both support obscuring passwords directly.
if err != nil { if err != nil {
return err return err
} }
err = config.PasswordRemote(args[0], in) err = config.PasswordRemote(context.Background(), args[0], in)
if err != nil { if err != nil {
return err return err
} }
@ -253,6 +253,7 @@ To disconnect the remote use "rclone config disconnect".
This normally means going through the interactive oauth flow again. This normally means going through the interactive oauth flow again.
`, `,
RunE: func(command *cobra.Command, args []string) error { RunE: func(command *cobra.Command, args []string) error {
ctx := context.Background()
cmd.CheckArgs(1, 1, command, args) cmd.CheckArgs(1, 1, command, args)
fsInfo, configName, _, config, err := fs.ConfigFs(args[0]) fsInfo, configName, _, config, err := fs.ConfigFs(args[0])
if err != nil { if err != nil {
@ -261,7 +262,7 @@ This normally means going through the interactive oauth flow again.
if fsInfo.Config == nil { if fsInfo.Config == nil {
return errors.Errorf("%s: doesn't support Reconnect", configName) return errors.Errorf("%s: doesn't support Reconnect", configName)
} }
fsInfo.Config(configName, config) fsInfo.Config(ctx, configName, config)
return nil return nil
}, },
} }

View file

@ -745,7 +745,7 @@ func Confirm(Default bool) bool {
// If AutoConfirm is set, it will look up the value in m and return // If AutoConfirm is set, it will look up the value in m and return
// that, but if it isn't set then it will return the Default value // that, but if it isn't set then it will return the Default value
// passed in // passed in
func ConfirmWithConfig(m configmap.Getter, configName string, Default bool) bool { func ConfirmWithConfig(ctx context.Context, m configmap.Getter, configName string, Default bool) bool {
if fs.Config.AutoConfirm { if fs.Config.AutoConfirm {
configString, ok := m.Get(configName) configString, ok := m.Get(configName)
if ok { if ok {
@ -902,7 +902,7 @@ func RemoteConfig(name string) {
f := MustFindByName(name) f := MustFindByName(name)
if f.Config != nil { if f.Config != nil {
m := fs.ConfigMap(f, name) m := fs.ConfigMap(f, name)
f.Config(name, m) f.Config(context.Background(), name, m)
} }
} }
@ -1024,7 +1024,7 @@ func ChooseOption(o *fs.Option, name string) string {
} }
// Suppress the confirm prompts and return a function to undo that // Suppress the confirm prompts and return a function to undo that
func suppressConfirm() func() { func suppressConfirm(ctx context.Context) func() {
old := fs.Config.AutoConfirm old := fs.Config.AutoConfirm
fs.Config.AutoConfirm = true fs.Config.AutoConfirm = true
return func() { return func() {
@ -1034,7 +1034,7 @@ func suppressConfirm() func() {
// UpdateRemote adds the keyValues passed in to the remote of name. // UpdateRemote adds the keyValues passed in to the remote of name.
// keyValues should be key, value pairs. // keyValues should be key, value pairs.
func UpdateRemote(name string, keyValues rc.Params, doObscure, noObscure bool) error { func UpdateRemote(ctx context.Context, name string, keyValues rc.Params, doObscure, noObscure bool) error {
if doObscure && noObscure { if doObscure && noObscure {
return errors.New("can't use --obscure and --no-obscure together") return errors.New("can't use --obscure and --no-obscure together")
} }
@ -1042,7 +1042,7 @@ func UpdateRemote(name string, keyValues rc.Params, doObscure, noObscure bool) e
if err != nil { if err != nil {
return err return err
} }
defer suppressConfirm()() defer suppressConfirm(ctx)()
// Work out which options need to be obscured // Work out which options need to be obscured
needsObscure := map[string]struct{}{} needsObscure := map[string]struct{}{}
@ -1087,7 +1087,7 @@ func UpdateRemote(name string, keyValues rc.Params, doObscure, noObscure bool) e
// CreateRemote creates a new remote with name, provider and a list of // CreateRemote creates a new remote with name, provider and a list of
// parameters which are key, value pairs. If update is set then it // parameters which are key, value pairs. If update is set then it
// adds the new keys rather than replacing all of them. // adds the new keys rather than replacing all of them.
func CreateRemote(name string, provider string, keyValues rc.Params, doObscure, noObscure bool) error { func CreateRemote(ctx context.Context, name string, provider string, keyValues rc.Params, doObscure, noObscure bool) error {
err := fspath.CheckConfigName(name) err := fspath.CheckConfigName(name)
if err != nil { if err != nil {
return err return err
@ -1097,21 +1097,21 @@ func CreateRemote(name string, provider string, keyValues rc.Params, doObscure,
// Set the type // Set the type
getConfigData().SetValue(name, "type", provider) getConfigData().SetValue(name, "type", provider)
// Set the remaining values // Set the remaining values
return UpdateRemote(name, keyValues, doObscure, noObscure) return UpdateRemote(ctx, name, keyValues, doObscure, noObscure)
} }
// PasswordRemote adds the keyValues passed in to the remote of name. // PasswordRemote adds the keyValues passed in to the remote of name.
// keyValues should be key, value pairs. // keyValues should be key, value pairs.
func PasswordRemote(name string, keyValues rc.Params) error { func PasswordRemote(ctx context.Context, name string, keyValues rc.Params) error {
err := fspath.CheckConfigName(name) err := fspath.CheckConfigName(name)
if err != nil { if err != nil {
return err return err
} }
defer suppressConfirm()() defer suppressConfirm(ctx)()
for k, v := range keyValues { for k, v := range keyValues {
keyValues[k] = obscure.MustObscure(fmt.Sprint(v)) keyValues[k] = obscure.MustObscure(fmt.Sprint(v))
} }
return UpdateRemote(name, keyValues, false, true) return UpdateRemote(ctx, name, keyValues, false, true)
} }
// JSONListProviders prints all the providers and options in JSON format // JSONListProviders prints all the providers and options in JSON format
@ -1387,8 +1387,8 @@ func SetPassword() {
// //
// rclone authorize "fs name" // rclone authorize "fs name"
// rclone authorize "fs name" "client id" "client secret" // rclone authorize "fs name" "client id" "client secret"
func Authorize(args []string, noAutoBrowser bool) { func Authorize(ctx context.Context, args []string, noAutoBrowser bool) {
defer suppressConfirm()() defer suppressConfirm(ctx)()
switch len(args) { switch len(args) {
case 1, 3: case 1, 3:
default: default:
@ -1417,7 +1417,7 @@ func Authorize(args []string, noAutoBrowser bool) {
} }
m := fs.ConfigMap(f, name) m := fs.ConfigMap(f, name)
f.Config(name, m) f.Config(ctx, name, m)
} }
// FileGetFlag gets the config key under section returning the // FileGetFlag gets the config key under section returning the

View file

@ -180,6 +180,7 @@ func TestNewRemoteName(t *testing.T) {
} }
func TestCreateUpdatePasswordRemote(t *testing.T) { func TestCreateUpdatePasswordRemote(t *testing.T) {
ctx := context.Background()
defer testConfigFile(t, "update.conf")() defer testConfigFile(t, "update.conf")()
for _, doObscure := range []bool{false, true} { for _, doObscure := range []bool{false, true} {
@ -188,7 +189,7 @@ func TestCreateUpdatePasswordRemote(t *testing.T) {
break break
} }
t.Run(fmt.Sprintf("doObscure=%v,noObscure=%v", doObscure, noObscure), func(t *testing.T) { t.Run(fmt.Sprintf("doObscure=%v,noObscure=%v", doObscure, noObscure), func(t *testing.T) {
require.NoError(t, CreateRemote("test2", "config_test_remote", rc.Params{ require.NoError(t, CreateRemote(ctx, "test2", "config_test_remote", rc.Params{
"bool": true, "bool": true,
"pass": "potato", "pass": "potato",
}, doObscure, noObscure)) }, doObscure, noObscure))
@ -203,7 +204,7 @@ func TestCreateUpdatePasswordRemote(t *testing.T) {
assert.Equal(t, "potato", gotPw) assert.Equal(t, "potato", gotPw)
wantPw := obscure.MustObscure("potato2") wantPw := obscure.MustObscure("potato2")
require.NoError(t, UpdateRemote("test2", rc.Params{ require.NoError(t, UpdateRemote(ctx, "test2", rc.Params{
"bool": false, "bool": false,
"pass": wantPw, "pass": wantPw,
"spare": "spare", "spare": "spare",
@ -218,7 +219,7 @@ func TestCreateUpdatePasswordRemote(t *testing.T) {
} }
assert.Equal(t, wantPw, gotPw) assert.Equal(t, wantPw, gotPw)
require.NoError(t, PasswordRemote("test2", rc.Params{ require.NoError(t, PasswordRemote(ctx, "test2", rc.Params{
"pass": "potato3", "pass": "potato3",
})) }))
@ -427,8 +428,9 @@ func TestMatchProvider(t *testing.T) {
} }
func TestFileRefresh(t *testing.T) { func TestFileRefresh(t *testing.T) {
ctx := context.Background()
defer testConfigFile(t, "refresh.conf")() defer testConfigFile(t, "refresh.conf")()
require.NoError(t, CreateRemote("refresh_test", "config_test_remote", rc.Params{ require.NoError(t, CreateRemote(ctx, "refresh_test", "config_test_remote", rc.Params{
"bool": true, "bool": true,
}, false, false)) }, false, false))
b, err := ioutil.ReadFile(ConfigPath) b, err := ioutil.ReadFile(ConfigPath)

View file

@ -152,11 +152,11 @@ func rcConfig(ctx context.Context, in rc.Params, what string) (out rc.Params, er
if err != nil { if err != nil {
return nil, err return nil, err
} }
return nil, CreateRemote(name, remoteType, parameters, doObscure, noObscure) return nil, CreateRemote(ctx, name, remoteType, parameters, doObscure, noObscure)
case "update": case "update":
return nil, UpdateRemote(name, parameters, doObscure, noObscure) return nil, UpdateRemote(ctx, name, parameters, doObscure, noObscure)
case "password": case "password":
return nil, PasswordRemote(name, parameters) return nil, PasswordRemote(ctx, name, parameters)
} }
panic("unknown rcConfig type") panic("unknown rcConfig type")
} }

View file

@ -86,7 +86,7 @@ type RegInfo struct {
// the parent of that object and ErrorIsFile. // the parent of that object and ErrorIsFile.
NewFs func(ctx context.Context, name string, root string, config configmap.Mapper) (Fs, error) `json:"-"` NewFs func(ctx context.Context, name string, root string, config configmap.Mapper) (Fs, error) `json:"-"`
// Function to call to help with config // Function to call to help with config
Config func(name string, config configmap.Mapper) `json:"-"` Config func(ctx context.Context, name string, config configmap.Mapper) `json:"-"`
// Options for the Fs configuration // Options for the Fs configuration
Options Options Options Options
// The command help, if any // The command help, if any

View file

@ -288,8 +288,8 @@ func (ts *TokenSource) OnExpiry() <-chan time.Time {
var _ oauth2.TokenSource = (*TokenSource)(nil) var _ oauth2.TokenSource = (*TokenSource)(nil)
// Context returns a context with our HTTP Client baked in for oauth2 // Context returns a context with our HTTP Client baked in for oauth2
func Context(client *http.Client) context.Context { func Context(ctx context.Context, client *http.Client) context.Context {
return context.WithValue(context.Background(), oauth2.HTTPClient, client) return context.WithValue(ctx, oauth2.HTTPClient, client)
} }
// overrideCredentials sets the ClientID and ClientSecret from the // overrideCredentials sets the ClientID and ClientSecret from the
@ -327,7 +327,7 @@ func overrideCredentials(name string, m configmap.Mapper, origConfig *oauth2.Con
// configures a Client with it. It returns the client and a // configures a Client with it. It returns the client and a
// TokenSource which Invalidate may need to be called on. It uses the // TokenSource which Invalidate may need to be called on. It uses the
// httpClient passed in as the base client. // httpClient passed in as the base client.
func NewClientWithBaseClient(name string, m configmap.Mapper, config *oauth2.Config, baseClient *http.Client) (*http.Client, *TokenSource, error) { func NewClientWithBaseClient(ctx context.Context, name string, m configmap.Mapper, config *oauth2.Config, baseClient *http.Client) (*http.Client, *TokenSource, error) {
config, _ = overrideCredentials(name, m, config) config, _ = overrideCredentials(name, m, config)
token, err := GetToken(name, m) token, err := GetToken(name, m)
if err != nil { if err != nil {
@ -335,7 +335,7 @@ func NewClientWithBaseClient(name string, m configmap.Mapper, config *oauth2.Con
} }
// Set our own http client in the context // Set our own http client in the context
ctx := Context(baseClient) ctx = Context(ctx, baseClient)
// Wrap the TokenSource in our TokenSource which saves changed // Wrap the TokenSource in our TokenSource which saves changed
// tokens in the config file // tokens in the config file
@ -352,8 +352,8 @@ func NewClientWithBaseClient(name string, m configmap.Mapper, config *oauth2.Con
// NewClient gets a token from the config file and configures a Client // NewClient gets a token from the config file and configures a Client
// with it. It returns the client and a TokenSource which Invalidate may need to be called on // with it. It returns the client and a TokenSource which Invalidate may need to be called on
func NewClient(name string, m configmap.Mapper, oauthConfig *oauth2.Config) (*http.Client, *TokenSource, error) { func NewClient(ctx context.Context, name string, m configmap.Mapper, oauthConfig *oauth2.Config) (*http.Client, *TokenSource, error) {
return NewClientWithBaseClient(name, m, oauthConfig, fshttp.NewClient(fs.Config)) return NewClientWithBaseClient(ctx, name, m, oauthConfig, fshttp.NewClient(fs.Config))
} }
// AuthResult is returned from the web server after authorization // AuthResult is returned from the web server after authorization
@ -394,7 +394,7 @@ type Options struct {
// If opt is nil it will use the default Options // If opt is nil it will use the default Options
// //
// It may run an internal webserver to receive the results // It may run an internal webserver to receive the results
func Config(id, name string, m configmap.Mapper, oauthConfig *oauth2.Config, opt *Options) error { func Config(ctx context.Context, id, name string, m configmap.Mapper, oauthConfig *oauth2.Config, opt *Options) error {
if opt == nil { if opt == nil {
opt = &Options{} opt = &Options{}
} }
@ -408,7 +408,7 @@ func Config(id, name string, m configmap.Mapper, oauthConfig *oauth2.Config, opt
tokenString, ok := m.Get("token") tokenString, ok := m.Get("token")
if ok && tokenString != "" { if ok && tokenString != "" {
fmt.Printf("Already have a token - refresh?\n") fmt.Printf("Already have a token - refresh?\n")
if !config.ConfirmWithConfig(m, "config_refresh_token", true) { if !config.ConfirmWithConfig(ctx, m, "config_refresh_token", true) {
return nil return nil
} }
} }
@ -418,7 +418,7 @@ func Config(id, name string, m configmap.Mapper, oauthConfig *oauth2.Config, opt
fmt.Printf("Use auto config?\n") fmt.Printf("Use auto config?\n")
fmt.Printf(" * Say Y if not sure\n") fmt.Printf(" * Say Y if not sure\n")
fmt.Printf(" * Say N if you are working on a remote or headless machine\n") fmt.Printf(" * Say N if you are working on a remote or headless machine\n")
return config.ConfirmWithConfig(m, "config_is_local", true) return config.ConfirmWithConfig(ctx, m, "config_is_local", true)
} }
// Detect whether we should use internal web server // Detect whether we should use internal web server
@ -526,7 +526,7 @@ version recommended):
} }
// Exchange the code for a token // Exchange the code for a token
ctx := Context(fshttp.NewClient(fs.Config)) ctx = Context(ctx, fshttp.NewClient(fs.Config))
token, err := oauthConfig.Exchange(ctx, auth.Code) token, err := oauthConfig.Exchange(ctx, auth.Code)
if err != nil { if err != nil {
return errors.Wrap(err, "failed to get token") return errors.Wrap(err, "failed to get token")