Skip to content

Commit

Permalink
Add more context when failing to unmarshal JSON
Browse files Browse the repository at this point in the history
As mentioned in #830, we occasionally have a situation where non-JSON
data comes back to a client even though this is not supposed to happen.
Unfortunately, when it does, the error message produced by Go isn't very
good and doesn't help much with debugging.

Here we add a new helper function for unmarshaling JSON in standard or
error responses that behaves normally in the normal cases, and in case
of a failure, adds more context to the standard library's error and
logs.
  • Loading branch information
brandur committed Apr 15, 2019
1 parent 2fb23ba commit c3f8af1
Show file tree
Hide file tree
Showing 2 changed files with 74 additions and 2 deletions.
31 changes: 29 additions & 2 deletions stripe.go
Original file line number Diff line number Diff line change
Expand Up @@ -441,7 +441,7 @@ func (s *BackendImplementation) Do(req *http.Request, body *bytes.Buffer, v inte
s.LeveledLogger.Debugf("Response: %s\n", string(resBody))

if v != nil {
return json.Unmarshal(resBody, v)
return s.UnmarshalJSONVerbose(res.StatusCode, resBody, v)
}

return nil
Expand All @@ -450,9 +450,10 @@ func (s *BackendImplementation) Do(req *http.Request, body *bytes.Buffer, v inte
// ResponseToError converts a stripe response to an Error.
func (s *BackendImplementation) ResponseToError(res *http.Response, resBody []byte) error {
var raw rawError
if err := json.Unmarshal(resBody, &raw); err != nil {
if err := s.UnmarshalJSONVerbose(res.StatusCode, resBody, &raw); err != nil {
return err
}

// no error in resBody
if raw.E == nil {
err := errors.New(string(resBody))
Expand Down Expand Up @@ -505,6 +506,32 @@ func (s *BackendImplementation) SetNetworkRetriesSleep(sleep bool) {
s.networkRetriesSleep = sleep
}

// UnmarshalJSONVerbose unmarshals JSON, but in case of a failure logs and
// produces a more descriptive error.
func (s *BackendImplementation) UnmarshalJSONVerbose(statusCode int, body []byte, v interface{}) error {
err := json.Unmarshal(body, v)
if err != nil {
// If we got invalid JSON back then something totally unexpected is
// happening (caused by a bug on the server side). Put a sample of the
// response body into the error message so we can get a better feel for
// what the problem was.
bodySample := string(body)
if len(bodySample) > 500 {
bodySample = bodySample[0:500] + " ..."
}

// Make sure a multi-line response ends up all on one line
bodySample = strings.Replace(bodySample, "\n", "\\n", -1)

newErr := fmt.Errorf("Couldn't deserialize JSON (response status: %v, body sample: '%s'): %v",
statusCode, bodySample, err)
s.LeveledLogger.Errorf("%s", newErr.Error())
return newErr
}

return nil
}

// Checks if an error is a problem that we should retry on. This includes both
// socket errors that may represent an intermittent problem and some special
// HTTP statuses.
Expand Down
45 changes: 45 additions & 0 deletions stripe_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -527,6 +527,51 @@ func TestStripeAccount(t *testing.T) {
assert.Equal(t, TestMerchantID, req.Header.Get("Stripe-Account"))
}

func TestUnmarshalJSONVerbose(t *testing.T) {
type testServerResponse struct {
Message string `json:"message"`
}

backend := stripe.GetBackend(stripe.APIBackend).(*stripe.BackendImplementation)

// Valid JSON
{
type testServerResponse struct {
Message string `json:"message"`
}

var sample testServerResponse
err := backend.UnmarshalJSONVerbose(200, []byte(`{"message":"hello"}`), &sample)
assert.NoError(t, err)
assert.Equal(t, "hello", sample.Message)
}

// Invalid JSON (short)
{
body := `server error`

var sample testServerResponse
err := backend.UnmarshalJSONVerbose(200, []byte(body), &sample)
assert.Regexp(t,
fmt.Sprintf(`^Couldn't deserialize JSON \(response status: 200, body sample: '%s'\): invalid character`, body),
err)
}

// Invalid JSON (long, and therefore truncated)
{
// Assembles a body that's at least as long as our maximum sample.
// body is ~130 characters * 5.
bodyText := `this is a really long body that will be truncated when added to the error message to protect against dumping huge responses in logs `
body := bodyText + bodyText + bodyText + bodyText + bodyText

var sample testServerResponse
err := backend.UnmarshalJSONVerbose(200, []byte(body), &sample)
assert.Regexp(t,
fmt.Sprintf(`^Couldn't deserialize JSON \(response status: 200, body sample: '%s ...'\): invalid character`, body[0:500]),
err)
}
}

func TestUserAgent(t *testing.T) {
c := stripe.GetBackend(stripe.APIBackend).(*stripe.BackendImplementation)

Expand Down

0 comments on commit c3f8af1

Please sign in to comment.