mirror of https://github.com/linkerd/linkerd2.git
155 lines
4.0 KiB
Go
155 lines
4.0 KiB
Go
package public
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/binary"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
|
|
"github.com/golang/protobuf/proto"
|
|
pb "github.com/linkerd/linkerd2/controller/gen/public"
|
|
log "github.com/sirupsen/logrus"
|
|
"google.golang.org/grpc/status"
|
|
)
|
|
|
|
const (
|
|
errorHeader = "linkerd-error"
|
|
defaultHTTPErrorStatusCode = http.StatusInternalServerError
|
|
contentTypeHeader = "Content-Type"
|
|
protobufContentType = "application/octet-stream"
|
|
numBytesForMessageLength = 4
|
|
)
|
|
|
|
type httpError struct {
|
|
Code int
|
|
WrappedError error
|
|
}
|
|
|
|
type flushableResponseWriter interface {
|
|
http.ResponseWriter
|
|
http.Flusher
|
|
}
|
|
|
|
func (e httpError) Error() string {
|
|
return fmt.Sprintf("HTTP error, status Code [%d], wrapped error is: %v", e.Code, e.WrappedError)
|
|
}
|
|
|
|
func httpRequestToProto(req *http.Request, protoRequestOut proto.Message) error {
|
|
bytes, err := ioutil.ReadAll(req.Body)
|
|
if err != nil {
|
|
return httpError{
|
|
Code: http.StatusBadRequest,
|
|
WrappedError: err,
|
|
}
|
|
}
|
|
|
|
err = proto.Unmarshal(bytes, protoRequestOut)
|
|
if err != nil {
|
|
return httpError{
|
|
Code: http.StatusBadRequest,
|
|
WrappedError: err,
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func writeErrorToHTTPResponse(w http.ResponseWriter, errorObtained error) {
|
|
statusCode := defaultHTTPErrorStatusCode
|
|
errorToReturn := errorObtained
|
|
|
|
if httpErr, ok := errorObtained.(httpError); ok {
|
|
statusCode = httpErr.Code
|
|
errorToReturn = httpErr.WrappedError
|
|
}
|
|
|
|
w.Header().Set(errorHeader, http.StatusText(statusCode))
|
|
|
|
errorMessageToReturn := errorToReturn.Error()
|
|
if grpcError, ok := status.FromError(errorObtained); ok {
|
|
errorMessageToReturn = grpcError.Message()
|
|
}
|
|
|
|
errorAsProto := &pb.ApiError{Error: errorMessageToReturn}
|
|
|
|
err := writeProtoToHTTPResponse(w, errorAsProto)
|
|
if err != nil {
|
|
log.Errorf("Error writing error to http response: %v", err)
|
|
w.Header().Set(errorHeader, err.Error())
|
|
}
|
|
}
|
|
|
|
func writeProtoToHTTPResponse(w http.ResponseWriter, msg proto.Message) error {
|
|
w.Header().Set(contentTypeHeader, protobufContentType)
|
|
marshalledProtobufMessage, err := proto.Marshal(msg)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
fullPayload := serializeAsPayload(marshalledProtobufMessage)
|
|
_, err = w.Write(fullPayload)
|
|
return err
|
|
}
|
|
|
|
func newStreamingWriter(w http.ResponseWriter) (flushableResponseWriter, error) {
|
|
flushableWriter, ok := w.(flushableResponseWriter)
|
|
if !ok {
|
|
return nil, fmt.Errorf("streaming not supported by this writer")
|
|
}
|
|
|
|
flushableWriter.Header().Set("Connection", "keep-alive")
|
|
flushableWriter.Header().Set("Transfer-Encoding", "chunked")
|
|
return flushableWriter, nil
|
|
}
|
|
|
|
func serializeAsPayload(messageContentsInBytes []byte) []byte {
|
|
lengthOfThePayload := uint32(len(messageContentsInBytes))
|
|
|
|
messageLengthInBytes := make([]byte, numBytesForMessageLength)
|
|
binary.LittleEndian.PutUint32(messageLengthInBytes, lengthOfThePayload)
|
|
|
|
return append(messageLengthInBytes, messageContentsInBytes...)
|
|
}
|
|
|
|
func deserializePayloadFromReader(reader *bufio.Reader) ([]byte, error) {
|
|
messageLengthAsBytes := make([]byte, numBytesForMessageLength)
|
|
_, err := io.ReadFull(reader, messageLengthAsBytes)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error while reading message length: %v", err)
|
|
}
|
|
messageLength := int(binary.LittleEndian.Uint32(messageLengthAsBytes))
|
|
|
|
messageContentsAsBytes := make([]byte, messageLength)
|
|
_, err = io.ReadFull(reader, messageContentsAsBytes)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error while reading bytes from message: %v", err)
|
|
}
|
|
|
|
return messageContentsAsBytes, nil
|
|
}
|
|
|
|
func checkIfResponseHasError(rsp *http.Response) error {
|
|
errorMsg := rsp.Header.Get(errorHeader)
|
|
|
|
if errorMsg != "" {
|
|
reader := bufio.NewReader(rsp.Body)
|
|
var apiError pb.ApiError
|
|
|
|
err := fromByteStreamToProtocolBuffers(reader, &apiError)
|
|
if err != nil {
|
|
return fmt.Errorf("Response has %s header [%s], but response body didn't contain protobuf error: %v", errorHeader, errorMsg, err)
|
|
}
|
|
|
|
return errors.New(apiError.Error)
|
|
}
|
|
|
|
if rsp.StatusCode != http.StatusOK {
|
|
return fmt.Errorf("Unexpected API response: %s", rsp.Status)
|
|
}
|
|
|
|
return nil
|
|
}
|