forked from TrueCloudLab/rclone
Compare commits
12 commits
tcl/master
...
fix-dlna-s
Author | SHA1 | Date | |
---|---|---|---|
|
b4ef890aa0 | ||
|
874d66658e | ||
|
3af757e26d | ||
|
fef1b61585 | ||
|
3fca7a60a5 | ||
|
6b3f41fa0c | ||
|
3d0ee47aa2 | ||
|
da70088b11 | ||
|
1bc9b94cf2 | ||
|
15a026d3be | ||
|
ad122c6f6f | ||
|
b155231cdd |
18 changed files with 203 additions and 40 deletions
|
@ -32,15 +32,27 @@ jobs:
|
||||||
- name: Get actual major version
|
- name: Get actual major version
|
||||||
id: actual_major_version
|
id: actual_major_version
|
||||||
run: echo ::set-output name=ACTUAL_MAJOR_VERSION::$(echo $GITHUB_REF | cut -d / -f 3 | sed 's/v//g' | cut -d "." -f 1)
|
run: echo ::set-output name=ACTUAL_MAJOR_VERSION::$(echo $GITHUB_REF | cut -d / -f 3 | sed 's/v//g' | cut -d "." -f 1)
|
||||||
- name: Build and publish image
|
- name: Set up QEMU
|
||||||
uses: ilteoood/docker_buildx@1.1.0
|
uses: docker/setup-qemu-action@v3
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
tag: latest,${{ steps.actual_patch_version.outputs.ACTUAL_PATCH_VERSION }},${{ steps.actual_minor_version.outputs.ACTUAL_MINOR_VERSION }},${{ steps.actual_major_version.outputs.ACTUAL_MAJOR_VERSION }}
|
username: ${{ secrets.DOCKER_HUB_USER }}
|
||||||
imageName: rclone/rclone
|
password: ${{ secrets.DOCKER_HUB_PASSWORD }}
|
||||||
platform: linux/amd64,linux/386,linux/arm64,linux/arm/v7,linux/arm/v6
|
- name: Build and publish image
|
||||||
publish: true
|
uses: docker/build-push-action@v6
|
||||||
dockerHubUser: ${{ secrets.DOCKER_HUB_USER }}
|
with:
|
||||||
dockerHubPassword: ${{ secrets.DOCKER_HUB_PASSWORD }}
|
file: Dockerfile
|
||||||
|
context: .
|
||||||
|
platforms: linux/amd64,linux/386,linux/arm64,linux/arm/v7,linux/arm/v6
|
||||||
|
push: true
|
||||||
|
tags: |
|
||||||
|
rclone/rclone:latest
|
||||||
|
rclone/rclone:${{ steps.actual_patch_version.outputs.ACTUAL_PATCH_VERSION }}
|
||||||
|
rclone/rclone:${{ steps.actual_minor_version.outputs.ACTUAL_MINOR_VERSION }}
|
||||||
|
rclone/rclone:${{ steps.actual_major_version.outputs.ACTUAL_MAJOR_VERSION }}
|
||||||
|
|
||||||
build_docker_volume_plugin:
|
build_docker_volume_plugin:
|
||||||
if: github.repository == 'rclone/rclone'
|
if: github.repository == 'rclone/rclone'
|
||||||
|
|
|
@ -168,6 +168,8 @@ docker buildx build -t rclone/rclone:testing --progress=plain --platform linux/a
|
||||||
|
|
||||||
To make a full build then set the tags correctly and add `--push`
|
To make a full build then set the tags correctly and add `--push`
|
||||||
|
|
||||||
|
Note that you can't only build one architecture - you need to build them all.
|
||||||
|
|
||||||
```
|
```
|
||||||
docker buildx build --platform linux/amd64,linux/386,linux/arm64,linux/arm/v7 -t rclone/rclone:1.54.1 -t rclone/rclone:1.54 -t rclone/rclone:1 -t rclone/rclone:latest --push .
|
docker buildx build --platform linux/amd64,linux/386,linux/arm64,linux/arm/v7,linux/arm/v6 -t rclone/rclone:1.54.1 -t rclone/rclone:1.54 -t rclone/rclone:1 -t rclone/rclone:latest --push .
|
||||||
```
|
```
|
||||||
|
|
2
VERSION
2
VERSION
|
@ -1 +1 @@
|
||||||
v1.68.0
|
v1.69.0
|
||||||
|
|
|
@ -942,7 +942,8 @@ func errorHandler(resp *http.Response) error {
|
||||||
// Decode error response
|
// Decode error response
|
||||||
errResponse := new(api.Error)
|
errResponse := new(api.Error)
|
||||||
err := rest.DecodeJSON(resp, &errResponse)
|
err := rest.DecodeJSON(resp, &errResponse)
|
||||||
if err != nil {
|
// Redirects have no body so don't report an error
|
||||||
|
if err != nil && resp.Header.Get("Location") == "" {
|
||||||
fs.Debugf(nil, "Couldn't decode error response: %v", err)
|
fs.Debugf(nil, "Couldn't decode error response: %v", err)
|
||||||
}
|
}
|
||||||
if errResponse.ErrorInfo.Code == "" {
|
if errResponse.ErrorInfo.Code == "" {
|
||||||
|
|
|
@ -11,6 +11,7 @@ import (
|
||||||
"path"
|
"path"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"regexp"
|
"regexp"
|
||||||
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/anacrolix/dms/dlna"
|
"github.com/anacrolix/dms/dlna"
|
||||||
|
@ -158,6 +159,18 @@ func (cds *contentDirectoryService) readContainer(o object, host string) (ret []
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Sort the directory entries by directories first then alphabetically by name
|
||||||
|
sort.Slice(dirEntries, func(i, j int) bool {
|
||||||
|
iNode, jNode := dirEntries[i], dirEntries[j]
|
||||||
|
iIsDir, jIsDir := iNode.IsDir(), jNode.IsDir()
|
||||||
|
if iIsDir && !jIsDir {
|
||||||
|
return true
|
||||||
|
} else if !iIsDir && jIsDir {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return strings.ToLower(iNode.Name()) < strings.ToLower(jNode.Name())
|
||||||
|
})
|
||||||
|
|
||||||
dirEntries, mediaResources := mediaWithResources(dirEntries)
|
dirEntries, mediaResources := mediaWithResources(dirEntries)
|
||||||
for _, de := range dirEntries {
|
for _, de := range dirEntries {
|
||||||
child := object{
|
child := object{
|
||||||
|
|
|
@ -251,6 +251,15 @@ func getVFSOption(vfsOpt *vfscommon.Options, opt rc.Params, key string) (ok bool
|
||||||
err = getFVarP(&vfsOpt.ReadAhead, opt, key)
|
err = getFVarP(&vfsOpt.ReadAhead, opt, key)
|
||||||
case "vfs-used-is-size":
|
case "vfs-used-is-size":
|
||||||
vfsOpt.UsedIsSize, err = opt.GetBool(key)
|
vfsOpt.UsedIsSize, err = opt.GetBool(key)
|
||||||
|
case "vfs-read-chunk-streams":
|
||||||
|
intVal, err = opt.GetInt64(key)
|
||||||
|
if err == nil {
|
||||||
|
if intVal >= 0 && intVal <= math.MaxInt {
|
||||||
|
vfsOpt.ChunkStreams = int(intVal)
|
||||||
|
} else {
|
||||||
|
err = fmt.Errorf("key %q (%v) overflows int", key, intVal)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// unprefixed vfs options
|
// unprefixed vfs options
|
||||||
case "no-modtime":
|
case "no-modtime":
|
||||||
|
|
|
@ -6,7 +6,9 @@ package cmdtest
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"os"
|
"os"
|
||||||
|
"regexp"
|
||||||
"runtime"
|
"runtime"
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
|
@ -344,4 +346,42 @@ func TestEnvironmentVariables(t *testing.T) {
|
||||||
env = ""
|
env = ""
|
||||||
out, err = rcloneEnv(env, "version", "-vv", "--use-json-log")
|
out, err = rcloneEnv(env, "version", "-vv", "--use-json-log")
|
||||||
jsonLogOK()
|
jsonLogOK()
|
||||||
|
|
||||||
|
// Find all the File filter lines in out and return them
|
||||||
|
parseFileFilters := func(out string) (extensions []string) {
|
||||||
|
// Match: - (^|/)[^/]*\.jpg$
|
||||||
|
find := regexp.MustCompile(`^- \(\^\|\/\)\[\^\/\]\*\\\.(.*?)\$$`)
|
||||||
|
for _, line := range strings.Split(out, "\n") {
|
||||||
|
if m := find.FindStringSubmatch(line); m != nil {
|
||||||
|
extensions = append(extensions, m[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return extensions
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure that multiple valued (stringArray) environment variables are handled properly
|
||||||
|
env = ``
|
||||||
|
out, err = rcloneEnv(env, "version", "-vv", "--dump", "filters", "--exclude", "*.gif", "--exclude", "*.tif")
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, []string{"gif", "tif"}, parseFileFilters(out))
|
||||||
|
|
||||||
|
env = `RCLONE_EXCLUDE=*.jpg`
|
||||||
|
out, err = rcloneEnv(env, "version", "-vv", "--dump", "filters", "--exclude", "*.gif")
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, []string{"jpg", "gif"}, parseFileFilters(out))
|
||||||
|
|
||||||
|
env = `RCLONE_EXCLUDE=*.jpg,*.png`
|
||||||
|
out, err = rcloneEnv(env, "version", "-vv", "--dump", "filters", "--exclude", "*.gif", "--exclude", "*.tif")
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, []string{"jpg", "png", "gif", "tif"}, parseFileFilters(out))
|
||||||
|
|
||||||
|
env = `RCLONE_EXCLUDE="*.jpg","*.png"`
|
||||||
|
out, err = rcloneEnv(env, "version", "-vv", "--dump", "filters")
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, []string{"jpg", "png"}, parseFileFilters(out))
|
||||||
|
|
||||||
|
env = `RCLONE_EXCLUDE="*.,,,","*.png"`
|
||||||
|
out, err = rcloneEnv(env, "version", "-vv", "--dump", "filters")
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, []string{",,,", "png"}, parseFileFilters(out))
|
||||||
}
|
}
|
||||||
|
|
|
@ -889,3 +889,5 @@ put them back in again.` >}}
|
||||||
* Mathieu Moreau <mrx23dot@users.noreply.github.com>
|
* Mathieu Moreau <mrx23dot@users.noreply.github.com>
|
||||||
* fsantagostinobietti <6057026+fsantagostinobietti@users.noreply.github.com>
|
* fsantagostinobietti <6057026+fsantagostinobietti@users.noreply.github.com>
|
||||||
* Oleg Kunitsyn <114359669+hiddenmarten@users.noreply.github.com>
|
* Oleg Kunitsyn <114359669+hiddenmarten@users.noreply.github.com>
|
||||||
|
* Divyam <47589864+divyam234@users.noreply.github.com>
|
||||||
|
* ttionya <ttionya@users.noreply.github.com>
|
||||||
|
|
|
@ -2911,6 +2911,22 @@ so they take exactly the same form.
|
||||||
|
|
||||||
The options set by environment variables can be seen with the `-vv` flag, e.g. `rclone version -vv`.
|
The options set by environment variables can be seen with the `-vv` flag, e.g. `rclone version -vv`.
|
||||||
|
|
||||||
|
Options that can appear multiple times (type `stringArray`) are
|
||||||
|
treated slighly differently as environment variables can only be
|
||||||
|
defined once. In order to allow a simple mechanism for adding one or
|
||||||
|
many items, the input is treated as a [CSV encoded](https://godoc.org/encoding/csv)
|
||||||
|
string. For example
|
||||||
|
|
||||||
|
| Environment Variable | Equivalent options |
|
||||||
|
|----------------------|--------------------|
|
||||||
|
| `RCLONE_EXCLUDE="*.jpg"` | `--exclude "*.jpg"` |
|
||||||
|
| `RCLONE_EXCLUDE="*.jpg,*.png"` | `--exclude "*.jpg"` `--exclude "*.png"` |
|
||||||
|
| `RCLONE_EXCLUDE='"*.jpg","*.png"'` | `--exclude "*.jpg"` `--exclude "*.png"` |
|
||||||
|
| `RCLONE_EXCLUDE='"/directory with comma , in it /**"'` | `--exclude "/directory with comma , in it /**" |
|
||||||
|
|
||||||
|
If `stringArray` options are defined as environment variables **and**
|
||||||
|
options on the command line then all the values will be used.
|
||||||
|
|
||||||
### Config file ###
|
### Config file ###
|
||||||
|
|
||||||
You can set defaults for values in the config file on an individual
|
You can set defaults for values in the config file on an individual
|
||||||
|
|
|
@ -401,6 +401,38 @@ there for more details.
|
||||||
|
|
||||||
Setting this flag increases the chance for undetected upload failures.
|
Setting this flag increases the chance for undetected upload failures.
|
||||||
|
|
||||||
|
### Increasing performance
|
||||||
|
|
||||||
|
#### Using server-side copy
|
||||||
|
|
||||||
|
If you are copying objects between S3 buckets in the same region, you should
|
||||||
|
use server-side copy.
|
||||||
|
This is much faster than downloading and re-uploading the objects, as no data is transferred.
|
||||||
|
|
||||||
|
For rclone to use server-side copy, you must use the same remote for the source and destination.
|
||||||
|
|
||||||
|
rclone copy s3:source-bucket s3:destination-bucket
|
||||||
|
|
||||||
|
When using server-side copy, the performance is limited by the rate at which rclone issues
|
||||||
|
API requests to S3.
|
||||||
|
See below for how to increase the number of API requests rclone makes.
|
||||||
|
|
||||||
|
#### Increasing the rate of API requests
|
||||||
|
|
||||||
|
You can increase the rate of API requests to S3 by increasing the parallelism using `--transfers` and `--checkers`
|
||||||
|
options.
|
||||||
|
|
||||||
|
Rclone uses a very conservative defaults for these settings, as not all providers support high rates of requests.
|
||||||
|
Depending on your provider, you can increase significantly the number of transfers and checkers.
|
||||||
|
|
||||||
|
For example, with AWS S3, if you can increase the number of checkers to values like 200.
|
||||||
|
If you are doing a server-side copy, you can also increase the number of transfers to 200.
|
||||||
|
|
||||||
|
rclone sync --transfers 200 --checkers 200 --checksum s3:source-bucket s3:destination-bucket
|
||||||
|
|
||||||
|
You will need to experiment with these values to find the optimal settings for your setup.
|
||||||
|
|
||||||
|
|
||||||
### Versions
|
### Versions
|
||||||
|
|
||||||
When bucket versioning is enabled (this can be done with rclone with
|
When bucket versioning is enabled (this can be done with rclone with
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
v1.68.0
|
v1.69.0
|
|
@ -2,7 +2,7 @@
|
||||||
package configstruct
|
package configstruct
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/csv"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
@ -31,7 +31,7 @@ func camelToSnake(in string) string {
|
||||||
//
|
//
|
||||||
// Builtin types are expected to be encoding as their natural
|
// Builtin types are expected to be encoding as their natural
|
||||||
// stringificatons as produced by fmt.Sprint except for []string which
|
// stringificatons as produced by fmt.Sprint except for []string which
|
||||||
// is expected to be encoded as JSON with empty array encoded as "".
|
// is expected to be encoded a a CSV with empty array encoded as "".
|
||||||
//
|
//
|
||||||
// Any other types are expected to be encoded by their String()
|
// Any other types are expected to be encoded by their String()
|
||||||
// methods and decoded by their `Set(s string) error` methods.
|
// methods and decoded by their `Set(s string) error` methods.
|
||||||
|
@ -58,14 +58,18 @@ func StringToInterface(def interface{}, in string) (newValue interface{}, err er
|
||||||
case time.Duration:
|
case time.Duration:
|
||||||
newValue, err = time.ParseDuration(in)
|
newValue, err = time.ParseDuration(in)
|
||||||
case []string:
|
case []string:
|
||||||
// JSON decode arrays of strings
|
// CSV decode arrays of strings - ideally we would use
|
||||||
if in != "" {
|
// fs.CommaSepList here but we can't as it would cause
|
||||||
var out []string
|
// a circular import.
|
||||||
err = json.Unmarshal([]byte(in), &out)
|
if len(in) == 0 {
|
||||||
newValue = out
|
|
||||||
} else {
|
|
||||||
// Empty string we will treat as empty array
|
|
||||||
newValue = []string{}
|
newValue = []string{}
|
||||||
|
} else {
|
||||||
|
r := csv.NewReader(strings.NewReader(in))
|
||||||
|
newValue, err = r.Read()
|
||||||
|
switch _err := err.(type) {
|
||||||
|
case *csv.ParseError:
|
||||||
|
err = _err.Err // remove line numbers from the error message
|
||||||
|
}
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
// Try using a Set method
|
// Try using a Set method
|
||||||
|
|
|
@ -204,9 +204,11 @@ func TestStringToInterface(t *testing.T) {
|
||||||
{"1m1s", fs.Duration(0), fs.Duration(61 * time.Second), ""},
|
{"1m1s", fs.Duration(0), fs.Duration(61 * time.Second), ""},
|
||||||
{"1potato", fs.Duration(0), nil, `parsing "1potato" as fs.Duration failed: parsing time "1potato" as "2006-01-02": cannot parse "1potato" as "2006"`},
|
{"1potato", fs.Duration(0), nil, `parsing "1potato" as fs.Duration failed: parsing time "1potato" as "2006-01-02": cannot parse "1potato" as "2006"`},
|
||||||
{``, []string{}, []string{}, ""},
|
{``, []string{}, []string{}, ""},
|
||||||
{`[]`, []string(nil), []string{}, ""},
|
{`""`, []string(nil), []string{""}, ""},
|
||||||
{`["hello"]`, []string{}, []string{"hello"}, ""},
|
{`hello`, []string{}, []string{"hello"}, ""},
|
||||||
{`["hello","world!"]`, []string(nil), []string{"hello", "world!"}, ""},
|
{`"hello"`, []string{}, []string{"hello"}, ""},
|
||||||
|
{`hello,world!`, []string(nil), []string{"hello", "world!"}, ""},
|
||||||
|
{`"hello","world!"`, []string(nil), []string{"hello", "world!"}, ""},
|
||||||
{"1s", time.Duration(0), time.Second, ""},
|
{"1s", time.Duration(0), time.Second, ""},
|
||||||
{"1m1s", time.Duration(0), 61 * time.Second, ""},
|
{"1m1s", time.Duration(0), 61 * time.Second, ""},
|
||||||
{"1potato", time.Duration(0), nil, `parsing "1potato" as time.Duration failed: time: unknown unit "potato" in duration "1potato"`},
|
{"1potato", time.Duration(0), nil, `parsing "1potato" as time.Duration failed: time: unknown unit "potato" in duration "1potato"`},
|
||||||
|
|
|
@ -143,12 +143,32 @@ func installFlag(flags *pflag.FlagSet, name string, groupsString string) {
|
||||||
// Read default from environment if possible
|
// Read default from environment if possible
|
||||||
envKey := fs.OptionToEnv(name)
|
envKey := fs.OptionToEnv(name)
|
||||||
if envValue, envFound := os.LookupEnv(envKey); envFound {
|
if envValue, envFound := os.LookupEnv(envKey); envFound {
|
||||||
err := flags.Set(name, envValue)
|
isStringArray := false
|
||||||
if err != nil {
|
opt, isOption := flag.Value.(*fs.Option)
|
||||||
fs.Fatalf(nil, "Invalid value when setting --%s from environment variable %s=%q: %v", name, envKey, envValue, err)
|
if isOption {
|
||||||
|
_, isStringArray = opt.Default.([]string)
|
||||||
|
}
|
||||||
|
if isStringArray {
|
||||||
|
// Treat stringArray differently, treating the environment variable as a CSV array
|
||||||
|
var list fs.CommaSepList
|
||||||
|
err := list.Set(envValue)
|
||||||
|
if err != nil {
|
||||||
|
fs.Fatalf(nil, "Invalid value when setting stringArray --%s from environment variable %s=%q: %v", name, envKey, envValue, err)
|
||||||
|
}
|
||||||
|
// Set both the Value (so items on the command line get added) and DefValue so the help is correct
|
||||||
|
opt.Value = ([]string)(list)
|
||||||
|
flag.DefValue = list.String()
|
||||||
|
for _, v := range list {
|
||||||
|
fs.Debugf(nil, "Setting --%s %q from environment variable %s=%q", name, v, envKey, envValue)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
err := flags.Set(name, envValue)
|
||||||
|
if err != nil {
|
||||||
|
fs.Fatalf(nil, "Invalid value when setting --%s from environment variable %s=%q: %v", name, envKey, envValue, err)
|
||||||
|
}
|
||||||
|
fs.Debugf(nil, "Setting --%s %q from environment variable %s=%q", name, flag.Value, envKey, envValue)
|
||||||
|
flag.DefValue = envValue
|
||||||
}
|
}
|
||||||
fs.Debugf(nil, "Setting --%s %q from environment variable %s=%q", name, flag.Value, envKey, envValue)
|
|
||||||
flag.DefValue = envValue
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add flag to Group if it is a global flag
|
// Add flag to Group if it is a global flag
|
||||||
|
|
|
@ -85,7 +85,7 @@ var OptionsInfo = fs.Options{{
|
||||||
Groups: "RC",
|
Groups: "RC",
|
||||||
}, {
|
}, {
|
||||||
Name: "metrics_addr",
|
Name: "metrics_addr",
|
||||||
Default: []string{""},
|
Default: []string{},
|
||||||
Help: "IPaddress:Port or :Port to bind metrics server to",
|
Help: "IPaddress:Port or :Port to bind metrics server to",
|
||||||
Groups: "Metrics",
|
Groups: "Metrics",
|
||||||
}}.
|
}}.
|
||||||
|
|
|
@ -37,7 +37,7 @@ func init() {
|
||||||
// If the server wasn't configured the *Server returned may be nil
|
// If the server wasn't configured the *Server returned may be nil
|
||||||
func MetricsStart(ctx context.Context, opt *rc.Options) (*MetricsServer, error) {
|
func MetricsStart(ctx context.Context, opt *rc.Options) (*MetricsServer, error) {
|
||||||
jobs.SetOpt(opt) // set the defaults for jobs
|
jobs.SetOpt(opt) // set the defaults for jobs
|
||||||
if opt.MetricsHTTP.ListenAddr[0] != "" {
|
if len(opt.MetricsHTTP.ListenAddr) > 0 {
|
||||||
// Serve on the DefaultServeMux so can have global registrations appear
|
// Serve on the DefaultServeMux so can have global registrations appear
|
||||||
s, err := newMetricsServer(ctx, opt)
|
s, err := newMetricsServer(ctx, opt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -264,14 +264,9 @@ func (o *Option) String() string {
|
||||||
if len(stringArray) == 0 {
|
if len(stringArray) == 0 {
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
// Encode string arrays as JSON
|
// Encode string arrays as CSV
|
||||||
// The default Go encoding can't be decoded uniquely
|
// The default Go encoding can't be decoded uniquely
|
||||||
buf, err := json.Marshal(stringArray)
|
return CommaSepList(stringArray).String()
|
||||||
if err != nil {
|
|
||||||
Errorf(nil, "Can't encode default value for %q key - ignoring: %v", o.Name, err)
|
|
||||||
return "[]"
|
|
||||||
}
|
|
||||||
return string(buf)
|
|
||||||
}
|
}
|
||||||
return fmt.Sprint(v)
|
return fmt.Sprint(v)
|
||||||
}
|
}
|
||||||
|
@ -531,7 +526,22 @@ func (oi *OptionsInfo) load() error {
|
||||||
// their values read from the options, environment variables and
|
// their values read from the options, environment variables and
|
||||||
// command line parameters.
|
// command line parameters.
|
||||||
func GlobalOptionsInit() error {
|
func GlobalOptionsInit() error {
|
||||||
for _, opt := range OptionsRegistry {
|
var keys []string
|
||||||
|
for key := range OptionsRegistry {
|
||||||
|
keys = append(keys, key)
|
||||||
|
}
|
||||||
|
sort.Slice(keys, func(i, j int) bool {
|
||||||
|
// Sort alphabetically, but with "main" first
|
||||||
|
if keys[i] == "main" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
if keys[j] == "main" {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return keys[i] < keys[j]
|
||||||
|
})
|
||||||
|
for _, key := range keys {
|
||||||
|
opt := OptionsRegistry[key]
|
||||||
err := opt.load()
|
err := opt.load()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
package fs
|
package fs
|
||||||
|
|
||||||
// VersionTag of rclone
|
// VersionTag of rclone
|
||||||
var VersionTag = "v1.68.0"
|
var VersionTag = "v1.69.0"
|
||||||
|
|
Loading…
Reference in a new issue