mirror of
https://github.com/ceph/ceph-csi.git
synced 2025-01-10 22:09:30 +00:00
25e3a961c3
When the initial DeleteVolume times out (as it does on slow clusters due to the low 10 second limit), the external-provisioner calls it again. The CSI standard requires the second call to succeed if the volume has been deleted in the meantime. This didn't work because DeleteVolume returned an error when failing to find the volume info file: rbdplugin: E1008 08:05:35.631783 1 utils.go:100] GRPC error: rbd: open err /var/lib/kubelet/plugins/csi-rbdplugin/controller/csi-rbd-622a252c-cad0-11e8-9112-deadbeef0101.json/open /var/lib/kubelet/plugins/csi-rbdplugin/controller/csi-rbd-622a252c-cad0-11e8-9112-deadbeef0101.json: no such file or directory The fix is to treat a missing volume info file as "volume already deleted" and return success. To detect this, the original os error must be wrapped, otherwise the caller of loadVolInfo cannot determine the root cause. Note that further work may be needed to make the driver really resilient, for example there are probably concurrency issues. But for now this fixes: #82
293 lines
5.4 KiB
Go
293 lines
5.4 KiB
Go
package errors
|
|
|
|
import (
|
|
"fmt"
|
|
"runtime"
|
|
"testing"
|
|
)
|
|
|
|
var initpc, _, _, _ = runtime.Caller(0)
|
|
|
|
func TestFrameLine(t *testing.T) {
|
|
var tests = []struct {
|
|
Frame
|
|
want int
|
|
}{{
|
|
Frame(initpc),
|
|
9,
|
|
}, {
|
|
func() Frame {
|
|
var pc, _, _, _ = runtime.Caller(0)
|
|
return Frame(pc)
|
|
}(),
|
|
20,
|
|
}, {
|
|
func() Frame {
|
|
var pc, _, _, _ = runtime.Caller(1)
|
|
return Frame(pc)
|
|
}(),
|
|
28,
|
|
}, {
|
|
Frame(0), // invalid PC
|
|
0,
|
|
}}
|
|
|
|
for _, tt := range tests {
|
|
got := tt.Frame.line()
|
|
want := tt.want
|
|
if want != got {
|
|
t.Errorf("Frame(%v): want: %v, got: %v", uintptr(tt.Frame), want, got)
|
|
}
|
|
}
|
|
}
|
|
|
|
type X struct{}
|
|
|
|
func (x X) val() Frame {
|
|
var pc, _, _, _ = runtime.Caller(0)
|
|
return Frame(pc)
|
|
}
|
|
|
|
func (x *X) ptr() Frame {
|
|
var pc, _, _, _ = runtime.Caller(0)
|
|
return Frame(pc)
|
|
}
|
|
|
|
func TestFrameFormat(t *testing.T) {
|
|
var tests = []struct {
|
|
Frame
|
|
format string
|
|
want string
|
|
}{{
|
|
Frame(initpc),
|
|
"%s",
|
|
"stack_test.go",
|
|
}, {
|
|
Frame(initpc),
|
|
"%+s",
|
|
"github.com/pkg/errors.init\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go",
|
|
}, {
|
|
Frame(0),
|
|
"%s",
|
|
"unknown",
|
|
}, {
|
|
Frame(0),
|
|
"%+s",
|
|
"unknown",
|
|
}, {
|
|
Frame(initpc),
|
|
"%d",
|
|
"9",
|
|
}, {
|
|
Frame(0),
|
|
"%d",
|
|
"0",
|
|
}, {
|
|
Frame(initpc),
|
|
"%n",
|
|
"init",
|
|
}, {
|
|
func() Frame {
|
|
var x X
|
|
return x.ptr()
|
|
}(),
|
|
"%n",
|
|
`\(\*X\).ptr`,
|
|
}, {
|
|
func() Frame {
|
|
var x X
|
|
return x.val()
|
|
}(),
|
|
"%n",
|
|
"X.val",
|
|
}, {
|
|
Frame(0),
|
|
"%n",
|
|
"",
|
|
}, {
|
|
Frame(initpc),
|
|
"%v",
|
|
"stack_test.go:9",
|
|
}, {
|
|
Frame(initpc),
|
|
"%+v",
|
|
"github.com/pkg/errors.init\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:9",
|
|
}, {
|
|
Frame(0),
|
|
"%v",
|
|
"unknown:0",
|
|
}}
|
|
|
|
for i, tt := range tests {
|
|
testFormatRegexp(t, i, tt.Frame, tt.format, tt.want)
|
|
}
|
|
}
|
|
|
|
func TestFuncname(t *testing.T) {
|
|
tests := []struct {
|
|
name, want string
|
|
}{
|
|
{"", ""},
|
|
{"runtime.main", "main"},
|
|
{"github.com/pkg/errors.funcname", "funcname"},
|
|
{"funcname", "funcname"},
|
|
{"io.copyBuffer", "copyBuffer"},
|
|
{"main.(*R).Write", "(*R).Write"},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
got := funcname(tt.name)
|
|
want := tt.want
|
|
if got != want {
|
|
t.Errorf("funcname(%q): want: %q, got %q", tt.name, want, got)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestTrimGOPATH(t *testing.T) {
|
|
var tests = []struct {
|
|
Frame
|
|
want string
|
|
}{{
|
|
Frame(initpc),
|
|
"github.com/pkg/errors/stack_test.go",
|
|
}}
|
|
|
|
for i, tt := range tests {
|
|
pc := tt.Frame.pc()
|
|
fn := runtime.FuncForPC(pc)
|
|
file, _ := fn.FileLine(pc)
|
|
got := trimGOPATH(fn.Name(), file)
|
|
testFormatRegexp(t, i, got, "%s", tt.want)
|
|
}
|
|
}
|
|
|
|
func TestStackTrace(t *testing.T) {
|
|
tests := []struct {
|
|
err error
|
|
want []string
|
|
}{{
|
|
New("ooh"), []string{
|
|
"github.com/pkg/errors.TestStackTrace\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:172",
|
|
},
|
|
}, {
|
|
Wrap(New("ooh"), "ahh"), []string{
|
|
"github.com/pkg/errors.TestStackTrace\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:177", // this is the stack of Wrap, not New
|
|
},
|
|
}, {
|
|
Cause(Wrap(New("ooh"), "ahh")), []string{
|
|
"github.com/pkg/errors.TestStackTrace\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:182", // this is the stack of New
|
|
},
|
|
}, {
|
|
func() error { return New("ooh") }(), []string{
|
|
`github.com/pkg/errors.(func·009|TestStackTrace.func1)` +
|
|
"\n\t.+/github.com/pkg/errors/stack_test.go:187", // this is the stack of New
|
|
"github.com/pkg/errors.TestStackTrace\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:187", // this is the stack of New's caller
|
|
},
|
|
}, {
|
|
Cause(func() error {
|
|
return func() error {
|
|
return Errorf("hello %s", fmt.Sprintf("world"))
|
|
}()
|
|
}()), []string{
|
|
`github.com/pkg/errors.(func·010|TestStackTrace.func2.1)` +
|
|
"\n\t.+/github.com/pkg/errors/stack_test.go:196", // this is the stack of Errorf
|
|
`github.com/pkg/errors.(func·011|TestStackTrace.func2)` +
|
|
"\n\t.+/github.com/pkg/errors/stack_test.go:197", // this is the stack of Errorf's caller
|
|
"github.com/pkg/errors.TestStackTrace\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:198", // this is the stack of Errorf's caller's caller
|
|
},
|
|
}}
|
|
for i, tt := range tests {
|
|
x, ok := tt.err.(interface {
|
|
StackTrace() StackTrace
|
|
})
|
|
if !ok {
|
|
t.Errorf("expected %#v to implement StackTrace() StackTrace", tt.err)
|
|
continue
|
|
}
|
|
st := x.StackTrace()
|
|
for j, want := range tt.want {
|
|
testFormatRegexp(t, i, st[j], "%+v", want)
|
|
}
|
|
}
|
|
}
|
|
|
|
func stackTrace() StackTrace {
|
|
const depth = 8
|
|
var pcs [depth]uintptr
|
|
n := runtime.Callers(1, pcs[:])
|
|
var st stack = pcs[0:n]
|
|
return st.StackTrace()
|
|
}
|
|
|
|
func TestStackTraceFormat(t *testing.T) {
|
|
tests := []struct {
|
|
StackTrace
|
|
format string
|
|
want string
|
|
}{{
|
|
nil,
|
|
"%s",
|
|
`\[\]`,
|
|
}, {
|
|
nil,
|
|
"%v",
|
|
`\[\]`,
|
|
}, {
|
|
nil,
|
|
"%+v",
|
|
"",
|
|
}, {
|
|
nil,
|
|
"%#v",
|
|
`\[\]errors.Frame\(nil\)`,
|
|
}, {
|
|
make(StackTrace, 0),
|
|
"%s",
|
|
`\[\]`,
|
|
}, {
|
|
make(StackTrace, 0),
|
|
"%v",
|
|
`\[\]`,
|
|
}, {
|
|
make(StackTrace, 0),
|
|
"%+v",
|
|
"",
|
|
}, {
|
|
make(StackTrace, 0),
|
|
"%#v",
|
|
`\[\]errors.Frame{}`,
|
|
}, {
|
|
stackTrace()[:2],
|
|
"%s",
|
|
`\[stack_test.go stack_test.go\]`,
|
|
}, {
|
|
stackTrace()[:2],
|
|
"%v",
|
|
`\[stack_test.go:225 stack_test.go:272\]`,
|
|
}, {
|
|
stackTrace()[:2],
|
|
"%+v",
|
|
"\n" +
|
|
"github.com/pkg/errors.stackTrace\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:225\n" +
|
|
"github.com/pkg/errors.TestStackTraceFormat\n" +
|
|
"\t.+/github.com/pkg/errors/stack_test.go:276",
|
|
}, {
|
|
stackTrace()[:2],
|
|
"%#v",
|
|
`\[\]errors.Frame{stack_test.go:225, stack_test.go:284}`,
|
|
}}
|
|
|
|
for i, tt := range tests {
|
|
testFormatRegexp(t, i, tt.StackTrace, tt.format, tt.want)
|
|
}
|
|
}
|