forked from TrueCloudLab/rclone
backends: remove log.Fatal and replace with error returns #5234
This changes the Config interface so that it returns an error.
This commit is contained in:
parent
ef3c350686
commit
b78c9a65fa
28 changed files with 179 additions and 166 deletions
|
@ -16,7 +16,6 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -70,11 +69,12 @@ func init() {
|
||||||
Prefix: "acd",
|
Prefix: "acd",
|
||||||
Description: "Amazon Drive",
|
Description: "Amazon Drive",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "checkpoint",
|
Name: "checkpoint",
|
||||||
|
|
|
@ -17,7 +17,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -84,7 +83,7 @@ func init() {
|
||||||
Name: "box",
|
Name: "box",
|
||||||
Description: "Box",
|
Description: "Box",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
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")
|
||||||
|
@ -93,15 +92,16 @@ func init() {
|
||||||
if ok && boxSubTypeOk && jsonFile != "" && boxSubType != "" {
|
if ok && boxSubTypeOk && jsonFile != "" && boxSubType != "" {
|
||||||
err = refreshJWTToken(ctx, jsonFile, boxSubType, name, m)
|
err = refreshJWTToken(ctx, jsonFile, boxSubType, name, m)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to configure token with jwt authentication: %v", err)
|
return errors.Wrap(err, "failed to configure token with jwt authentication")
|
||||||
}
|
}
|
||||||
// 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(ctx, "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)
|
return errors.Wrap(err, "failed to configure token with oauth authentication")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "root_folder_id",
|
Name: "root_folder_id",
|
||||||
|
|
2
backend/cache/cache_internal_test.go
vendored
2
backend/cache/cache_internal_test.go
vendored
|
@ -836,7 +836,7 @@ func newRun() *run {
|
||||||
if uploadDir == "" {
|
if uploadDir == "" {
|
||||||
r.tmpUploadDir, err = ioutil.TempDir("", "rclonecache-tmp")
|
r.tmpUploadDir, err = ioutil.TempDir("", "rclonecache-tmp")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to create temp dir: %v", err)
|
panic(fmt.Sprintf("Failed to create temp dir: %v", err))
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
r.tmpUploadDir = uploadDir
|
r.tmpUploadDir = uploadDir
|
||||||
|
|
|
@ -14,7 +14,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"mime"
|
"mime"
|
||||||
"net/http"
|
"net/http"
|
||||||
"path"
|
"path"
|
||||||
|
@ -183,13 +182,12 @@ func init() {
|
||||||
Description: "Google Drive",
|
Description: "Google Drive",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
CommandHelp: commandHelp,
|
CommandHelp: commandHelp,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
// 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)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Errorf(nil, "Couldn't parse config into struct: %v", err)
|
return errors.Wrap(err, "couldn't parse config into struct")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fill in the scopes
|
// Fill in the scopes
|
||||||
|
@ -202,13 +200,14 @@ func init() {
|
||||||
if opt.ServiceAccountFile == "" && opt.ServiceAccountCredentials == "" {
|
if opt.ServiceAccountFile == "" && opt.ServiceAccountCredentials == "" {
|
||||||
err = oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err = configTeamDrive(ctx, opt, m, name)
|
err = configTeamDrive(ctx, opt, m, name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to configure Shared Drive: %v", err)
|
return errors.Wrap(err, "failed to configure Shared Drive")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(driveOAuthOptions(), []fs.Option{{
|
Options: append(driveOAuthOptions(), []fs.Option{{
|
||||||
Name: "scope",
|
Name: "scope",
|
||||||
|
@ -522,7 +521,7 @@ If this flag is set then rclone will ignore shortcut files completely.
|
||||||
} {
|
} {
|
||||||
for mimeType, extension := range m {
|
for mimeType, extension := range m {
|
||||||
if err := mime.AddExtensionType(extension, mimeType); err != nil {
|
if err := mime.AddExtensionType(extension, mimeType); err != nil {
|
||||||
log.Fatalf("Failed to register MIME type %q: %v", mimeType, err)
|
fs.Errorf("Failed to register MIME type %q: %v", mimeType, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,7 +25,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"path"
|
"path"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -144,7 +143,7 @@ func init() {
|
||||||
Name: "dropbox",
|
Name: "dropbox",
|
||||||
Description: "Dropbox",
|
Description: "Dropbox",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
opt := oauthutil.Options{
|
opt := oauthutil.Options{
|
||||||
NoOffline: true,
|
NoOffline: true,
|
||||||
OAuth2Opts: []oauth2.AuthCodeOption{
|
OAuth2Opts: []oauth2.AuthCodeOption{
|
||||||
|
@ -153,8 +152,9 @@ func init() {
|
||||||
}
|
}
|
||||||
err := oauthutil.Config(ctx, "dropbox", name, m, getOauthConfig(m), &opt)
|
err := oauthutil.Config(ctx, "dropbox", name, m, getOauthConfig(m), &opt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to configure token: %v", err)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "chunk_size",
|
Name: "chunk_size",
|
||||||
|
|
|
@ -35,8 +35,6 @@ func init() {
|
||||||
fs.Register(&fs.RegInfo{
|
fs.Register(&fs.RegInfo{
|
||||||
Name: "fichier",
|
Name: "fichier",
|
||||||
Description: "1Fichier",
|
Description: "1Fichier",
|
||||||
Config: func(ctx context.Context, name string, config configmap.Mapper) {
|
|
||||||
},
|
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Help: "Your API Key, get it from https://1fichier.com/console/params.pl",
|
Help: "Your API Key, get it from https://1fichier.com/console/params.pl",
|
||||||
|
|
|
@ -19,7 +19,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -76,17 +75,18 @@ 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(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
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 nil
|
||||||
}
|
}
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "project_number",
|
Name: "project_number",
|
||||||
|
|
|
@ -8,7 +8,6 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
golog "log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -78,13 +77,12 @@ func init() {
|
||||||
Prefix: "gphotos",
|
Prefix: "gphotos",
|
||||||
Description: "Google Photos",
|
Description: "Google Photos",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
// 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)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Errorf(nil, "Couldn't parse config into struct: %v", err)
|
return errors.Wrap(err, "couldn't parse config into struct")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fill in the scopes
|
// Fill in the scopes
|
||||||
|
@ -97,7 +95,7 @@ func init() {
|
||||||
// Do the oauth
|
// Do the oauth
|
||||||
err = oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Warn the user
|
// Warn the user
|
||||||
|
@ -108,6 +106,7 @@ func init() {
|
||||||
|
|
||||||
`)
|
`)
|
||||||
|
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "read_only",
|
Name: "read_only",
|
||||||
|
|
|
@ -11,7 +11,6 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
@ -56,11 +55,12 @@ func init() {
|
||||||
Name: "hubic",
|
Name: "hubic",
|
||||||
Description: "Hubic",
|
Description: "Hubic",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, swift.SharedOptions...),
|
Options: append(oauthutil.SharedOptions, swift.SharedOptions...),
|
||||||
})
|
})
|
||||||
|
|
|
@ -10,7 +10,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
@ -87,12 +86,12 @@ func init() {
|
||||||
Name: "jottacloud",
|
Name: "jottacloud",
|
||||||
Description: "Jottacloud",
|
Description: "Jottacloud",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
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)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to parse config version - corrupted config")
|
return errors.Wrap(err, "failed to parse config version - corrupted config")
|
||||||
}
|
}
|
||||||
refresh = (ver != configVersion) && (ver != v1configVersion)
|
refresh = (ver != configVersion) && (ver != v1configVersion)
|
||||||
}
|
}
|
||||||
|
@ -104,7 +103,7 @@ func init() {
|
||||||
if ok && tokenString != "" {
|
if ok && tokenString != "" {
|
||||||
fmt.Printf("Already have a token - refresh?\n")
|
fmt.Printf("Already have a token - refresh?\n")
|
||||||
if !config.Confirm(false) {
|
if !config.Confirm(false) {
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -116,11 +115,13 @@ func init() {
|
||||||
|
|
||||||
switch config.ChooseNumber("Your choice", 1, 3) {
|
switch config.ChooseNumber("Your choice", 1, 3) {
|
||||||
case 1:
|
case 1:
|
||||||
v2config(ctx, name, m)
|
return v2config(ctx, name, m)
|
||||||
case 2:
|
case 2:
|
||||||
v1config(ctx, name, m)
|
return v1config(ctx, name, m)
|
||||||
case 3:
|
case 3:
|
||||||
teliaCloudConfig(ctx, name, m)
|
return teliaCloudConfig(ctx, name, m)
|
||||||
|
default:
|
||||||
|
return errors.New("unknown config choice")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
|
@ -242,7 +243,7 @@ func shouldRetry(ctx context.Context, resp *http.Response, err error) (bool, err
|
||||||
return fserrors.ShouldRetry(err) || fserrors.ShouldRetryHTTP(resp, retryErrorCodes), err
|
return fserrors.ShouldRetry(err) || fserrors.ShouldRetryHTTP(resp, retryErrorCodes), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func teliaCloudConfig(ctx context.Context, name string, m configmap.Mapper) {
|
func teliaCloudConfig(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
teliaCloudOauthConfig := &oauth2.Config{
|
teliaCloudOauthConfig := &oauth2.Config{
|
||||||
Endpoint: oauth2.Endpoint{
|
Endpoint: oauth2.Endpoint{
|
||||||
AuthURL: teliaCloudAuthURL,
|
AuthURL: teliaCloudAuthURL,
|
||||||
|
@ -255,15 +256,14 @@ func teliaCloudConfig(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
|
|
||||||
err := oauthutil.Config(ctx, "jottacloud", name, m, teliaCloudOauthConfig, nil)
|
err := oauthutil.Config(ctx, "jottacloud", name, m, teliaCloudOauthConfig, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to configure token: %v", err)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
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(ctx, name, m, teliaCloudOauthConfig)
|
oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, teliaCloudOauthConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to load oAuthClient: %s", err)
|
return errors.Wrap(err, "failed to load oAuthClient")
|
||||||
}
|
}
|
||||||
|
|
||||||
srv := rest.NewClient(oAuthClient).SetRoot(rootURL)
|
srv := rest.NewClient(oAuthClient).SetRoot(rootURL)
|
||||||
|
@ -271,7 +271,7 @@ func teliaCloudConfig(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
|
|
||||||
device, mountpoint, err := setupMountpoint(ctx, srv, apiSrv)
|
device, mountpoint, err := setupMountpoint(ctx, srv, apiSrv)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to setup mountpoint: %s", err)
|
return errors.Wrap(err, "failed to setup mountpoint")
|
||||||
}
|
}
|
||||||
m.Set(configDevice, device)
|
m.Set(configDevice, device)
|
||||||
m.Set(configMountpoint, mountpoint)
|
m.Set(configMountpoint, mountpoint)
|
||||||
|
@ -280,17 +280,18 @@ func teliaCloudConfig(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
m.Set("configVersion", strconv.Itoa(configVersion))
|
m.Set("configVersion", strconv.Itoa(configVersion))
|
||||||
m.Set(configClientID, teliaCloudClientID)
|
m.Set(configClientID, teliaCloudClientID)
|
||||||
m.Set(configTokenURL, teliaCloudTokenURL)
|
m.Set(configTokenURL, teliaCloudTokenURL)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// v1config configure a jottacloud backend using legacy authentication
|
// v1config configure a jottacloud backend using legacy authentication
|
||||||
func v1config(ctx context.Context, name string, m configmap.Mapper) {
|
func v1config(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
srv := rest.NewClient(fshttp.NewClient(ctx))
|
srv := rest.NewClient(fshttp.NewClient(ctx))
|
||||||
|
|
||||||
fmt.Printf("\nDo you want to create a machine specific API key?\n\nRclone has it's own Jottacloud API KEY which works fine as long as one only uses rclone on a single machine. When you want to use rclone with this account on more than one machine it's recommended to create a machine specific API key. These keys can NOT be shared between machines.\n\n")
|
fmt.Printf("\nDo you want to create a machine specific API key?\n\nRclone has it's own Jottacloud API KEY which works fine as long as one only uses rclone on a single machine. When you want to use rclone with this account on more than one machine it's recommended to create a machine specific API key. These keys can NOT be shared between machines.\n\n")
|
||||||
if config.Confirm(false) {
|
if config.Confirm(false) {
|
||||||
deviceRegistration, err := registerDevice(ctx, srv)
|
deviceRegistration, err := registerDevice(ctx, srv)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to register device: %v", err)
|
return errors.Wrap(err, "failed to register device")
|
||||||
}
|
}
|
||||||
|
|
||||||
m.Set(configClientID, deviceRegistration.ClientID)
|
m.Set(configClientID, deviceRegistration.ClientID)
|
||||||
|
@ -318,18 +319,18 @@ func v1config(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
|
|
||||||
token, err := doAuthV1(ctx, srv, username, password)
|
token, err := doAuthV1(ctx, srv, username, password)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to get oauth token: %s", err)
|
return errors.Wrap(err, "failed to get oauth token")
|
||||||
}
|
}
|
||||||
err = oauthutil.PutToken(name, m, &token, true)
|
err = oauthutil.PutToken(name, m, &token, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Error while saving token: %s", err)
|
return errors.Wrap(err, "error while saving token")
|
||||||
}
|
}
|
||||||
|
|
||||||
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(ctx, 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)
|
return errors.Wrap(err, "failed to load oAuthClient")
|
||||||
}
|
}
|
||||||
|
|
||||||
srv = rest.NewClient(oAuthClient).SetRoot(rootURL)
|
srv = rest.NewClient(oAuthClient).SetRoot(rootURL)
|
||||||
|
@ -337,13 +338,14 @@ func v1config(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
|
|
||||||
device, mountpoint, err := setupMountpoint(ctx, srv, apiSrv)
|
device, mountpoint, err := setupMountpoint(ctx, srv, apiSrv)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to setup mountpoint: %s", err)
|
return errors.Wrap(err, "failed to setup mountpoint")
|
||||||
}
|
}
|
||||||
m.Set(configDevice, device)
|
m.Set(configDevice, device)
|
||||||
m.Set(configMountpoint, mountpoint)
|
m.Set(configMountpoint, mountpoint)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.Set("configVersion", strconv.Itoa(v1configVersion))
|
m.Set("configVersion", strconv.Itoa(v1configVersion))
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// registerDevice register a new device for use with the jottacloud API
|
// registerDevice register a new device for use with the jottacloud API
|
||||||
|
@ -418,7 +420,7 @@ func doAuthV1(ctx context.Context, srv *rest.Client, username, password string)
|
||||||
}
|
}
|
||||||
|
|
||||||
// v2config configure a jottacloud backend using the modern JottaCli token based authentication
|
// v2config configure a jottacloud backend using the modern JottaCli token based authentication
|
||||||
func v2config(ctx context.Context, name string, m configmap.Mapper) {
|
func v2config(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
srv := rest.NewClient(fshttp.NewClient(ctx))
|
srv := rest.NewClient(fshttp.NewClient(ctx))
|
||||||
|
|
||||||
fmt.Printf("Generate a personal login token here: https://www.jottacloud.com/web/secure\n")
|
fmt.Printf("Generate a personal login token here: https://www.jottacloud.com/web/secure\n")
|
||||||
|
@ -430,31 +432,32 @@ func v2config(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
|
|
||||||
token, err := doAuthV2(ctx, srv, loginToken, m)
|
token, err := doAuthV2(ctx, srv, loginToken, m)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to get oauth token: %s", err)
|
return errors.Wrap(err, "failed to get oauth token")
|
||||||
}
|
}
|
||||||
err = oauthutil.PutToken(name, m, &token, true)
|
err = oauthutil.PutToken(name, m, &token, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Error while saving token: %s", err)
|
return errors.Wrap(err, "error while saving token")
|
||||||
}
|
}
|
||||||
|
|
||||||
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(ctx, 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)
|
return errors.Wrap(err, "failed to load oAuthClient")
|
||||||
}
|
}
|
||||||
|
|
||||||
srv = rest.NewClient(oAuthClient).SetRoot(rootURL)
|
srv = rest.NewClient(oAuthClient).SetRoot(rootURL)
|
||||||
apiSrv := rest.NewClient(oAuthClient).SetRoot(apiURL)
|
apiSrv := rest.NewClient(oAuthClient).SetRoot(apiURL)
|
||||||
device, mountpoint, err := setupMountpoint(ctx, srv, apiSrv)
|
device, mountpoint, err := setupMountpoint(ctx, srv, apiSrv)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to setup mountpoint: %s", err)
|
return errors.Wrap(err, "failed to setup mountpoint")
|
||||||
}
|
}
|
||||||
m.Set(configDevice, device)
|
m.Set(configDevice, device)
|
||||||
m.Set(configMountpoint, mountpoint)
|
m.Set(configMountpoint, mountpoint)
|
||||||
}
|
}
|
||||||
|
|
||||||
m.Set("configVersion", strconv.Itoa(configVersion))
|
m.Set("configVersion", strconv.Itoa(configVersion))
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// doAuthV2 runs the actual token request for V2 authentication
|
// doAuthV2 runs the actual token request for V2 authentication
|
||||||
|
|
|
@ -6,8 +6,8 @@ import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
|
@ -48,7 +48,7 @@ func (w *BinWriter) Reader() io.Reader {
|
||||||
// WritePu16 writes a short as unsigned varint
|
// WritePu16 writes a short as unsigned varint
|
||||||
func (w *BinWriter) WritePu16(val int) {
|
func (w *BinWriter) WritePu16(val int) {
|
||||||
if val < 0 || val > 65535 {
|
if val < 0 || val > 65535 {
|
||||||
log.Fatalf("Invalid UInt16 %v", val)
|
panic(fmt.Sprintf("Invalid UInt16 %v", val))
|
||||||
}
|
}
|
||||||
w.WritePu64(int64(val))
|
w.WritePu64(int64(val))
|
||||||
}
|
}
|
||||||
|
@ -56,7 +56,7 @@ func (w *BinWriter) WritePu16(val int) {
|
||||||
// WritePu32 writes a signed long as unsigned varint
|
// WritePu32 writes a signed long as unsigned varint
|
||||||
func (w *BinWriter) WritePu32(val int64) {
|
func (w *BinWriter) WritePu32(val int64) {
|
||||||
if val < 0 || val > 4294967295 {
|
if val < 0 || val > 4294967295 {
|
||||||
log.Fatalf("Invalid UInt32 %v", val)
|
panic(fmt.Sprintf("Invalid UInt32 %v", val))
|
||||||
}
|
}
|
||||||
w.WritePu64(val)
|
w.WritePu64(val)
|
||||||
}
|
}
|
||||||
|
@ -64,7 +64,7 @@ func (w *BinWriter) WritePu32(val int64) {
|
||||||
// WritePu64 writes an unsigned (actually, signed) long as unsigned varint
|
// WritePu64 writes an unsigned (actually, signed) long as unsigned varint
|
||||||
func (w *BinWriter) WritePu64(val int64) {
|
func (w *BinWriter) WritePu64(val int64) {
|
||||||
if val < 0 {
|
if val < 0 {
|
||||||
log.Fatalf("Invalid UInt64 %v", val)
|
panic(fmt.Sprintf("Invalid UInt64 %v", val))
|
||||||
}
|
}
|
||||||
w.b.Write(w.a[:binary.PutUvarint(w.a, uint64(val))])
|
w.b.Write(w.a[:binary.PutUvarint(w.a, uint64(val))])
|
||||||
}
|
}
|
||||||
|
@ -123,7 +123,7 @@ func (r *BinReader) check(err error) bool {
|
||||||
r.err = err
|
r.err = err
|
||||||
}
|
}
|
||||||
if err != io.EOF {
|
if err != io.EOF {
|
||||||
log.Fatalf("Error parsing response: %v", err)
|
panic(fmt.Sprintf("Error parsing response: %v", err))
|
||||||
}
|
}
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,6 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -99,7 +98,7 @@ func init() {
|
||||||
Name: "onedrive",
|
Name: "onedrive",
|
||||||
Description: "Microsoft OneDrive",
|
Description: "Microsoft OneDrive",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
region, _ := m.Get("region")
|
region, _ := m.Get("region")
|
||||||
graphURL := graphAPIEndpoint[region] + "/v1.0"
|
graphURL := graphAPIEndpoint[region] + "/v1.0"
|
||||||
oauthConfig.Endpoint = oauth2.Endpoint{
|
oauthConfig.Endpoint = oauth2.Endpoint{
|
||||||
|
@ -109,13 +108,12 @@ func init() {
|
||||||
ci := fs.GetConfig(ctx)
|
ci := fs.GetConfig(ctx)
|
||||||
err := oauthutil.Config(ctx, "onedrive", name, m, oauthConfig, nil)
|
err := oauthutil.Config(ctx, "onedrive", name, m, oauthConfig, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to configure token: %v", err)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stop if we are running non-interactive config
|
// Stop if we are running non-interactive config
|
||||||
if ci.AutoConfirm {
|
if ci.AutoConfirm {
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
type driveResource struct {
|
type driveResource struct {
|
||||||
|
@ -138,7 +136,7 @@ func init() {
|
||||||
|
|
||||||
oAuthClient, _, err := oauthutil.NewClient(ctx, 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)
|
return errors.Wrap(err, "failed to configure OneDrive")
|
||||||
}
|
}
|
||||||
srv := rest.NewClient(oAuthClient)
|
srv := rest.NewClient(oAuthClient)
|
||||||
|
|
||||||
|
@ -203,19 +201,18 @@ func init() {
|
||||||
sites := siteResponse{}
|
sites := siteResponse{}
|
||||||
_, err := srv.CallJSON(ctx, &opts, nil, &sites)
|
_, err := srv.CallJSON(ctx, &opts, nil, &sites)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to query available sites: %v", err)
|
return errors.Wrap(err, "failed to query available sites")
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(sites.Sites) == 0 {
|
if len(sites.Sites) == 0 {
|
||||||
log.Fatalf("Search for '%s' returned no results", searchTerm)
|
return errors.Errorf("search for %q returned no results", searchTerm)
|
||||||
} else {
|
}
|
||||||
fmt.Printf("Found %d sites, please select the one you want to use:\n", len(sites.Sites))
|
fmt.Printf("Found %d sites, please select the one you want to use:\n", len(sites.Sites))
|
||||||
for index, site := range sites.Sites {
|
for index, site := range sites.Sites {
|
||||||
fmt.Printf("%d: %s (%s) id=%s\n", index, site.SiteName, site.SiteURL, site.SiteID)
|
fmt.Printf("%d: %s (%s) id=%s\n", index, site.SiteName, site.SiteURL, site.SiteID)
|
||||||
}
|
}
|
||||||
siteID = sites.Sites[config.ChooseNumber("Chose drive to use:", 0, len(sites.Sites)-1)].SiteID
|
siteID = sites.Sites[config.ChooseNumber("Chose drive to use:", 0, len(sites.Sites)-1)].SiteID
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// if we use server-relative URL for finding the drive
|
// if we use server-relative URL for finding the drive
|
||||||
if relativePath != "" {
|
if relativePath != "" {
|
||||||
|
@ -227,7 +224,7 @@ func init() {
|
||||||
site := siteResource{}
|
site := siteResource{}
|
||||||
_, err := srv.CallJSON(ctx, &opts, nil, &site)
|
_, err := srv.CallJSON(ctx, &opts, nil, &site)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to query available site by relative path: %v", err)
|
return errors.Wrap(err, "failed to query available site by relative path")
|
||||||
}
|
}
|
||||||
siteID = site.SiteID
|
siteID = site.SiteID
|
||||||
}
|
}
|
||||||
|
@ -247,7 +244,7 @@ func init() {
|
||||||
drives := drivesResponse{}
|
drives := drivesResponse{}
|
||||||
_, err := srv.CallJSON(ctx, &opts, nil, &drives)
|
_, err := srv.CallJSON(ctx, &opts, nil, &drives)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to query available drives: %v", err)
|
return errors.Wrap(err, "failed to query available drives")
|
||||||
}
|
}
|
||||||
|
|
||||||
// Also call /me/drive as sometimes /me/drives doesn't return it #4068
|
// Also call /me/drive as sometimes /me/drives doesn't return it #4068
|
||||||
|
@ -256,7 +253,7 @@ func init() {
|
||||||
meDrive := driveResource{}
|
meDrive := driveResource{}
|
||||||
_, err := srv.CallJSON(ctx, &opts, nil, &meDrive)
|
_, err := srv.CallJSON(ctx, &opts, nil, &meDrive)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to query available drives: %v", err)
|
return errors.Wrap(err, "failed to query available drives")
|
||||||
}
|
}
|
||||||
found := false
|
found := false
|
||||||
for _, drive := range drives.Drives {
|
for _, drive := range drives.Drives {
|
||||||
|
@ -273,15 +270,14 @@ func init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(drives.Drives) == 0 {
|
if len(drives.Drives) == 0 {
|
||||||
log.Fatalf("No drives found")
|
return errors.New("no drives found")
|
||||||
} else {
|
}
|
||||||
fmt.Printf("Found %d drives, please select the one you want to use:\n", len(drives.Drives))
|
fmt.Printf("Found %d drives, please select the one you want to use:\n", len(drives.Drives))
|
||||||
for index, drive := range drives.Drives {
|
for index, drive := range drives.Drives {
|
||||||
fmt.Printf("%d: %s (%s) id=%s\n", index, drive.DriveName, drive.DriveType, drive.DriveID)
|
fmt.Printf("%d: %s (%s) id=%s\n", index, drive.DriveName, drive.DriveType, drive.DriveID)
|
||||||
}
|
}
|
||||||
finalDriveID = drives.Drives[config.ChooseNumber("Chose drive to use:", 0, len(drives.Drives)-1)].DriveID
|
finalDriveID = drives.Drives[config.ChooseNumber("Chose drive to use:", 0, len(drives.Drives)-1)].DriveID
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Test the driveID and get drive type
|
// Test the driveID and get drive type
|
||||||
opts = rest.Opts{
|
opts = rest.Opts{
|
||||||
|
@ -291,17 +287,18 @@ func init() {
|
||||||
var rootItem api.Item
|
var rootItem api.Item
|
||||||
_, err = srv.CallJSON(ctx, &opts, nil, &rootItem)
|
_, err = srv.CallJSON(ctx, &opts, nil, &rootItem)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to query root for drive %s: %v", finalDriveID, err)
|
return errors.Wrapf(err, "failed to query root for drive %s", finalDriveID)
|
||||||
}
|
}
|
||||||
|
|
||||||
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(ctx, m, "config_drive_ok", true) {
|
if !config.ConfirmWithConfig(ctx, m, "config_drive_ok", true) {
|
||||||
log.Fatalf("Cancelled by user")
|
return errors.New("cancelled by user")
|
||||||
}
|
}
|
||||||
|
|
||||||
m.Set(configDriveID, finalDriveID)
|
m.Set(configDriveID, finalDriveID)
|
||||||
m.Set(configDriveType, rootItem.ParentReference.DriveType)
|
m.Set(configDriveType, rootItem.ParentReference.DriveType)
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "region",
|
Name: "region",
|
||||||
|
|
|
@ -12,7 +12,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -72,7 +71,7 @@ func init() {
|
||||||
Name: "pcloud",
|
Name: "pcloud",
|
||||||
Description: "Pcloud",
|
Description: "Pcloud",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
optc := new(Options)
|
optc := new(Options)
|
||||||
err := configstruct.Set(m, optc)
|
err := configstruct.Set(m, optc)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -100,8 +99,9 @@ func init() {
|
||||||
}
|
}
|
||||||
err = oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: config.ConfigEncoding,
|
Name: config.ConfigEncoding,
|
||||||
|
|
|
@ -20,7 +20,6 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
@ -78,11 +77,12 @@ func init() {
|
||||||
Name: "premiumizeme",
|
Name: "premiumizeme",
|
||||||
Description: "premiumize.me",
|
Description: "premiumize.me",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Name: "api_key",
|
Name: "api_key",
|
||||||
|
|
|
@ -2,10 +2,10 @@ package putio
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"log"
|
|
||||||
"regexp"
|
"regexp"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
"github.com/rclone/rclone/fs"
|
"github.com/rclone/rclone/fs"
|
||||||
"github.com/rclone/rclone/fs/config"
|
"github.com/rclone/rclone/fs/config"
|
||||||
"github.com/rclone/rclone/fs/config/configmap"
|
"github.com/rclone/rclone/fs/config/configmap"
|
||||||
|
@ -60,14 +60,15 @@ func init() {
|
||||||
Name: "putio",
|
Name: "putio",
|
||||||
Description: "Put.io",
|
Description: "Put.io",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
opt := oauthutil.Options{
|
opt := oauthutil.Options{
|
||||||
NoOffline: true,
|
NoOffline: true,
|
||||||
}
|
}
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Name: config.ConfigEncoding,
|
Name: config.ConfigEncoding,
|
||||||
|
|
|
@ -296,36 +296,32 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
|
||||||
}
|
}
|
||||||
|
|
||||||
// Config callback for 2FA
|
// Config callback for 2FA
|
||||||
func Config(ctx context.Context, name string, m configmap.Mapper) {
|
func Config(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
ci := fs.GetConfig(ctx)
|
ci := fs.GetConfig(ctx)
|
||||||
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"
|
||||||
fmt.Print("\nOperation not supported on this remote.\nIf you need a 2FA code on your account, use the command:\n\nrclone config reconnect <remote name>:\n\n")
|
return errors.New("operation not supported on this remote. If you need a 2FA code on your account, use the command: nrclone config reconnect <remote name>: ")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stop if we are running non-interactive config
|
// Stop if we are running non-interactive config
|
||||||
if ci.AutoConfirm {
|
if ci.AutoConfirm {
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
u, err := url.Parse(serverURL)
|
u, err := url.Parse(serverURL)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fs.Errorf(nil, "Invalid server URL %s", serverURL)
|
return errors.Errorf("invalid server URL %s", serverURL)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
is2faEnabled, _ := m.Get(config2FA)
|
is2faEnabled, _ := m.Get(config2FA)
|
||||||
if is2faEnabled != "true" {
|
if is2faEnabled != "true" {
|
||||||
fmt.Println("Two-factor authentication is not enabled on this account.")
|
return errors.New("two-factor authentication is not enabled on this account")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
username, _ := m.Get(configUser)
|
username, _ := m.Get(configUser)
|
||||||
if username == "" {
|
if username == "" {
|
||||||
fs.Errorf(nil, "A username is required")
|
return errors.New("a username is required")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
password, _ := m.Get(configPassword)
|
password, _ := m.Get(configPassword)
|
||||||
|
@ -376,6 +372,7 @@ func Config(ctx context.Context, name string, m configmap.Mapper) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// sets the AuthorizationToken up
|
// sets the AuthorizationToken up
|
||||||
|
|
|
@ -77,7 +77,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -136,7 +135,7 @@ func init() {
|
||||||
Name: "sharefile",
|
Name: "sharefile",
|
||||||
Description: "Citrix Sharefile",
|
Description: "Citrix Sharefile",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
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 {
|
||||||
|
@ -157,8 +156,9 @@ func init() {
|
||||||
}
|
}
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Name: "upload_cutoff",
|
Name: "upload_cutoff",
|
||||||
|
|
|
@ -16,7 +16,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -76,17 +75,17 @@ func init() {
|
||||||
Name: "sugarsync",
|
Name: "sugarsync",
|
||||||
Description: "Sugarsync",
|
Description: "Sugarsync",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
opt := new(Options)
|
opt := new(Options)
|
||||||
err := configstruct.Set(m, opt)
|
err := configstruct.Set(m, opt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to read options: %v", err)
|
return errors.Wrap(err, "failed to read options")
|
||||||
}
|
}
|
||||||
|
|
||||||
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(ctx, m, "config_refresh_token", true) {
|
if !config.ConfirmWithConfig(ctx, m, "config_refresh_token", true) {
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fmt.Printf("Username (email address)> ")
|
fmt.Printf("Username (email address)> ")
|
||||||
|
@ -114,10 +113,11 @@ func init() {
|
||||||
// return shouldRetry(ctx, resp, err)
|
// return shouldRetry(ctx, resp, err)
|
||||||
//})
|
//})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to get token: %v", err)
|
return errors.Wrap(err, "failed to get token")
|
||||||
}
|
}
|
||||||
opt.RefreshToken = resp.Header.Get("Location")
|
opt.RefreshToken = resp.Header.Get("Location")
|
||||||
m.Set("refresh_token", opt.RefreshToken)
|
m.Set("refresh_token", opt.RefreshToken)
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Name: "app_id",
|
Name: "app_id",
|
||||||
|
|
|
@ -7,7 +7,6 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"log"
|
|
||||||
"path"
|
"path"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
@ -42,7 +41,7 @@ func init() {
|
||||||
Name: "tardigrade",
|
Name: "tardigrade",
|
||||||
Description: "Tardigrade Decentralized Cloud Storage",
|
Description: "Tardigrade Decentralized Cloud Storage",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, configMapper configmap.Mapper) {
|
Config: func(ctx context.Context, name string, configMapper configmap.Mapper) error {
|
||||||
provider, _ := configMapper.Get(fs.ConfigProvider)
|
provider, _ := configMapper.Get(fs.ConfigProvider)
|
||||||
|
|
||||||
config.FileDeleteKey(name, fs.ConfigProvider)
|
config.FileDeleteKey(name, fs.ConfigProvider)
|
||||||
|
@ -54,7 +53,7 @@ func init() {
|
||||||
|
|
||||||
// satelliteString contains always default and passphrase can be empty
|
// satelliteString contains always default and passphrase can be empty
|
||||||
if apiKey == "" {
|
if apiKey == "" {
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
satellite, found := satMap[satelliteString]
|
satellite, found := satMap[satelliteString]
|
||||||
|
@ -64,12 +63,12 @@ func init() {
|
||||||
|
|
||||||
access, err := uplink.RequestAccessWithPassphrase(context.TODO(), satellite, apiKey, passphrase)
|
access, err := uplink.RequestAccessWithPassphrase(context.TODO(), satellite, apiKey, passphrase)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Couldn't create access grant: %v", err)
|
return errors.Wrap(err, "couldn't create access grant")
|
||||||
}
|
}
|
||||||
|
|
||||||
serializedAccess, err := access.Serialize()
|
serializedAccess, err := access.Serialize()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Couldn't serialize access grant: %v", err)
|
return errors.Wrap(err, "couldn't serialize access grant")
|
||||||
}
|
}
|
||||||
configMapper.Set("satellite_address", satellite)
|
configMapper.Set("satellite_address", satellite)
|
||||||
configMapper.Set("access_grant", serializedAccess)
|
configMapper.Set("access_grant", serializedAccess)
|
||||||
|
@ -78,8 +77,9 @@ func init() {
|
||||||
config.FileDeleteKey(name, "api_key")
|
config.FileDeleteKey(name, "api_key")
|
||||||
config.FileDeleteKey(name, "passphrase")
|
config.FileDeleteKey(name, "passphrase")
|
||||||
} else {
|
} else {
|
||||||
log.Fatalf("Invalid provider type: %s", provider)
|
return errors.Errorf("invalid provider type: %s", provider)
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: []fs.Option{
|
Options: []fs.Option{
|
||||||
{
|
{
|
||||||
|
|
|
@ -41,8 +41,6 @@ func init() {
|
||||||
fs.Register(&fs.RegInfo{
|
fs.Register(&fs.RegInfo{
|
||||||
Name: "uptobox",
|
Name: "uptobox",
|
||||||
Description: "Uptobox",
|
Description: "Uptobox",
|
||||||
Config: func(ctx context.Context, name string, config configmap.Mapper) {
|
|
||||||
},
|
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Options: []fs.Option{{
|
Options: []fs.Option{{
|
||||||
Help: "Your access Token, get it from https://uptobox.com/my_account",
|
Help: "Your access Token, get it from https://uptobox.com/my_account",
|
||||||
|
|
|
@ -60,12 +60,12 @@ func init() {
|
||||||
Name: "yandex",
|
Name: "yandex",
|
||||||
Description: "Yandex Disk",
|
Description: "Yandex Disk",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
err := oauthutil.Config(ctx, "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)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: config.ConfigEncoding,
|
Name: config.ConfigEncoding,
|
||||||
|
@ -251,22 +251,22 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
|
||||||
|
|
||||||
token, err := oauthutil.GetToken(name, m)
|
token, err := oauthutil.GetToken(name, m)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Couldn't read OAuth token (this should never happen).")
|
return nil, errors.Wrap(err, "couldn't read OAuth token")
|
||||||
}
|
}
|
||||||
if token.RefreshToken == "" {
|
if token.RefreshToken == "" {
|
||||||
log.Fatalf("Unable to get RefreshToken. If you are upgrading from older versions of rclone, please run `rclone config` and re-configure this backend.")
|
return nil, errors.New("unable to get RefreshToken. If you are upgrading from older versions of rclone, please run `rclone config` and re-configure this backend")
|
||||||
}
|
}
|
||||||
if token.TokenType != "OAuth" {
|
if token.TokenType != "OAuth" {
|
||||||
token.TokenType = "OAuth"
|
token.TokenType = "OAuth"
|
||||||
err = oauthutil.PutToken(name, m, token, false)
|
err = oauthutil.PutToken(name, m, token, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Couldn't save OAuth token (this should never happen).")
|
return nil, errors.Wrap(err, "couldn't save OAuth token")
|
||||||
}
|
}
|
||||||
log.Printf("Automatically upgraded OAuth config.")
|
log.Printf("Automatically upgraded OAuth config.")
|
||||||
}
|
}
|
||||||
oAuthClient, _, err := oauthutil.NewClient(ctx, 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)
|
return nil, errors.Wrap(err, "failed to configure Yandex")
|
||||||
}
|
}
|
||||||
|
|
||||||
ci := fs.GetConfig(ctx)
|
ci := fs.GetConfig(ctx)
|
||||||
|
|
|
@ -7,7 +7,6 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
"net/url"
|
||||||
"path"
|
"path"
|
||||||
|
@ -73,37 +72,41 @@ func init() {
|
||||||
Name: "zoho",
|
Name: "zoho",
|
||||||
Description: "Zoho",
|
Description: "Zoho",
|
||||||
NewFs: NewFs,
|
NewFs: NewFs,
|
||||||
Config: func(ctx context.Context, name string, m configmap.Mapper) {
|
Config: func(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
// Need to setup region before configuring oauth
|
// Need to setup region before configuring oauth
|
||||||
setupRegion(m)
|
err := setupRegion(m)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
opt := oauthutil.Options{
|
opt := oauthutil.Options{
|
||||||
// No refresh token unless ApprovalForce is set
|
// No refresh token unless ApprovalForce is set
|
||||||
OAuth2Opts: []oauth2.AuthCodeOption{oauth2.ApprovalForce},
|
OAuth2Opts: []oauth2.AuthCodeOption{oauth2.ApprovalForce},
|
||||||
}
|
}
|
||||||
if err := oauthutil.Config(ctx, "zoho", name, m, oauthConfig, &opt); err != nil {
|
if err := oauthutil.Config(ctx, "zoho", name, m, oauthConfig, &opt); err != nil {
|
||||||
log.Fatalf("Failed to configure token: %v", err)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
// We need to rewrite the token type to "Zoho-oauthtoken" because Zoho wants
|
// We need to rewrite the token type to "Zoho-oauthtoken" because Zoho wants
|
||||||
// it's own custom type
|
// it's own custom type
|
||||||
token, err := oauthutil.GetToken(name, m)
|
token, err := oauthutil.GetToken(name, m)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to read token: %v", err)
|
return errors.Wrap(err, "failed to read token")
|
||||||
}
|
}
|
||||||
if token.TokenType != "Zoho-oauthtoken" {
|
if token.TokenType != "Zoho-oauthtoken" {
|
||||||
token.TokenType = "Zoho-oauthtoken"
|
token.TokenType = "Zoho-oauthtoken"
|
||||||
err = oauthutil.PutToken(name, m, token, false)
|
err = oauthutil.PutToken(name, m, token, false)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to configure token: %v", err)
|
return errors.Wrap(err, "failed to configure token")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if fs.GetConfig(ctx).AutoConfirm {
|
if fs.GetConfig(ctx).AutoConfirm {
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if err = setupRoot(ctx, name, m); err != nil {
|
if err = setupRoot(ctx, name, m); err != nil {
|
||||||
log.Fatalf("Failed to configure root directory: %v", err)
|
return errors.Wrap(err, "failed to configure root directory")
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
},
|
},
|
||||||
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
Options: append(oauthutil.SharedOptions, []fs.Option{{
|
||||||
Name: "region",
|
Name: "region",
|
||||||
|
@ -164,15 +167,16 @@ type Object struct {
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
// ------------------------------------------------------------
|
||||||
|
|
||||||
func setupRegion(m configmap.Mapper) {
|
func setupRegion(m configmap.Mapper) error {
|
||||||
region, ok := m.Get("region")
|
region, ok := m.Get("region")
|
||||||
if !ok || region == "" {
|
if !ok || region == "" {
|
||||||
log.Fatalf("No region set\n")
|
return errors.New("no region set")
|
||||||
}
|
}
|
||||||
rootURL = fmt.Sprintf("https://workdrive.zoho.%s/api/v1", region)
|
rootURL = fmt.Sprintf("https://workdrive.zoho.%s/api/v1", region)
|
||||||
accountsURL = fmt.Sprintf("https://accounts.zoho.%s", region)
|
accountsURL = fmt.Sprintf("https://accounts.zoho.%s", region)
|
||||||
oauthConfig.Endpoint.AuthURL = fmt.Sprintf("https://accounts.zoho.%s/oauth/v2/auth", region)
|
oauthConfig.Endpoint.AuthURL = fmt.Sprintf("https://accounts.zoho.%s/oauth/v2/auth", region)
|
||||||
oauthConfig.Endpoint.TokenURL = fmt.Sprintf("https://accounts.zoho.%s/oauth/v2/token", region)
|
oauthConfig.Endpoint.TokenURL = fmt.Sprintf("https://accounts.zoho.%s/oauth/v2/token", region)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// ------------------------------------------------------------
|
// ------------------------------------------------------------
|
||||||
|
@ -208,7 +212,7 @@ func listWorkspaces(ctx context.Context, teamID string, srv *rest.Client) ([]api
|
||||||
func setupRoot(ctx context.Context, name string, m configmap.Mapper) error {
|
func setupRoot(ctx context.Context, name string, m configmap.Mapper) error {
|
||||||
oAuthClient, _, err := oauthutil.NewClient(ctx, 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)
|
return errors.Wrap(err, "failed to load oAuthClient")
|
||||||
}
|
}
|
||||||
authSrv := rest.NewClient(oAuthClient).SetRoot(accountsURL)
|
authSrv := rest.NewClient(oAuthClient).SetRoot(accountsURL)
|
||||||
opts := rest.Opts{
|
opts := rest.Opts{
|
||||||
|
@ -377,7 +381,10 @@ func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, e
|
||||||
if err := configstruct.Set(m, opt); err != nil {
|
if err := configstruct.Set(m, opt); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
setupRegion(m)
|
err := setupRegion(m)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
root = parsePath(root)
|
root = parsePath(root)
|
||||||
oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
|
oAuthClient, _, err := oauthutil.NewClient(ctx, name, m, oauthConfig)
|
||||||
|
|
|
@ -42,9 +42,9 @@ var configCommand = &cobra.Command{
|
||||||
remotes and manage existing ones. You may also set or remove a
|
remotes and manage existing ones. You may also set or remove a
|
||||||
password to protect your configuration.
|
password to protect your configuration.
|
||||||
`,
|
`,
|
||||||
Run: func(command *cobra.Command, args []string) {
|
RunE: func(command *cobra.Command, args []string) error {
|
||||||
cmd.CheckArgs(0, 0, command, args)
|
cmd.CheckArgs(0, 0, command, args)
|
||||||
config.EditConfig(context.Background())
|
return config.EditConfig(context.Background())
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -272,8 +272,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(ctx, configName, config)
|
return fsInfo.Config(ctx, configName, config)
|
||||||
return nil
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,7 +60,10 @@ func Authorize(ctx context.Context, args []string, noAutoBrowser bool) error {
|
||||||
m.AddSetter(outM)
|
m.AddSetter(outM)
|
||||||
m.AddGetter(outM, configmap.PriorityNormal)
|
m.AddGetter(outM, configmap.PriorityNormal)
|
||||||
|
|
||||||
ri.Config(ctx, name, m)
|
err = ri.Config(ctx, name, m)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
// Print the code for the user to paste
|
// Print the code for the user to paste
|
||||||
out := outM["token"]
|
out := outM["token"]
|
||||||
|
|
|
@ -455,7 +455,10 @@ func UpdateRemote(ctx context.Context, name string, keyValues rc.Params, doObscu
|
||||||
}
|
}
|
||||||
LoadedData().SetValue(name, k, vStr)
|
LoadedData().SetValue(name, k, vStr)
|
||||||
}
|
}
|
||||||
RemoteConfig(ctx, name)
|
err = RemoteConfig(ctx, name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
SaveConfig()
|
SaveConfig()
|
||||||
cache.ClearConfig(name) // remove any remotes based on this config from the cache
|
cache.ClearConfig(name) // remove any remotes based on this config from the cache
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -270,13 +270,14 @@ func OkRemote(name string) bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
// RemoteConfig runs the config helper for the remote if needed
|
// RemoteConfig runs the config helper for the remote if needed
|
||||||
func RemoteConfig(ctx context.Context, name string) {
|
func RemoteConfig(ctx context.Context, name string) error {
|
||||||
fmt.Printf("Remote config\n")
|
fmt.Printf("Remote config\n")
|
||||||
f := mustFindByName(name)
|
f := mustFindByName(name)
|
||||||
if f.Config != nil {
|
if f.Config != nil {
|
||||||
m := fs.ConfigMap(f, name, nil)
|
m := fs.ConfigMap(f, name, nil)
|
||||||
f.Config(ctx, name, m)
|
return f.Config(ctx, name, m)
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// matchProvider returns true if provider matches the providerConfig string.
|
// matchProvider returns true if provider matches the providerConfig string.
|
||||||
|
@ -456,7 +457,7 @@ func editOptions(ri *fs.RegInfo, name string, isNew bool) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewRemote make a new remote from its name
|
// NewRemote make a new remote from its name
|
||||||
func NewRemote(ctx context.Context, name string) {
|
func NewRemote(ctx context.Context, name string) error {
|
||||||
var (
|
var (
|
||||||
newType string
|
newType string
|
||||||
ri *fs.RegInfo
|
ri *fs.RegInfo
|
||||||
|
@ -476,16 +477,19 @@ func NewRemote(ctx context.Context, name string) {
|
||||||
LoadedData().SetValue(name, "type", newType)
|
LoadedData().SetValue(name, "type", newType)
|
||||||
|
|
||||||
editOptions(ri, name, true)
|
editOptions(ri, name, true)
|
||||||
RemoteConfig(ctx, name)
|
err = RemoteConfig(ctx, name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
if OkRemote(name) {
|
if OkRemote(name) {
|
||||||
SaveConfig()
|
SaveConfig()
|
||||||
return
|
return nil
|
||||||
}
|
}
|
||||||
EditRemote(ctx, ri, name)
|
return EditRemote(ctx, ri, name)
|
||||||
}
|
}
|
||||||
|
|
||||||
// EditRemote gets the user to edit a remote
|
// EditRemote gets the user to edit a remote
|
||||||
func EditRemote(ctx context.Context, ri *fs.RegInfo, name string) {
|
func EditRemote(ctx context.Context, ri *fs.RegInfo, name string) error {
|
||||||
ShowRemote(name)
|
ShowRemote(name)
|
||||||
fmt.Printf("Edit remote\n")
|
fmt.Printf("Edit remote\n")
|
||||||
for {
|
for {
|
||||||
|
@ -495,7 +499,7 @@ func EditRemote(ctx context.Context, ri *fs.RegInfo, name string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
SaveConfig()
|
SaveConfig()
|
||||||
RemoteConfig(ctx, name)
|
return RemoteConfig(ctx, name)
|
||||||
}
|
}
|
||||||
|
|
||||||
// DeleteRemote gets the user to delete a remote
|
// DeleteRemote gets the user to delete a remote
|
||||||
|
@ -560,7 +564,7 @@ func ShowConfig() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// EditConfig edits the config file interactively
|
// EditConfig edits the config file interactively
|
||||||
func EditConfig(ctx context.Context) {
|
func EditConfig(ctx context.Context) (err error) {
|
||||||
for {
|
for {
|
||||||
haveRemotes := len(LoadedData().GetSectionList()) != 0
|
haveRemotes := len(LoadedData().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"}
|
||||||
|
@ -577,9 +581,15 @@ func EditConfig(ctx context.Context) {
|
||||||
case 'e':
|
case 'e':
|
||||||
name := ChooseRemote()
|
name := ChooseRemote()
|
||||||
fs := mustFindByName(name)
|
fs := mustFindByName(name)
|
||||||
EditRemote(ctx, fs, name)
|
err = EditRemote(ctx, fs, name)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
case 'n':
|
case 'n':
|
||||||
NewRemote(ctx, NewRemoteName())
|
err = NewRemote(ctx, NewRemoteName())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
case 'd':
|
case 'd':
|
||||||
name := ChooseRemote()
|
name := ChooseRemote()
|
||||||
DeleteRemote(name)
|
DeleteRemote(name)
|
||||||
|
@ -590,8 +600,7 @@ func EditConfig(ctx context.Context) {
|
||||||
case 's':
|
case 's':
|
||||||
SetPassword()
|
SetPassword()
|
||||||
case 'q':
|
case 'q':
|
||||||
return
|
return nil
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -103,7 +103,7 @@ func TestCRUD(t *testing.T) {
|
||||||
"secret", // repeat
|
"secret", // repeat
|
||||||
"y", // looks good, save
|
"y", // looks good, save
|
||||||
})
|
})
|
||||||
config.NewRemote(ctx, "test")
|
require.NoError(t, config.NewRemote(ctx, "test"))
|
||||||
|
|
||||||
assert.Equal(t, []string{"test"}, config.Data().GetSectionList())
|
assert.Equal(t, []string{"test"}, config.Data().GetSectionList())
|
||||||
assert.Equal(t, "config_test_remote", config.FileGet("test", "type"))
|
assert.Equal(t, "config_test_remote", config.FileGet("test", "type"))
|
||||||
|
@ -146,7 +146,7 @@ func TestChooseOption(t *testing.T) {
|
||||||
assert.Equal(t, 1024, bits)
|
assert.Equal(t, 1024, bits)
|
||||||
return "not very random password", nil
|
return "not very random password", nil
|
||||||
}
|
}
|
||||||
config.NewRemote(ctx, "test")
|
require.NoError(t, config.NewRemote(ctx, "test"))
|
||||||
|
|
||||||
assert.Equal(t, "false", config.FileGet("test", "bool"))
|
assert.Equal(t, "false", config.FileGet("test", "bool"))
|
||||||
assert.Equal(t, "not very random password", obscure.MustReveal(config.FileGet("test", "pass")))
|
assert.Equal(t, "not very random password", obscure.MustReveal(config.FileGet("test", "pass")))
|
||||||
|
@ -158,7 +158,7 @@ func TestChooseOption(t *testing.T) {
|
||||||
"n", // not required
|
"n", // not required
|
||||||
"y", // looks good, save
|
"y", // looks good, save
|
||||||
})
|
})
|
||||||
config.NewRemote(ctx, "test")
|
require.NoError(t, config.NewRemote(ctx, "test"))
|
||||||
|
|
||||||
assert.Equal(t, "true", config.FileGet("test", "bool"))
|
assert.Equal(t, "true", config.FileGet("test", "bool"))
|
||||||
assert.Equal(t, "", config.FileGet("test", "pass"))
|
assert.Equal(t, "", config.FileGet("test", "pass"))
|
||||||
|
@ -175,7 +175,7 @@ func TestNewRemoteName(t *testing.T) {
|
||||||
"n", // not required
|
"n", // not required
|
||||||
"y", // looks good, save
|
"y", // looks good, save
|
||||||
})
|
})
|
||||||
config.NewRemote(ctx, "test")
|
require.NoError(t, config.NewRemote(ctx, "test"))
|
||||||
|
|
||||||
config.ReadLine = makeReadLine([]string{
|
config.ReadLine = makeReadLine([]string{
|
||||||
"test", // already exists
|
"test", // already exists
|
||||||
|
|
2
fs/fs.go
2
fs/fs.go
|
@ -89,7 +89,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(ctx context.Context, name string, config configmap.Mapper) `json:"-"`
|
Config func(ctx context.Context, name string, config configmap.Mapper) error `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
|
||||||
|
|
Loading…
Reference in a new issue