forked from TrueCloudLab/rclone
sia: fix and enable integration tests #4514
- setup correct path encoding (fixes backend test FsEncoding) - ignore range option if file is empty (fixes VFS test TestFileReadAtZeroLength) - cleanup stray files left after failed upload (fixes test FsPutError) - rebase code on master, adapt backend for rclone context passing - translate Siad errors to rclone native FS errors in sia errorHandler - TestSia: return proper backend options from the script - TestSia: use uptodate AntFarm image, nebulouslabs/siaantfarm is stale
This commit is contained in:
parent
0d1e017e09
commit
c5bc857f9b
3 changed files with 96 additions and 38 deletions
|
@ -59,6 +59,11 @@ func init() {
|
||||||
Help: config.ConfigEncodingHelp,
|
Help: config.ConfigEncodingHelp,
|
||||||
Advanced: true,
|
Advanced: true,
|
||||||
Default: encoder.EncodeInvalidUtf8 |
|
Default: encoder.EncodeInvalidUtf8 |
|
||||||
|
encoder.EncodeCtl |
|
||||||
|
encoder.EncodeDel |
|
||||||
|
encoder.EncodeHashPercent |
|
||||||
|
encoder.EncodeQuestion |
|
||||||
|
encoder.EncodeDot |
|
||||||
encoder.EncodeSlash,
|
encoder.EncodeSlash,
|
||||||
},
|
},
|
||||||
}})
|
}})
|
||||||
|
@ -135,11 +140,22 @@ func (o *Object) SetModTime(ctx context.Context, t time.Time) error {
|
||||||
|
|
||||||
// Open an object for read
|
// Open an object for read
|
||||||
func (o *Object) Open(ctx context.Context, options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
func (o *Object) Open(ctx context.Context, options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
|
var optionsFixed []fs.OpenOption
|
||||||
|
for _, opt := range options {
|
||||||
|
if optRange, ok := opt.(*fs.RangeOption); ok {
|
||||||
|
// Ignore range option if file is empty
|
||||||
|
if o.Size() == 0 && optRange.Start == 0 && optRange.End > 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
optionsFixed = append(optionsFixed, opt)
|
||||||
|
}
|
||||||
|
|
||||||
var resp *http.Response
|
var resp *http.Response
|
||||||
opts := rest.Opts{
|
opts := rest.Opts{
|
||||||
Method: "GET",
|
Method: "GET",
|
||||||
Path: path.Join("/renter/stream/", o.fs.root, o.fs.opt.Enc.FromStandardPath(o.remote)),
|
Path: path.Join("/renter/stream/", o.fs.root, o.fs.opt.Enc.FromStandardPath(o.remote)),
|
||||||
Options: options,
|
Options: optionsFixed,
|
||||||
}
|
}
|
||||||
err = o.fs.pacer.Call(func() (bool, error) {
|
err = o.fs.pacer.Call(func() (bool, error) {
|
||||||
resp, err = o.fs.srv.Call(ctx, &opts)
|
resp, err = o.fs.srv.Call(ctx, &opts)
|
||||||
|
@ -309,7 +325,27 @@ func (f *Fs) Put(ctx context.Context, in io.Reader, src fs.ObjectInfo, options .
|
||||||
size: src.Size(),
|
size: src.Size(),
|
||||||
}
|
}
|
||||||
|
|
||||||
return o, o.Update(ctx, in, src, options...)
|
err := o.Update(ctx, in, src, options...)
|
||||||
|
if err == nil {
|
||||||
|
return o, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cleanup stray files left after failed upload
|
||||||
|
for i := 0; i < 5; i++ {
|
||||||
|
cleanObj, cleanErr := f.NewObject(ctx, src.Remote())
|
||||||
|
if cleanErr == nil {
|
||||||
|
cleanErr = cleanObj.Remove(ctx)
|
||||||
|
}
|
||||||
|
if cleanErr == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
if cleanErr != fs.ErrorObjectNotFound {
|
||||||
|
fs.Logf(f, "%q: cleanup failed upload: %v", src.Remote(), cleanErr)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
time.Sleep(100 * time.Millisecond)
|
||||||
|
}
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// PutStream the object into the remote siad via uploadstream
|
// PutStream the object into the remote siad via uploadstream
|
||||||
|
@ -375,7 +411,7 @@ func (f *Fs) Rmdir(ctx context.Context, dir string) (err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewFs constructs an Fs from the path
|
// NewFs constructs an Fs from the path
|
||||||
func NewFs(name, root string, m configmap.Mapper) (fs.Fs, error) {
|
func NewFs(ctx context.Context, name, root string, m configmap.Mapper) (fs.Fs, 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)
|
||||||
|
@ -396,22 +432,25 @@ func NewFs(name, root string, m configmap.Mapper) (fs.Fs, error) {
|
||||||
rootIsDir := strings.HasSuffix(root, "/")
|
rootIsDir := strings.HasSuffix(root, "/")
|
||||||
root = strings.Trim(root, "/")
|
root = strings.Trim(root, "/")
|
||||||
|
|
||||||
config := fs.Config
|
|
||||||
if opt.UserAgent != "" {
|
|
||||||
config.UserAgent = opt.UserAgent
|
|
||||||
}
|
|
||||||
|
|
||||||
f := &Fs{
|
f := &Fs{
|
||||||
name: name,
|
name: name,
|
||||||
opt: *opt,
|
opt: *opt,
|
||||||
srv: rest.NewClient(fshttp.NewClient(config)).SetErrorHandler(errorHandler).SetRoot(u.String()),
|
|
||||||
root: root,
|
root: root,
|
||||||
pacer: fs.NewPacer(pacer.NewDefault(pacer.MinSleep(minSleep), pacer.MaxSleep(maxSleep), pacer.DecayConstant(decayConstant))),
|
|
||||||
}
|
}
|
||||||
|
f.pacer = fs.NewPacer(ctx, pacer.NewDefault(pacer.MinSleep(minSleep), pacer.MaxSleep(maxSleep), pacer.DecayConstant(decayConstant)))
|
||||||
|
|
||||||
f.features = (&fs.Features{
|
f.features = (&fs.Features{
|
||||||
CanHaveEmptyDirectories: true,
|
CanHaveEmptyDirectories: true,
|
||||||
}).Fill(f)
|
}).Fill(ctx, f)
|
||||||
|
|
||||||
|
// Adjust client config and pass it attached to context
|
||||||
|
cliCtx, cliCfg := fs.AddConfig(ctx)
|
||||||
|
if opt.UserAgent != "" {
|
||||||
|
cliCfg.UserAgent = opt.UserAgent
|
||||||
|
}
|
||||||
|
f.srv = rest.NewClient(fshttp.NewClient(cliCtx))
|
||||||
|
f.srv.SetRoot(u.String())
|
||||||
|
f.srv.SetErrorHandler(errorHandler)
|
||||||
|
|
||||||
if opt.APIPassword != "" {
|
if opt.APIPassword != "" {
|
||||||
opt.APIPassword, err = obscure.Reveal(opt.APIPassword)
|
opt.APIPassword, err = obscure.Reveal(opt.APIPassword)
|
||||||
|
@ -428,7 +467,6 @@ func NewFs(name, root string, m configmap.Mapper) (fs.Fs, error) {
|
||||||
if f.root == "." {
|
if f.root == "." {
|
||||||
f.root = ""
|
f.root = ""
|
||||||
}
|
}
|
||||||
ctx := context.Background()
|
|
||||||
_, err := f.NewObject(ctx, remote)
|
_, err := f.NewObject(ctx, remote)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if errors.Cause(err) == fs.ErrorObjectNotFound || errors.Cause(err) == fs.ErrorNotAFile {
|
if errors.Cause(err) == fs.ErrorObjectNotFound || errors.Cause(err) == fs.ErrorNotAFile {
|
||||||
|
@ -445,8 +483,8 @@ func NewFs(name, root string, m configmap.Mapper) (fs.Fs, error) {
|
||||||
return f, nil
|
return f, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Decode errors into meaningful ones, sadly this is using
|
// errorHandler translates Siad errors into native rclone filesystem errors.
|
||||||
// string matching since siad doesn't expose meaningful error codes
|
// Sadly this is using string matching since Siad can't expose meaningful codes.
|
||||||
func errorHandler(resp *http.Response) error {
|
func errorHandler(resp *http.Response) error {
|
||||||
body, err := rest.ReadBody(resp)
|
body, err := rest.ReadBody(resp)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -456,17 +494,24 @@ func errorHandler(resp *http.Response) error {
|
||||||
errResponse := new(api.Error)
|
errResponse := new(api.Error)
|
||||||
err = json.Unmarshal(body, &errResponse)
|
err = json.Unmarshal(body, &errResponse)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// set the Message to be the body if we can't parse the JSON
|
// Set the Message to be the body if we can't parse the JSON
|
||||||
errResponse.Message = strings.TrimSpace(string(body))
|
errResponse.Message = strings.TrimSpace(string(body))
|
||||||
}
|
}
|
||||||
errResponse.Status = resp.Status
|
errResponse.Status = resp.Status
|
||||||
errResponse.StatusCode = resp.StatusCode
|
errResponse.StatusCode = resp.StatusCode
|
||||||
|
|
||||||
if errResponse.StatusCode == 400 && errResponse.Message == "no file known with that path" {
|
msg := strings.Trim(errResponse.Message, "[]")
|
||||||
|
code := errResponse.StatusCode
|
||||||
|
switch {
|
||||||
|
case code == 400 && msg == "no file known with that path":
|
||||||
return fs.ErrorObjectNotFound
|
return fs.ErrorObjectNotFound
|
||||||
} else if errResponse.StatusCode == 500 && errResponse.Message == "failed to create directory: a siadir already exists at that location" {
|
case code == 400 && strings.HasPrefix(msg, "unable to get the fileinfo from the filesystem") && strings.HasSuffix(msg, "path does not exist"):
|
||||||
|
return fs.ErrorObjectNotFound
|
||||||
|
case code == 500 && strings.HasPrefix(msg, "failed to create directory") && strings.HasSuffix(msg, "a siadir already exists at that location"):
|
||||||
return fs.ErrorDirExists
|
return fs.ErrorDirExists
|
||||||
} else if errResponse.StatusCode == 500 && strings.HasSuffix(errResponse.Message, ": no such file or directory") {
|
case code == 500 && strings.HasPrefix(msg, "failed to get directory contents") && strings.HasSuffix(msg, "path does not exist"):
|
||||||
|
return fs.ErrorDirNotFound
|
||||||
|
case code == 500 && strings.HasSuffix(msg, "no such file or directory"):
|
||||||
return fs.ErrorDirNotFound
|
return fs.ErrorDirNotFound
|
||||||
}
|
}
|
||||||
return errResponse
|
return errResponse
|
||||||
|
|
|
@ -312,6 +312,9 @@ backends:
|
||||||
- backend: "seafile"
|
- backend: "seafile"
|
||||||
remote: "TestSeafileEncrypted:"
|
remote: "TestSeafileEncrypted:"
|
||||||
fastlist: true
|
fastlist: true
|
||||||
|
- backend: "sia"
|
||||||
|
remote: "TestSia:"
|
||||||
|
fastlist: false
|
||||||
- backend: "tardigrade"
|
- backend: "tardigrade"
|
||||||
remote: "TestTardigrade:"
|
remote: "TestTardigrade:"
|
||||||
fastlist: true
|
fastlist: true
|
||||||
|
|
|
@ -4,27 +4,36 @@ set -e
|
||||||
|
|
||||||
NAME=Sia
|
NAME=Sia
|
||||||
|
|
||||||
. $(dirname "$0")/docker.bash
|
# shellcheck disable=SC1090
|
||||||
|
. "$(dirname "$0")"/docker.bash
|
||||||
|
|
||||||
|
# wait until Sia test network is up,
|
||||||
|
# the Sia renter forms contracts on the blockchain
|
||||||
|
# and the renter is upload ready
|
||||||
|
wait_for_sia() {
|
||||||
|
until curl -A Sia-Agent -s "$1" | grep -q '"ready":true'
|
||||||
|
do
|
||||||
|
sleep 5
|
||||||
|
done
|
||||||
|
}
|
||||||
|
export -f wait_for_sia
|
||||||
|
|
||||||
start() {
|
start() {
|
||||||
docker run --rm -d --name $NAME \
|
# use non-production sia port in test
|
||||||
--publish 127.0.0.1:9980:9980 \
|
SIA_CONN="127.0.0.1:39980"
|
||||||
nebulouslabs/siaantfarm
|
# nebulouslabs/siaantfarm is stale, use uptodate image
|
||||||
|
ANTFARM_IMAGE=ivandeex/sia-antfarm:latest
|
||||||
|
|
||||||
# pull latest remote image (do not use latest local image)
|
# pull latest antfarm image (dont use local image)
|
||||||
docker pull nebulouslabs/siaantfarm:latest
|
docker pull --quiet $ANTFARM_IMAGE
|
||||||
|
|
||||||
# start antfarm with the default config
|
# start latest antfarm with default config
|
||||||
docker run --rm -d --name "$NAME" \
|
docker run --rm --detach --name "$NAME" \
|
||||||
-p "${SIA_CONN}:${SIA_PORT}" \
|
--publish "${SIA_CONN}:9980" \
|
||||||
nebulouslabs/siaantfarm:latest
|
$ANTFARM_IMAGE
|
||||||
|
|
||||||
# wait until Sia test network is up, the Sia renter forms contracts on the
|
# wait until the test network is upload ready
|
||||||
# blockchain and the renter is upload ready
|
timeout 300 bash -c "wait_for_sia ${SIA_CONN}/renter/uploadready"
|
||||||
until wget --user-agent="Sia-Agent" -q -O - "${SIA_CONN}/renter/uploadready" | grep '"ready":true'
|
|
||||||
do
|
|
||||||
sleep 1
|
|
||||||
done
|
|
||||||
|
|
||||||
# confirm backend type in the generated rclone.conf
|
# confirm backend type in the generated rclone.conf
|
||||||
echo "type=sia"
|
echo "type=sia"
|
||||||
|
@ -42,4 +51,5 @@ stop() {
|
||||||
fi
|
fi
|
||||||
}
|
}
|
||||||
|
|
||||||
. $(dirname "$0")/run.bash
|
# shellcheck disable=SC1090
|
||||||
|
. "$(dirname "$0")"/run.bash
|
||||||
|
|
Loading…
Reference in a new issue