2019-05-30 23:52:10 +01:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
2020-03-20 16:23:12 +01:00
|
|
|
"context"
|
2020-03-07 11:06:57 +00:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
2019-05-30 23:52:10 +01:00
|
|
|
"net/http"
|
2020-03-20 16:23:12 +01:00
|
|
|
"strings"
|
2019-05-30 23:52:10 +01:00
|
|
|
|
2021-01-20 13:54:31 +00:00
|
|
|
"github.com/asim/go-micro/v3/metadata"
|
|
|
|
"github.com/asim/go-micro/v3/registry"
|
|
|
|
"github.com/asim/go-micro/v3/selector"
|
2019-05-30 23:52:10 +01:00
|
|
|
)
|
|
|
|
|
2020-03-07 11:06:57 +00:00
|
|
|
// Write sets the status and body on a http ResponseWriter
|
|
|
|
func Write(w http.ResponseWriter, contentType string, status int, body string) {
|
|
|
|
w.Header().Set("Content-Length", fmt.Sprintf("%v", len(body)))
|
|
|
|
w.Header().Set("Content-Type", contentType)
|
|
|
|
w.WriteHeader(status)
|
|
|
|
fmt.Fprintf(w, `%v`, body)
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteBadRequestError sets a 400 status code
|
|
|
|
func WriteBadRequestError(w http.ResponseWriter, err error) {
|
|
|
|
rawBody, err := json.Marshal(map[string]string{
|
|
|
|
"error": err.Error(),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
WriteInternalServerError(w, err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
Write(w, "application/json", 400, string(rawBody))
|
|
|
|
}
|
|
|
|
|
|
|
|
// WriteInternalServerError sets a 500 status code
|
|
|
|
func WriteInternalServerError(w http.ResponseWriter, err error) {
|
|
|
|
rawBody, err := json.Marshal(map[string]string{
|
|
|
|
"error": err.Error(),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Println(err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
Write(w, "application/json", 500, string(rawBody))
|
|
|
|
}
|
|
|
|
|
2019-05-30 23:52:10 +01:00
|
|
|
func NewRoundTripper(opts ...Option) http.RoundTripper {
|
|
|
|
options := Options{
|
|
|
|
Registry: registry.DefaultRegistry,
|
|
|
|
}
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &roundTripper{
|
|
|
|
rt: http.DefaultTransport,
|
|
|
|
st: selector.Random,
|
|
|
|
opts: options,
|
|
|
|
}
|
|
|
|
}
|
2020-03-20 16:23:12 +01:00
|
|
|
|
|
|
|
// RequestToContext puts the `Authorization` header bearer token into context
|
|
|
|
// so calls to services will be authorized.
|
|
|
|
func RequestToContext(r *http.Request) context.Context {
|
|
|
|
ctx := context.Background()
|
|
|
|
md := make(metadata.Metadata)
|
|
|
|
for k, v := range r.Header {
|
|
|
|
md[k] = strings.Join(v, ",")
|
|
|
|
}
|
|
|
|
return metadata.NewContext(ctx, md)
|
|
|
|
}
|