Skip to content

Commit 86ed770

Browse files
Implement errors.Unwrap() on unexpected status code errors
With this patch, we add a method with signature `Unwrap() error` to all errors signaling an unexpected status code. All of them contain a value of type `gophercloud.ErrUnexpectedStatusCode` that exposes, among other useful values, the body of the response sent by the server. To access the response body of a request that resulted in an unexpected response code, try to unwrap the returned error to `gophercloud.ErrUnexpectedStatusCode`: ```Go pages, err := containers.List(client, nil).AllPages() if err != nil { var responseCodeError gophercloud.ErrUnexpectedResponseCode if errors.As(err, &responseCodeError) { log.Printf("unexpected response code. Response body:\n---\n%s\n---", responseCodeError.Body) } else { log.Printf("unknown error") } } ```
1 parent 6e1423b commit 86ed770

File tree

2 files changed

+102
-1
lines changed

2 files changed

+102
-1
lines changed

errors.go

+48
Original file line numberDiff line numberDiff line change
@@ -116,61 +116,109 @@ type ErrDefault400 struct {
116116
ErrUnexpectedResponseCode
117117
}
118118

119+
func (e ErrDefault400) Unwrap() error {
120+
return e.ErrUnexpectedResponseCode
121+
}
122+
119123
// ErrDefault401 is the default error type returned on a 401 HTTP response code.
120124
type ErrDefault401 struct {
121125
ErrUnexpectedResponseCode
122126
}
123127

128+
func (e ErrDefault401) Unwrap() error {
129+
return e.ErrUnexpectedResponseCode
130+
}
131+
124132
// ErrDefault403 is the default error type returned on a 403 HTTP response code.
125133
type ErrDefault403 struct {
126134
ErrUnexpectedResponseCode
127135
}
128136

137+
func (e ErrDefault403) Unwrap() error {
138+
return e.ErrUnexpectedResponseCode
139+
}
140+
129141
// ErrDefault404 is the default error type returned on a 404 HTTP response code.
130142
type ErrDefault404 struct {
131143
ErrUnexpectedResponseCode
132144
}
133145

146+
func (e ErrDefault404) Unwrap() error {
147+
return e.ErrUnexpectedResponseCode
148+
}
149+
134150
// ErrDefault405 is the default error type returned on a 405 HTTP response code.
135151
type ErrDefault405 struct {
136152
ErrUnexpectedResponseCode
137153
}
138154

155+
func (e ErrDefault405) Unwrap() error {
156+
return e.ErrUnexpectedResponseCode
157+
}
158+
139159
// ErrDefault408 is the default error type returned on a 408 HTTP response code.
140160
type ErrDefault408 struct {
141161
ErrUnexpectedResponseCode
142162
}
143163

164+
func (e ErrDefault408) Unwrap() error {
165+
return e.ErrUnexpectedResponseCode
166+
}
167+
144168
// ErrDefault409 is the default error type returned on a 409 HTTP response code.
145169
type ErrDefault409 struct {
146170
ErrUnexpectedResponseCode
147171
}
148172

173+
func (e ErrDefault409) Unwrap() error {
174+
return e.ErrUnexpectedResponseCode
175+
}
176+
149177
// ErrDefault429 is the default error type returned on a 429 HTTP response code.
150178
type ErrDefault429 struct {
151179
ErrUnexpectedResponseCode
152180
}
153181

182+
func (e ErrDefault429) Unwrap() error {
183+
return e.ErrUnexpectedResponseCode
184+
}
185+
154186
// ErrDefault500 is the default error type returned on a 500 HTTP response code.
155187
type ErrDefault500 struct {
156188
ErrUnexpectedResponseCode
157189
}
158190

191+
func (e ErrDefault500) Unwrap() error {
192+
return e.ErrUnexpectedResponseCode
193+
}
194+
159195
// ErrDefault502 is the default error type returned on a 502 HTTP response code.
160196
type ErrDefault502 struct {
161197
ErrUnexpectedResponseCode
162198
}
163199

200+
func (e ErrDefault502) Unwrap() error {
201+
return e.ErrUnexpectedResponseCode
202+
}
203+
164204
// ErrDefault503 is the default error type returned on a 503 HTTP response code.
165205
type ErrDefault503 struct {
166206
ErrUnexpectedResponseCode
167207
}
168208

209+
func (e ErrDefault503) Unwrap() error {
210+
return e.ErrUnexpectedResponseCode
211+
}
212+
169213
// ErrDefault504 is the default error type returned on a 504 HTTP response code.
170214
type ErrDefault504 struct {
171215
ErrUnexpectedResponseCode
172216
}
173217

