mirror of
https://github.com/rclone/rclone.git
synced 2026-01-06 18:43:50 +00:00
Switch to using the dep tool and update all the dependencies
This commit is contained in:
13
vendor/github.com/googleapis/gax-go/call_option.go
generated
vendored
13
vendor/github.com/googleapis/gax-go/call_option.go
generated
vendored
@@ -129,8 +129,21 @@ func (bo *Backoff) Pause() time.Duration {
|
||||
return d
|
||||
}
|
||||
|
||||
type grpcOpt []grpc.CallOption
|
||||
|
||||
func (o grpcOpt) Resolve(s *CallSettings) {
|
||||
s.GRPC = o
|
||||
}
|
||||
|
||||
func WithGRPCOptions(opt ...grpc.CallOption) CallOption {
|
||||
return grpcOpt(append([]grpc.CallOption(nil), opt...))
|
||||
}
|
||||
|
||||
type CallSettings struct {
|
||||
// Retry returns a Retryer to be used to control retry logic of a method call.
|
||||
// If Retry is nil or the returned Retryer is nil, the call will not be retried.
|
||||
Retry func() Retryer
|
||||
|
||||
// CallOptions to be forwarded to GRPC.
|
||||
GRPC []grpc.CallOption
|
||||
}
|
||||
|
||||
88
vendor/github.com/googleapis/gax-go/call_option_test.go
generated
vendored
Normal file
88
vendor/github.com/googleapis/gax-go/call_option_test.go
generated
vendored
Normal file
@@ -0,0 +1,88 @@
|
||||
// Copyright 2016, Google Inc.
|
||||
// All rights reserved.
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package gax
|
||||
|
||||
import (
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"google.golang.org/grpc"
|
||||
"google.golang.org/grpc/codes"
|
||||
)
|
||||
|
||||
var _ Retryer = &boRetryer{}
|
||||
|
||||
func TestBackofDefault(t *testing.T) {
|
||||
backoff := Backoff{}
|
||||
|
||||
max := []time.Duration{1, 2, 4, 8, 16, 30, 30, 30, 30, 30}
|
||||
for i, m := range max {
|
||||
max[i] = m * time.Second
|
||||
}
|
||||
|
||||
for i, w := range max {
|
||||
if d := backoff.Pause(); d > w {
|
||||
t.Errorf("Backoff duration should be at most %s, got %s", w, d)
|
||||
} else if i < len(max)-1 && backoff.cur != max[i+1] {
|
||||
t.Errorf("current envelope is %s, want %s", backoff.cur, max[i+1])
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestBackoffExponential(t *testing.T) {
|
||||
backoff := Backoff{Initial: 1, Max: 20, Multiplier: 2}
|
||||
want := []time.Duration{1, 2, 4, 8, 16, 20, 20, 20, 20, 20}
|
||||
for _, w := range want {
|
||||
if d := backoff.Pause(); d > w {
|
||||
t.Errorf("Backoff duration should be at most %s, got %s", w, d)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestOnCodes(t *testing.T) {
|
||||
// Lint errors grpc.Errorf in 1.6. It mistakenly expects the first arg to Errorf to be a string.
|
||||
errf := grpc.Errorf
|
||||
apiErr := errf(codes.Unavailable, "")
|
||||
tests := []struct {
|
||||
c []codes.Code
|
||||
retry bool
|
||||
}{
|
||||
{nil, false},
|
||||
{[]codes.Code{codes.DeadlineExceeded}, false},
|
||||
{[]codes.Code{codes.DeadlineExceeded, codes.Unavailable}, true},
|
||||
{[]codes.Code{codes.Unavailable}, true},
|
||||
}
|
||||
for _, tst := range tests {
|
||||
b := OnCodes(tst.c, Backoff{})
|
||||
if _, retry := b.Retry(apiErr); retry != tst.retry {
|
||||
t.Errorf("retriable codes: %v, error code: %s, retry: %t, want %t", tst.c, grpc.Code(apiErr), retry, tst.retry)
|
||||
}
|
||||
}
|
||||
}
|
||||
24
vendor/github.com/googleapis/gax-go/header.go
generated
vendored
Normal file
24
vendor/github.com/googleapis/gax-go/header.go
generated
vendored
Normal file
@@ -0,0 +1,24 @@
|
||||
package gax
|
||||
|
||||
import "bytes"
|
||||
|
||||
// XGoogHeader is for use by the Google Cloud Libraries only.
|
||||
//
|
||||
// XGoogHeader formats key-value pairs.
|
||||
// The resulting string is suitable for x-goog-api-client header.
|
||||
func XGoogHeader(keyval ...string) string {
|
||||
if len(keyval) == 0 {
|
||||
return ""
|
||||
}
|
||||
if len(keyval)%2 != 0 {
|
||||
panic("gax.Header: odd argument count")
|
||||
}
|
||||
var buf bytes.Buffer
|
||||
for i := 0; i < len(keyval); i += 2 {
|
||||
buf.WriteByte(' ')
|
||||
buf.WriteString(keyval[i])
|
||||
buf.WriteByte('/')
|
||||
buf.WriteString(keyval[i+1])
|
||||
}
|
||||
return buf.String()[1:]
|
||||
}
|
||||
19
vendor/github.com/googleapis/gax-go/header_test.go
generated
vendored
Normal file
19
vendor/github.com/googleapis/gax-go/header_test.go
generated
vendored
Normal file
@@ -0,0 +1,19 @@
|
||||
package gax
|
||||
|
||||
import "testing"
|
||||
|
||||
func TestXGoogHeader(t *testing.T) {
|
||||
for _, tst := range []struct {
|
||||
kv []string
|
||||
want string
|
||||
}{
|
||||
{nil, ""},
|
||||
{[]string{"abc", "def"}, "abc/def"},
|
||||
{[]string{"abc", "def", "xyz", "123", "foo", ""}, "abc/def xyz/123 foo/"},
|
||||
} {
|
||||
got := XGoogHeader(tst.kv...)
|
||||
if got != tst.want {
|
||||
t.Errorf("Header(%q) = %q, want %q", tst.kv, got, tst.want)
|
||||
}
|
||||
}
|
||||
}
|
||||
4
vendor/github.com/googleapis/gax-go/invoke.go
generated
vendored
4
vendor/github.com/googleapis/gax-go/invoke.go
generated
vendored
@@ -36,7 +36,7 @@ import (
|
||||
)
|
||||
|
||||
// A user defined call stub.
|
||||
type APICall func(context.Context) error
|
||||
type APICall func(context.Context, CallSettings) error
|
||||
|
||||
// Invoke calls the given APICall,
|
||||
// performing retries as specified by opts, if any.
|
||||
@@ -67,7 +67,7 @@ type sleeper func(ctx context.Context, d time.Duration) error
|
||||
func invoke(ctx context.Context, call APICall, settings CallSettings, sp sleeper) error {
|
||||
var retryer Retryer
|
||||
for {
|
||||
err := call(ctx)
|
||||
err := call(ctx, settings)
|
||||
if err == nil {
|
||||
return nil
|
||||
}
|
||||
|
||||
156
vendor/github.com/googleapis/gax-go/invoke_test.go
generated
vendored
Normal file
156
vendor/github.com/googleapis/gax-go/invoke_test.go
generated
vendored
Normal file
@@ -0,0 +1,156 @@
|
||||
// Copyright 2016, Google Inc.
|
||||
// All rights reserved.
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package gax
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"testing"
|
||||
"time"
|
||||
|
||||
"golang.org/x/net/context"
|
||||
)
|
||||
|
||||
var canceledContext context.Context
|
||||
|
||||
func init() {
|
||||
ctx, cancel := context.WithCancel(context.Background())
|
||||
cancel()
|
||||
canceledContext = ctx
|
||||
}
|
||||
|
||||
// recordSleeper is a test implementation of sleeper.
|
||||
type recordSleeper int
|
||||
|
||||
func (s *recordSleeper) sleep(ctx context.Context, _ time.Duration) error {
|
||||
*s++
|
||||
return ctx.Err()
|
||||
}
|
||||
|
||||
type boolRetryer bool
|
||||
|
||||
func (r boolRetryer) Retry(err error) (time.Duration, bool) { return 0, bool(r) }
|
||||
|
||||
func TestInvokeSuccess(t *testing.T) {
|
||||
apiCall := func(context.Context, CallSettings) error { return nil }
|
||||
var sp recordSleeper
|
||||
err := invoke(context.Background(), apiCall, CallSettings{}, sp.sleep)
|
||||
|
||||
if err != nil {
|
||||
t.Errorf("found error %s, want nil", err)
|
||||
}
|
||||
if sp != 0 {
|
||||
t.Errorf("slept %d times, should not have slept since the call succeeded", int(sp))
|
||||
}
|
||||
}
|
||||
|
||||
func TestInvokeNoRetry(t *testing.T) {
|
||||
apiErr := errors.New("foo error")
|
||||
apiCall := func(context.Context, CallSettings) error { return apiErr }
|
||||
var sp recordSleeper
|
||||
err := invoke(context.Background(), apiCall, CallSettings{}, sp.sleep)
|
||||
|
||||
if err != apiErr {
|
||||
t.Errorf("found error %s, want %s", err, apiErr)
|
||||
}
|
||||
if sp != 0 {
|
||||
t.Errorf("slept %d times, should not have slept since retry is not specified", int(sp))
|
||||
}
|
||||
}
|
||||
|
||||
func TestInvokeNilRetry(t *testing.T) {
|
||||
apiErr := errors.New("foo error")
|
||||
apiCall := func(context.Context, CallSettings) error { return apiErr }
|
||||
var settings CallSettings
|
||||
WithRetry(func() Retryer { return nil }).Resolve(&settings)
|
||||
var sp recordSleeper
|
||||
err := invoke(context.Background(), apiCall, settings, sp.sleep)
|
||||
|
||||
if err != apiErr {
|
||||
t.Errorf("found error %s, want %s", err, apiErr)
|
||||
}
|
||||
if sp != 0 {
|
||||
t.Errorf("slept %d times, should not have slept since retry is not specified", int(sp))
|
||||
}
|
||||
}
|
||||
|
||||
func TestInvokeNeverRetry(t *testing.T) {
|
||||
apiErr := errors.New("foo error")
|
||||
apiCall := func(context.Context, CallSettings) error { return apiErr }
|
||||
var settings CallSettings
|
||||
WithRetry(func() Retryer { return boolRetryer(false) }).Resolve(&settings)
|
||||
var sp recordSleeper
|
||||
err := invoke(context.Background(), apiCall, settings, sp.sleep)
|
||||
|
||||
if err != apiErr {
|
||||
t.Errorf("found error %s, want %s", err, apiErr)
|
||||
}
|
||||
if sp != 0 {
|
||||
t.Errorf("slept %d times, should not have slept since retry is not specified", int(sp))
|
||||
}
|
||||
}
|
||||
|
||||
func TestInvokeRetry(t *testing.T) {
|
||||
const target = 3
|
||||
|
||||
retryNum := 0
|
||||
apiErr := errors.New("foo error")
|
||||
apiCall := func(context.Context, CallSettings) error {
|
||||
retryNum++
|
||||
if retryNum < target {
|
||||
return apiErr
|
||||
}
|
||||
return nil
|
||||
}
|
||||
var settings CallSettings
|
||||
WithRetry(func() Retryer { return boolRetryer(true) }).Resolve(&settings)
|
||||
var sp recordSleeper
|
||||
err := invoke(context.Background(), apiCall, settings, sp.sleep)
|
||||
|
||||
if err != nil {
|
||||
t.Errorf("found error %s, want nil, call should have succeeded after %d tries", err, target)
|
||||
}
|
||||
if sp != target-1 {
|
||||
t.Errorf("retried %d times, want %d", int(sp), int(target-1))
|
||||
}
|
||||
}
|
||||
|
||||
func TestInvokeRetryTimeout(t *testing.T) {
|
||||
apiErr := errors.New("foo error")
|
||||
apiCall := func(context.Context, CallSettings) error { return apiErr }
|
||||
var settings CallSettings
|
||||
WithRetry(func() Retryer { return boolRetryer(true) }).Resolve(&settings)
|
||||
var sp recordSleeper
|
||||
|
||||
err := invoke(canceledContext, apiCall, settings, sp.sleep)
|
||||
|
||||
if err != context.Canceled {
|
||||
t.Errorf("found error %s, want %s", err, context.Canceled)
|
||||
}
|
||||
}
|
||||
211
vendor/github.com/googleapis/gax-go/path_template_test.go
generated
vendored
Normal file
211
vendor/github.com/googleapis/gax-go/path_template_test.go
generated
vendored
Normal file
@@ -0,0 +1,211 @@
|
||||
// Copyright 2016, Google Inc.
|
||||
// All rights reserved.
|
||||
//
|
||||
// Redistribution and use in source and binary forms, with or without
|
||||
// modification, are permitted provided that the following conditions are
|
||||
// met:
|
||||
//
|
||||
// * Redistributions of source code must retain the above copyright
|
||||
// notice, this list of conditions and the following disclaimer.
|
||||
// * Redistributions in binary form must reproduce the above
|
||||
// copyright notice, this list of conditions and the following disclaimer
|
||||
// in the documentation and/or other materials provided with the
|
||||
// distribution.
|
||||
// * Neither the name of Google Inc. nor the names of its
|
||||
// contributors may be used to endorse or promote products derived from
|
||||
// this software without specific prior written permission.
|
||||
//
|
||||
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
||||
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
||||
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
||||
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
||||
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
|
||||
package gax
|
||||
|
||||
import "testing"
|
||||
|
||||
func TestPathTemplateMatchRender(t *testing.T) {
|
||||
testCases := []struct {
|
||||
message string
|
||||
template string
|
||||
path string
|
||||
values map[string]string
|
||||
}{
|
||||
{
|
||||
"base",
|
||||
"buckets/*/*/objects/*",
|
||||
"buckets/f/o/objects/bar",
|
||||
map[string]string{"$0": "f", "$1": "o", "$2": "bar"},
|
||||
},
|
||||
{
|
||||
"path wildcards",
|
||||
"bar/**/foo/*",
|
||||
"bar/foo/foo/foo/bar",
|
||||
map[string]string{"$0": "foo/foo", "$1": "bar"},
|
||||
},
|
||||
{
|
||||
"named binding",
|
||||
"buckets/{foo}/objects/*",
|
||||
"buckets/foo/objects/bar",
|
||||
map[string]string{"$0": "bar", "foo": "foo"},
|
||||
},
|
||||
{
|
||||
"named binding with colon",
|
||||
"buckets/{foo}/objects/*",
|
||||
"buckets/foo:boo/objects/bar",
|
||||
map[string]string{"$0": "bar", "foo": "foo:boo"},
|
||||
},
|
||||
{
|
||||
"named binding with complex patterns",
|
||||
"buckets/{foo=x/*/y/**}/objects/*",
|
||||
"buckets/x/foo/y/bar/baz/objects/quox",
|
||||
map[string]string{"$0": "quox", "foo": "x/foo/y/bar/baz"},
|
||||
},
|
||||
{
|
||||
"starts with slash",
|
||||
"/foo/*",
|
||||
"/foo/bar",
|
||||
map[string]string{"$0": "bar"},
|
||||
},
|
||||
}
|
||||
for _, testCase := range testCases {
|
||||
pt, err := NewPathTemplate(testCase.template)
|
||||
if err != nil {
|
||||
t.Errorf("[%s] Failed to parse template %s: %v", testCase.message, testCase.template, err)
|
||||
continue
|
||||
}
|
||||
values, err := pt.Match(testCase.path)
|
||||
if err != nil {
|
||||
t.Errorf("[%s] PathTemplate '%s' failed to match with '%s', %v", testCase.message, testCase.template, testCase.path, err)
|
||||
continue
|
||||
}
|
||||
for key, expected := range testCase.values {
|
||||
actual, ok := values[key]
|
||||
if !ok {
|
||||
t.Errorf("[%s] The matched data misses the value for %s", testCase.message, key)
|
||||
continue
|
||||
}
|
||||
delete(values, key)
|
||||
if actual != expected {
|
||||
t.Errorf("[%s] Failed to match: value for '%s' is expected '%s' but is actually '%s'", testCase.message, key, expected, actual)
|
||||
}
|
||||
}
|
||||
if len(values) != 0 {
|
||||
t.Errorf("[%s] The matched data has unexpected keys: %v", testCase.message, values)
|
||||
}
|
||||
built, err := pt.Render(testCase.values)
|
||||
if err != nil || built != testCase.path {
|
||||
t.Errorf("[%s] Built path '%s' is different from the expected '%s', %v", testCase.message, built, testCase.path, err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestPathTemplateMatchFailure(t *testing.T) {
|
||||
testCases := []struct {
|
||||
message string
|
||||
template string
|
||||
path string
|
||||
}{
|
||||
{
|
||||
"too many paths",
|
||||
"buckets/*/*/objects/*",
|
||||
"buckets/f/o/o/objects/bar",
|
||||
},
|
||||
{
|
||||
"missing last path",
|
||||
"buckets/*/*/objects/*",
|
||||
"buckets/f/o/objects",
|
||||
},
|
||||
{
|
||||
"too many paths at end",
|
||||
"buckets/*/*/objects/*",
|
||||
"buckets/f/o/objects/too/long",
|
||||
},
|
||||
}
|
||||
for _, testCase := range testCases {
|
||||
pt, err := NewPathTemplate(testCase.template)
|
||||
if err != nil {
|
||||
t.Errorf("[%s] Failed to parse path %s: %v", testCase.message, testCase.template, err)
|
||||
continue
|
||||
}
|
||||
if values, err := pt.Match(testCase.path); err == nil {
|
||||
t.Errorf("[%s] PathTemplate %s doesn't expect to match %s, but succeeded somehow. Match result: %v", testCase.message, testCase.template, testCase.path, values)
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestPathTemplateRenderTooManyValues(t *testing.T) {
|
||||
// Test cases where Render() succeeds but Match() doesn't return the same map.
|
||||
testCases := []struct {
|
||||
message string
|
||||
template string
|
||||
values map[string]string
|
||||
expected string
|
||||
}{
|
||||
{
|
||||
"too many",
|
||||
"bar/*/foo/*",
|
||||
map[string]string{"$0": "_1", "$1": "_2", "$2": "_3"},
|
||||
"bar/_1/foo/_2",
|
||||
},
|
||||
}
|
||||
for _, testCase := range testCases {
|
||||
pt, err := NewPathTemplate(testCase.template)
|
||||
if err != nil {
|
||||
t.Errorf("[%s] Failed to parse template %s (error %v)", testCase.message, testCase.template, err)
|
||||
continue
|
||||
}
|
||||
if result, err := pt.Render(testCase.values); err != nil || result != testCase.expected {
|
||||
t.Errorf("[%s] Failed to build the path (expected '%s' but returned '%s'", testCase.message, testCase.expected, result)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestPathTemplateParseErrors(t *testing.T) {
|
||||
testCases := []struct {
|
||||
message string
|
||||
template string
|
||||
}{
|
||||
{
|
||||
"multiple path wildcards",
|
||||
"foo/**/bar/**",
|
||||
},
|
||||
{
|
||||
"recursive named bindings",
|
||||
"foo/{foo=foo/{bar}/baz/*}/baz/*",
|
||||
},
|
||||
{
|
||||
"complicated multiple path wildcards patterns",
|
||||
"foo/{foo=foo/**/bar/*}/baz/**",
|
||||
},
|
||||
{
|
||||
"consective slashes",
|
||||
"foo//bar",
|
||||
},
|
||||
{
|
||||
"invalid variable pattern",
|
||||
"foo/{foo=foo/*/}bar",
|
||||
},
|
||||
{
|
||||
"same name multiple times",
|
||||
"foo/{foo}/bar/{foo}",
|
||||
},
|
||||
{
|
||||
"empty string after '='",
|
||||
"foo/{foo=}/bar",
|
||||
},
|
||||
}
|
||||
for _, testCase := range testCases {
|
||||
if pt, err := NewPathTemplate(testCase.template); err == nil {
|
||||
t.Errorf("[%s] Template '%s' should fail to be parsed, but succeeded and returned %+v", testCase.message, testCase.template, pt)
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user