2018-05-23 18:29:01 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
2019-05-13 17:02:05 +00:00
|
|
|
"bytes"
|
2018-06-28 19:32:16 +00:00
|
|
|
"context"
|
2018-10-31 23:43:18 +00:00
|
|
|
"encoding/json"
|
2019-05-13 17:02:05 +00:00
|
|
|
stderrors "errors"
|
2018-10-29 19:10:33 +00:00
|
|
|
"fmt"
|
2019-05-13 17:02:05 +00:00
|
|
|
"io"
|
2018-05-23 18:29:01 +00:00
|
|
|
"net/http"
|
|
|
|
|
2019-01-08 00:37:16 +00:00
|
|
|
platform "github.com/influxdata/influxdb"
|
2019-05-13 17:02:05 +00:00
|
|
|
"github.com/pkg/errors"
|
2018-05-23 18:29:01 +00:00
|
|
|
)
|
|
|
|
|
2018-06-28 19:32:16 +00:00
|
|
|
const (
|
2018-11-07 18:55:52 +00:00
|
|
|
// PlatformErrorCodeHeader shows the error code of platform error.
|
|
|
|
PlatformErrorCodeHeader = "X-Platform-Error-Code"
|
2018-06-28 19:32:16 +00:00
|
|
|
)
|
|
|
|
|
2018-08-20 21:15:04 +00:00
|
|
|
// AuthzError is returned for authorization errors. When this error type is returned,
|
|
|
|
// the user can be presented with a generic "authorization failed" error, but
|
|
|
|
// the system can log the underlying AuthzError() so that operators have insight
|
|
|
|
// into what actually failed with authorization.
|
|
|
|
type AuthzError interface {
|
|
|
|
error
|
|
|
|
AuthzError() error
|
|
|
|
}
|
|
|
|
|
2018-10-29 19:10:33 +00:00
|
|
|
// CheckErrorStatus for status and any error in the response.
|
2019-01-24 01:02:37 +00:00
|
|
|
func CheckErrorStatus(code int, res *http.Response) error {
|
|
|
|
err := CheckError(res)
|
2018-11-07 18:55:52 +00:00
|
|
|
if err != nil {
|
2018-10-29 19:10:33 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if res.StatusCode != code {
|
|
|
|
return fmt.Errorf("unexpected status code: %s", res.Status)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-05-23 18:29:01 +00:00
|
|
|
// CheckError reads the http.Response and returns an error if one exists.
|
|
|
|
// It will automatically recognize the errors returned by Influx services
|
|
|
|
// and decode the error into an internal error type. If the error cannot
|
|
|
|
// be determined in that way, it will create a generic error message.
|
|
|
|
//
|
|
|
|
// If there is no error, then this returns nil.
|
2019-01-24 01:02:37 +00:00
|
|
|
func CheckError(resp *http.Response) (err error) {
|
2018-05-23 18:29:01 +00:00
|
|
|
switch resp.StatusCode / 100 {
|
|
|
|
case 4, 5:
|
|
|
|
// We will attempt to parse this error outside of this block.
|
|
|
|
case 2:
|
|
|
|
return nil
|
|
|
|
default:
|
|
|
|
// TODO(jsternberg): Figure out what to do here?
|
2019-01-24 00:15:42 +00:00
|
|
|
return &platform.Error{
|
|
|
|
Code: platform.EInternal,
|
|
|
|
Msg: fmt.Sprintf("unexpected status code: %d %s", resp.StatusCode, resp.Status),
|
2018-05-23 18:29:01 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-24 00:15:42 +00:00
|
|
|
pe := new(platform.Error)
|
2019-05-13 17:02:05 +00:00
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if _, err := io.Copy(&buf, resp.Body); err != nil {
|
|
|
|
return &platform.Error{
|
|
|
|
Code: platform.EInternal,
|
|
|
|
Msg: err.Error(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
parseErr := json.Unmarshal(buf.Bytes(), pe)
|
2019-01-24 00:15:42 +00:00
|
|
|
if parseErr != nil {
|
2019-05-13 17:02:05 +00:00
|
|
|
return errors.Wrap(stderrors.New(buf.String()), parseErr.Error())
|
2018-05-23 18:29:01 +00:00
|
|
|
}
|
2019-01-25 01:07:47 +00:00
|
|
|
return pe
|
2018-05-23 18:29:01 +00:00
|
|
|
}
|
2018-06-28 19:32:16 +00:00
|
|
|
|
2019-06-27 01:33:20 +00:00
|
|
|
// ErrorHandler is the error handler in http package.
|
|
|
|
type ErrorHandler int
|
|
|
|
|
|
|
|
// HandleHTTPError encodes err with the appropriate status code and format,
|
2019-01-25 01:07:47 +00:00
|
|
|
// sets the X-Platform-Error-Code headers on the response.
|
|
|
|
// We're no longer using X-Influx-Error and X-Influx-Reference.
|
2018-06-28 19:32:16 +00:00
|
|
|
// and sets the response status to the corresponding status code.
|
2019-06-27 01:33:20 +00:00
|
|
|
func (h ErrorHandler) HandleHTTPError(ctx context.Context, err error, w http.ResponseWriter) {
|
2018-06-28 19:32:16 +00:00
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
2018-10-31 23:43:18 +00:00
|
|
|
|
2019-01-24 00:15:42 +00:00
|
|
|
code := platform.ErrorCode(err)
|
|
|
|
httpCode, ok := statusCodePlatformError[code]
|
2018-06-28 19:32:16 +00:00
|
|
|
if !ok {
|
2019-01-24 00:15:42 +00:00
|
|
|
httpCode = http.StatusBadRequest
|
2018-06-28 19:32:16 +00:00
|
|
|
}
|
2019-01-24 00:15:42 +00:00
|
|
|
w.Header().Set(PlatformErrorCodeHeader, code)
|
|
|
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
|
|
|
w.WriteHeader(httpCode)
|
2019-01-24 18:56:05 +00:00
|
|
|
var e error
|
2019-01-24 00:15:42 +00:00
|
|
|
if pe, ok := err.(*platform.Error); ok {
|
2019-01-24 18:56:05 +00:00
|
|
|
e = &platform.Error{
|
|
|
|
Code: code,
|
|
|
|
Op: platform.ErrorOp(err),
|
|
|
|
Msg: platform.ErrorMessage(err),
|
|
|
|
Err: pe.Err,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
e = &platform.Error{
|
|
|
|
Code: platform.EInternal,
|
|
|
|
Err: err,
|
|
|
|
}
|
2018-06-28 19:32:16 +00:00
|
|
|
}
|
2019-01-24 00:15:42 +00:00
|
|
|
b, _ := json.Marshal(e)
|
|
|
|
_, _ = w.Write(b)
|
2018-09-28 18:33:35 +00:00
|
|
|
}
|
|
|
|
|
2019-01-18 22:43:00 +00:00
|
|
|
// UnauthorizedError encodes a error message and status code for unauthorized access.
|
2019-06-27 01:33:20 +00:00
|
|
|
func UnauthorizedError(ctx context.Context, h platform.HTTPErrorHandler, w http.ResponseWriter) {
|
|
|
|
h.HandleHTTPError(ctx, &platform.Error{
|
2019-01-18 22:43:00 +00:00
|
|
|
Code: platform.EUnauthorized,
|
|
|
|
Msg: "unauthorized access",
|
|
|
|
}, w)
|
|
|
|
}
|
|
|
|
|
2018-10-31 23:43:18 +00:00
|
|
|
// statusCodePlatformError is the map convert platform.Error to error
|
|
|
|
var statusCodePlatformError = map[string]int{
|
2019-01-24 00:15:42 +00:00
|
|
|
platform.EInternal: http.StatusInternalServerError,
|
|
|
|
platform.EInvalid: http.StatusBadRequest,
|
|
|
|
platform.EUnprocessableEntity: http.StatusUnprocessableEntity,
|
|
|
|
platform.EEmptyValue: http.StatusBadRequest,
|
|
|
|
platform.EConflict: http.StatusUnprocessableEntity,
|
|
|
|
platform.ENotFound: http.StatusNotFound,
|
|
|
|
platform.EUnavailable: http.StatusServiceUnavailable,
|
|
|
|
platform.EForbidden: http.StatusForbidden,
|
2019-04-16 23:50:35 +00:00
|
|
|
platform.ETooManyRequests: http.StatusTooManyRequests,
|
2019-01-24 00:15:42 +00:00
|
|
|
platform.EUnauthorized: http.StatusUnauthorized,
|
|
|
|
platform.EMethodNotAllowed: http.StatusMethodNotAllowed,
|
2018-10-31 23:43:18 +00:00
|
|
|
}
|