218+
func (e ErrDefault504) Unwrap() error {
219+
return e.ErrUnexpectedResponseCode
220+
}
221+
174222
func (e ErrDefault400) Error() string {
175223
e.DefaultErrString = fmt.Sprintf(
176224
"Bad request with: [%s %s], error message: %s",

testing/errors_test.go

+54-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package testing
22

33
import (
4+
"errors"
45
"testing"
56

67
"github.com/gophercloud/gophercloud"
@@ -13,7 +14,7 @@ func returnsUnexpectedResp(code int) gophercloud.ErrUnexpectedResponseCode {
1314
Method: "GET",
1415
Expected: []int{200},
1516
Actual: code,
16-
Body: nil,
17+
Body: []byte("the response body"),
1718
ResponseHeader: nil,
1819
}
1920
}
@@ -24,6 +25,17 @@ func TestGetResponseCode404(t *testing.T) {
2425
err, ok := err404.(gophercloud.StatusCodeError)
2526
th.AssertEquals(t, true, ok)
2627
th.AssertEquals(t, err.GetStatusCode(), 404)
28+
29+
t.Run("wraps ErrUnexpectedResponseCode", func(t *testing.T) {
30+
var unexpectedResponseCode gophercloud.ErrUnexpectedResponseCode
31+
if errors.As(err, &unexpectedResponseCode) {
32+
if want, have := "the response body", string(unexpectedResponseCode.Body); want != have {
33+
t.Errorf("expected the wrapped error to contain the response body, found %q", have)
34+
}
35+
} else {
36+
t.Errorf("err.Unwrap() didn't return ErrUnexpectedResponseCode")
37+
}
38+
})
2739
}
2840

2941
func TestGetResponseCode502(t *testing.T) {
@@ -32,6 +44,17 @@ func TestGetResponseCode502(t *testing.T) {
3244
err, ok := err502.(gophercloud.StatusCodeError)
3345
th.AssertEquals(t, true, ok)
3446
th.AssertEquals(t, err.GetStatusCode(), 502)
47+
48+
t.Run("wraps ErrUnexpectedResponseCode", func(t *testing.T) {
49+
var unexpectedResponseCode gophercloud.ErrUnexpectedResponseCode
50+
if errors.As(err, &unexpectedResponseCode) {
51+
if want, have := "the response body", string(unexpectedResponseCode.Body); want != have {
52+
t.Errorf("expected the wrapped error to contain the response body, found %q", have)
53+
}
54+
} else {
55+
t.Errorf("err.Unwrap() didn't return ErrUnexpectedResponseCode")
56+
}
57+
})
3558
}
3659

3760
func TestGetResponseCode504(t *testing.T) {
@@ -40,4 +63,34 @@ func TestGetResponseCode504(t *testing.T) {
4063
err, ok := err504.(gophercloud.StatusCodeError)
4164
th.AssertEquals(t, true, ok)
4265
th.AssertEquals(t, err.GetStatusCode(), 504)
66+
67+
t.Run("wraps ErrUnexpectedResponseCode", func(t *testing.T) {
68+
var unexpectedResponseCode gophercloud.ErrUnexpectedResponseCode
69+
if errors.As(err, &unexpectedResponseCode) {
70+
if want, have := "the response body", string(unexpectedResponseCode.Body); want != have {
71+
t.Errorf("expected the wrapped error to contain the response body, found %q", have)
72+
}
73+
} else {
74+
t.Errorf("err.Unwrap() didn't return ErrUnexpectedResponseCode")
75+
}
76+
})
4377
}
78+
79+
// Compile-time check that all response-code errors implement `Unwrap()`
80+
type unwrapper interface {
81+
Unwrap() error
82+
}
83+
84+
var (
85+
_ unwrapper = gophercloud.ErrDefault401{}
86+
_ unwrapper = gophercloud.ErrDefault403{}
87+
_ unwrapper = gophercloud.ErrDefault404{}
88+
_ unwrapper = gophercloud.ErrDefault405{}
89+
_ unwrapper = gophercloud.ErrDefault408{}
90+
_ unwrapper = gophercloud.ErrDefault409{}
91+
_ unwrapper = gophercloud.ErrDefault429{}
92+
_ unwrapper = gophercloud.ErrDefault500{}
93+
_ unwrapper = gophercloud.ErrDefault502{}
94+
_ unwrapper = gophercloud.ErrDefault503{}
95+
_ unwrapper = gophercloud.ErrDefault504{}
96+
)

0 commit comments

Comments
 (0)