2023-04-11 22:20:37 +03:00
|
|
|
package http // import "go.unistack.org/micro/v4/util/http"
|
2019-05-30 23:52:10 +01:00
|
|
|
|
|
|
|
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
|
|
|
|
2023-04-11 22:20:37 +03:00
|
|
|
"go.unistack.org/micro/v4/metadata"
|
|
|
|
"go.unistack.org/micro/v4/selector/random"
|
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))
|
|
|
|
}
|
|
|
|
|
2021-02-14 16:16:01 +03:00
|
|
|
// NewRoundTripper creates new http RoundTripper
|
2019-05-30 23:52:10 +01:00
|
|
|
func NewRoundTripper(opts ...Option) http.RoundTripper {
|
2020-08-25 13:44:41 +03:00
|
|
|
options := Options{}
|
2019-05-30 23:52:10 +01:00
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &roundTripper{
|
|
|
|
rt: http.DefaultTransport,
|
2020-07-01 17:06:59 +01:00
|
|
|
st: random.NewSelector(),
|
2019-05-30 23:52:10 +01:00
|
|
|
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 {
|
2021-02-13 15:35:56 +03:00
|
|
|
md := metadata.New(len(r.Header))
|
2020-03-20 16:23:12 +01:00
|
|
|
for k, v := range r.Header {
|
2021-02-13 15:35:56 +03:00
|
|
|
md.Set(k, strings.Join(v, ","))
|
2020-03-20 16:23:12 +01:00
|
|
|
}
|
2021-02-13 15:35:56 +03:00
|
|
|
return metadata.NewIncomingContext(r.Context(), md)
|
2020-03-20 16:23:12 +01:00
|
|
|
}
|