2019-06-03 18:44:43 +01:00
|
|
|
// Package http provides a micro rpc to http proxy
|
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"path"
|
|
|
|
|
2020-08-19 17:47:17 +03:00
|
|
|
"github.com/unistack-org/micro/v3/errors"
|
|
|
|
"github.com/unistack-org/micro/v3/proxy"
|
|
|
|
"github.com/unistack-org/micro/v3/server"
|
2019-06-03 18:44:43 +01:00
|
|
|
)
|
|
|
|
|
2019-06-07 13:42:39 +01:00
|
|
|
// Proxy will proxy rpc requests as http POST requests. It is a server.Proxy
|
|
|
|
type Proxy struct {
|
2019-12-16 14:55:47 +00:00
|
|
|
options proxy.Options
|
2019-06-07 13:42:39 +01:00
|
|
|
|
2019-06-03 18:44:43 +01:00
|
|
|
// The http backend to call
|
2019-06-07 13:42:39 +01:00
|
|
|
Endpoint string
|
2019-06-03 18:44:43 +01:00
|
|
|
|
|
|
|
// first request
|
|
|
|
first bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func getMethod(hdr map[string]string) string {
|
|
|
|
switch hdr["Micro-Method"] {
|
|
|
|
case "GET", "HEAD", "POST", "PUT", "DELETE", "CONNECT", "OPTIONS", "TRACE", "PATCH":
|
|
|
|
return hdr["Micro-Method"]
|
|
|
|
default:
|
|
|
|
return "POST"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getEndpoint(hdr map[string]string) string {
|
|
|
|
ep := hdr["Micro-Endpoint"]
|
|
|
|
if len(ep) > 0 && ep[0] == '/' {
|
|
|
|
return ep
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2019-11-25 16:31:43 +00:00
|
|
|
func getTopic(hdr map[string]string) string {
|
|
|
|
ep := hdr["Micro-Topic"]
|
|
|
|
if len(ep) > 0 && ep[0] == '/' {
|
|
|
|
return ep
|
|
|
|
}
|
|
|
|
return "/" + hdr["Micro-Topic"]
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProcessMessage handles incoming asynchronous messages
|
|
|
|
func (p *Proxy) ProcessMessage(ctx context.Context, msg server.Message) error {
|
|
|
|
if p.Endpoint == "" {
|
|
|
|
p.Endpoint = proxy.DefaultEndpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
// get the header
|
|
|
|
hdr := msg.Header()
|
|
|
|
|
|
|
|
// get topic
|
|
|
|
// use /topic as endpoint
|
|
|
|
endpoint := getTopic(hdr)
|
|
|
|
|
|
|
|
// set the endpoint
|
|
|
|
if len(endpoint) == 0 {
|
|
|
|
endpoint = p.Endpoint
|
|
|
|
} else {
|
|
|
|
// add endpoint to backend
|
|
|
|
u, err := url.Parse(p.Endpoint)
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(msg.Topic(), err.Error())
|
|
|
|
}
|
|
|
|
u.Path = path.Join(u.Path, endpoint)
|
|
|
|
endpoint = u.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
// send to backend
|
|
|
|
hreq, err := http.NewRequest("POST", endpoint, bytes.NewReader(msg.Body()))
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(msg.Topic(), err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// set the headers
|
|
|
|
for k, v := range hdr {
|
|
|
|
hreq.Header.Set(k, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
// make the call
|
|
|
|
hrsp, err := http.DefaultClient.Do(hreq)
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(msg.Topic(), err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// read body
|
|
|
|
b, err := ioutil.ReadAll(hrsp.Body)
|
|
|
|
hrsp.Body.Close()
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(msg.Topic(), err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
if hrsp.StatusCode != 200 {
|
|
|
|
return errors.New(msg.Topic(), string(b), int32(hrsp.StatusCode))
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
2019-08-23 14:05:11 +01:00
|
|
|
}
|
|
|
|
|
2019-06-03 18:44:43 +01:00
|
|
|
// ServeRequest honours the server.Router interface
|
2019-06-07 13:42:39 +01:00
|
|
|
func (p *Proxy) ServeRequest(ctx context.Context, req server.Request, rsp server.Response) error {
|
|
|
|
if p.Endpoint == "" {
|
|
|
|
p.Endpoint = proxy.DefaultEndpoint
|
2019-06-03 18:44:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
// get data
|
|
|
|
body, err := req.Read()
|
|
|
|
if err == io.EOF {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// get the header
|
|
|
|
hdr := req.Header()
|
|
|
|
|
|
|
|
// get method
|
|
|
|
method := getMethod(hdr)
|
|
|
|
|
|
|
|
// get endpoint
|
|
|
|
endpoint := getEndpoint(hdr)
|
|
|
|
|
|
|
|
// set the endpoint
|
|
|
|
if len(endpoint) == 0 {
|
2019-06-07 13:42:39 +01:00
|
|
|
endpoint = p.Endpoint
|
2019-06-03 18:44:43 +01:00
|
|
|
} else {
|
|
|
|
// add endpoint to backend
|
2019-06-07 13:42:39 +01:00
|
|
|
u, err := url.Parse(p.Endpoint)
|
2019-06-03 18:44:43 +01:00
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(req.Service(), err.Error())
|
|
|
|
}
|
|
|
|
u.Path = path.Join(u.Path, endpoint)
|
|
|
|
endpoint = u.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
// send to backend
|
|
|
|
hreq, err := http.NewRequest(method, endpoint, bytes.NewReader(body))
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(req.Service(), err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// set the headers
|
|
|
|
for k, v := range hdr {
|
|
|
|
hreq.Header.Set(k, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
// make the call
|
|
|
|
hrsp, err := http.DefaultClient.Do(hreq)
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(req.Service(), err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// read body
|
|
|
|
b, err := ioutil.ReadAll(hrsp.Body)
|
|
|
|
hrsp.Body.Close()
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(req.Service(), err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// set response headers
|
|
|
|
hdr = map[string]string{}
|
2019-11-01 15:07:53 +00:00
|
|
|
for k := range hrsp.Header {
|
2019-06-03 18:44:43 +01:00
|
|
|
hdr[k] = hrsp.Header.Get(k)
|
|
|
|
}
|
|
|
|
// write the header
|
|
|
|
rsp.WriteHeader(hdr)
|
|
|
|
// write the body
|
|
|
|
err = rsp.Write(b)
|
|
|
|
if err == io.EOF {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return errors.InternalServerError(req.Service(), err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-16 15:18:20 +00:00
|
|
|
func (p *Proxy) String() string {
|
2019-12-16 17:36:47 +00:00
|
|
|
return "http"
|
2019-12-16 15:18:20 +00:00
|
|
|
}
|
|
|
|
|
2019-06-07 13:42:39 +01:00
|
|
|
// NewSingleHostProxy returns a router which sends requests to a single http backend
|
|
|
|
func NewSingleHostProxy(url string) proxy.Proxy {
|
|
|
|
return &Proxy{
|
|
|
|
Endpoint: url,
|
2019-06-03 18:44:43 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-07 13:42:39 +01:00
|
|
|
// NewProxy returns a new proxy which will route using a http client
|
2019-12-16 14:55:47 +00:00
|
|
|
func NewProxy(opts ...proxy.Option) proxy.Proxy {
|
|
|
|
var options proxy.Options
|
|
|
|
for _, o := range opts {
|
|
|
|
o(&options)
|
2019-06-07 13:42:39 +01:00
|
|
|
}
|
|
|
|
|
2019-12-16 14:55:47 +00:00
|
|
|
p := new(Proxy)
|
|
|
|
p.Endpoint = options.Endpoint
|
|
|
|
p.options = options
|
|
|
|
|
2019-06-07 13:42:39 +01:00
|
|
|
return p
|
2019-06-03 18:44:43 +01:00
|
|
|
}
|