micro/proxy/mucp/mucp.go

676 lines
16 KiB
Go
Raw Normal View History

2019-06-03 18:44:43 +01:00
// Package mucp transparently forwards the incoming request using a go-micro client.
package mucp
import (
"context"
2019-07-24 19:03:13 +01:00
"fmt"
2019-06-03 18:44:43 +01:00
"io"
2019-08-23 14:05:11 +01:00
"sort"
2019-06-03 18:44:43 +01:00
"strings"
2019-06-27 12:56:52 +01:00
"sync"
2019-09-20 16:25:29 +01:00
"time"
2019-06-03 18:44:43 +01:00
"github.com/unistack-org/micro/v3/client"
grpcc "github.com/unistack-org/micro/v3/client/grpc"
"github.com/unistack-org/micro/v3/codec"
"github.com/unistack-org/micro/v3/codec/bytes"
"github.com/unistack-org/micro/v3/errors"
"github.com/unistack-org/micro/v3/logger"
"github.com/unistack-org/micro/v3/metadata"
"github.com/unistack-org/micro/v3/proxy"
"github.com/unistack-org/micro/v3/router"
"github.com/unistack-org/micro/v3/router/registry"
"github.com/unistack-org/micro/v3/selector"
"github.com/unistack-org/micro/v3/selector/roundrobin"
"github.com/unistack-org/micro/v3/server"
"google.golang.org/grpc"
2019-06-03 18:44:43 +01:00
)
// Proxy will transparently proxy requests to an endpoint.
// If no endpoint is specified it will call a service using the client.
type Proxy struct {
// embed options
2019-12-16 14:55:47 +00:00
options proxy.Options
// Endpoint specifies the fixed service endpoint to call.
2019-06-03 18:44:43 +01:00
Endpoint string
2019-08-23 14:05:11 +01:00
// The client to use for outbound requests in the local network
2019-06-03 18:44:43 +01:00
Client client.Client
2019-06-27 12:56:52 +01:00
2019-08-23 14:05:11 +01:00
// Links are used for outbound requests not in the local network
Links map[string]client.Client
2019-06-27 12:56:52 +01:00
// The router for routes
Router router.Router
// A fib of routes service:address
sync.RWMutex
Routes map[string]map[uint64]router.Route
2020-08-05 17:38:41 +01:00
// selector used for load balancing
Selector selector.Selector
2019-06-03 18:44:43 +01:00
}
// read client request and write to server
func readLoop(r server.Request, s client.Stream) error {
// request to backend server
req := s.Request()
for {
// get data from client
// no need to decode it
body, err := r.Read()
if err == io.EOF {
return nil
}
2019-06-18 18:51:52 +01:00
2019-06-03 18:44:43 +01:00
if err != nil {
return err
}
// get the header from client
hdr := r.Header()
msg := &codec.Message{
Type: codec.Request,
Header: hdr,
Body: body,
}
2019-06-18 18:51:52 +01:00
2019-06-03 18:44:43 +01:00
// write the raw request
err = req.Codec().Write(msg, nil)
if err == io.EOF {
return nil
} else if err != nil {
return err
}
}
}
// toNodes returns a list of node addresses from given routes
2019-08-23 14:05:11 +01:00
func toNodes(routes []router.Route) []string {
nodes := make([]string, 0, len(routes))
for _, node := range routes {
address := node.Address
if len(node.Gateway) > 0 {
address = node.Gateway
2019-06-27 12:56:52 +01:00
}
nodes = append(nodes, address)
2019-06-27 12:56:52 +01:00
}
return nodes
}
2019-06-27 12:56:52 +01:00
2019-10-25 22:46:43 +01:00
func toSlice(r map[uint64]router.Route) []router.Route {
routes := make([]router.Route, 0, len(r))
2019-10-25 22:46:43 +01:00
for _, v := range r {
routes = append(routes, v)
}
// sort the routes in order of metric
sort.Slice(routes, func(i, j int) bool { return routes[i].Metric < routes[j].Metric })
return routes
}
func (p *Proxy) filterRoutes(ctx context.Context, routes []router.Route) []router.Route {
md, ok := metadata.FromContext(ctx)
if !ok {
return routes
}
//nolint:prealloc
var filteredRoutes []router.Route
// filter the routes based on our headers
for _, route := range routes {
// process only routes for this id
2020-03-26 18:50:00 +00:00
if id, ok := md.Get("Micro-Router"); ok && len(id) > 0 {
if route.Router != id {
// skip routes that don't match
continue
}
}
// only process routes with this network
if net, ok := md.Get("Micro-Namespace"); ok && len(net) > 0 {
if route.Network != router.DefaultNetwork && route.Network != net {
// skip routes that don't match
continue
}
}
// process only this gateway
2020-03-26 18:50:00 +00:00
if gw, ok := md.Get("Micro-Gateway"); ok && len(gw) > 0 {
// if the gateway matches our address
// special case, take the routes with no gateway
// TODO: should we strip the gateway from the context?
if gw == p.Router.Options().Address {
if len(route.Gateway) > 0 && route.Gateway != gw {
continue
}
// otherwise its a local route and we're keeping it
} else {
// gateway does not match our own
if route.Gateway != gw {
continue
}
}
}
// TODO: address based filtering
// address := md["Micro-Address"]
// TODO: label based filtering
// requires new field in routing table : route.Labels
// passed the filter checks
filteredRoutes = append(filteredRoutes, route)
}
if logger.V(logger.TraceLevel, logger.DefaultLogger) {
logger.Tracef("Proxy filtered routes %+v", filteredRoutes)
}
return filteredRoutes
}
2019-08-23 14:05:11 +01:00
func (p *Proxy) getLink(r router.Route) (client.Client, error) {
if r.Link == "local" || len(p.Links) == 0 {
return p.Client, nil
}
l, ok := p.Links[r.Link]
if !ok {
2019-08-23 14:09:57 +01:00
return nil, errors.InternalServerError("go.micro.proxy", "link not found")
2019-08-23 14:05:11 +01:00
}
return l, nil
}
func (p *Proxy) getRoute(ctx context.Context, service string) ([]router.Route, error) {
2019-06-27 12:56:52 +01:00
// lookup the route cache first
p.RLock()
2019-10-25 22:46:43 +01:00
cached, ok := p.Routes[service]
p.RUnlock()
2019-06-27 12:56:52 +01:00
if ok {
return p.filterRoutes(ctx, toSlice(cached)), nil
2019-06-27 12:56:52 +01:00
}
2019-10-25 22:46:43 +01:00
// cache routes for the service
routes, err := p.cacheRoutes(service)
if err != nil {
return nil, err
}
return p.filterRoutes(ctx, routes), nil
2019-10-25 22:46:43 +01:00
}
func (p *Proxy) cacheRoutes(service string) ([]router.Route, error) {
// lookup the routes in the router
results, err := p.Router.Lookup(router.QueryService(service), router.QueryNetwork("*"))
if err != nil {
// assumption that we're ok with stale routes
logger.Debugf("Failed to lookup route for %s: %v", service, err)
// otherwise return the error
return nil, err
2019-06-27 12:56:52 +01:00
}
// update the proxy cache
p.Lock()
// delete the existing reference to the service
delete(p.Routes, service)
for _, route := range results {
2019-08-23 14:05:11 +01:00
// create if does not exist
if _, ok := p.Routes[service]; !ok {
p.Routes[service] = make(map[uint64]router.Route)
}
// cache the route based on its unique hash
p.Routes[service][route.Hash()] = route
2019-06-27 12:56:52 +01:00
}
// make a copy of the service routes
2019-10-25 22:46:43 +01:00
routes := p.Routes[service]
p.Unlock()
2019-06-27 12:56:52 +01:00
// return routes to the caller
2019-08-23 14:05:11 +01:00
return toSlice(routes), nil
2019-06-27 12:56:52 +01:00
}
2019-10-25 22:46:43 +01:00
// refreshMetrics will refresh any metrics for our local cached routes.
// we may not receive new watch events for these as they change.
func (p *Proxy) refreshMetrics() {
// get a list of services to update
p.RLock()
services := make([]string, 0, len(p.Routes))
for service := range p.Routes {
2019-10-25 22:46:43 +01:00
services = append(services, service)
}
2019-10-25 22:46:43 +01:00
p.RUnlock()
// get and cache the routes for the service
for _, service := range services {
p.cacheRoutes(service)
}
}
// manageRoutes applies action on a given route to Proxy route cache
func (p *Proxy) manageRoutes(route router.Route, action string) error {
// we only cache what we are actually concerned with
p.Lock()
defer p.Unlock()
if logger.V(logger.TraceLevel, logger.DefaultLogger) {
logger.Tracef("Proxy taking route action %v %+v\n", action, route)
}
2019-07-24 19:03:13 +01:00
switch action {
case "create", "update":
if _, ok := p.Routes[route.Service]; !ok {
2019-10-25 22:46:43 +01:00
return fmt.Errorf("not called %s", route.Service)
2019-07-24 19:03:13 +01:00
}
p.Routes[route.Service][route.Hash()] = route
case "delete":
// delete that specific route
2019-07-24 19:03:13 +01:00
delete(p.Routes[route.Service], route.Hash())
// clean up the cache entirely
if len(p.Routes[route.Service]) == 0 {
delete(p.Routes, route.Service)
}
2019-07-24 19:03:13 +01:00
default:
return fmt.Errorf("unknown action: %s", action)
}
return nil
}
// watchRoutes watches service routes and updates proxy cache
func (p *Proxy) watchRoutes() {
// route watcher
w, err := p.Router.Watch()
if err != nil {
logger.Debugf("Error watching router: %v", err)
2019-07-24 19:03:13 +01:00
return
}
2020-01-19 23:15:57 +00:00
defer w.Stop()
2019-07-24 19:03:13 +01:00
for {
event, err := w.Next()
if err != nil {
logger.Debugf("Error watching router: %v", err)
2019-07-24 19:03:13 +01:00
return
}
2019-12-03 15:25:58 +08:00
if err := p.manageRoutes(event.Route, event.Type.String()); err != nil {
2019-07-24 19:03:13 +01:00
// TODO: should we bail here?
continue
}
}
}
// ProcessMessage acts as a message exchange and forwards messages to ongoing topics
// TODO: should we look at p.Endpoint and only send to the local endpoint? probably
func (p *Proxy) ProcessMessage(ctx context.Context, msg server.Message) error {
// TODO: check that we're not broadcast storming by sending to the same topic
// that we're actually subscribed to
if logger.V(logger.TraceLevel, logger.DefaultLogger) {
logger.Tracef("Proxy received message for %s", msg.Topic())
}
var errors []string
// directly publish to the local client
if err := p.Client.Publish(ctx, msg); err != nil {
errors = append(errors, err.Error())
}
// publish to all links
for _, client := range p.Links {
if err := client.Publish(ctx, msg); err != nil {
errors = append(errors, err.Error())
}
}
if len(errors) == 0 {
return nil
}
// there is no error...muahaha
return fmt.Errorf("Message processing error: %s", strings.Join(errors, "\n"))
2019-08-23 14:05:11 +01:00
}
2019-06-03 18:44:43 +01:00
// ServeRequest honours the server.Router interface
func (p *Proxy) ServeRequest(ctx context.Context, req server.Request, rsp server.Response) error {
2019-08-23 14:05:11 +01:00
// determine if its local routing
var local bool
// address to call
var addresses []string
// routes
var routes []router.Route
// service name to call
2019-06-03 18:44:43 +01:00
service := req.Service()
2019-08-23 14:05:11 +01:00
// endpoint to call
2019-06-03 18:44:43 +01:00
endpoint := req.Endpoint()
2019-08-23 14:05:11 +01:00
2019-08-29 13:10:06 +01:00
if len(service) == 0 {
return errors.BadRequest("go.micro.proxy", "service name is blank")
}
if logger.V(logger.TraceLevel, logger.DefaultLogger) {
logger.Tracef("Proxy received request for %s %s", service, endpoint)
}
2019-08-23 14:05:11 +01:00
// are we network routing or local routing
if len(p.Links) == 0 {
local = true
}
2019-06-03 18:44:43 +01:00
// if there is a proxy being used, e.g. micro network, we don't need to
// look up the routes since they'll be ignored by the client
if len(p.Client.Options().Proxy) > 0 {
return p.serveRequest(ctx, p.Client, service, endpoint, req, rsp)
}
2019-06-27 12:56:52 +01:00
// call a specific backend endpoint either by name or address
if len(p.Endpoint) > 0 {
2019-06-03 18:44:43 +01:00
// address:port
if parts := strings.Split(p.Endpoint, ":"); len(parts) > 1 {
2019-06-27 12:56:52 +01:00
addresses = []string{p.Endpoint}
2019-06-03 18:44:43 +01:00
} else {
2019-06-27 12:56:52 +01:00
// get route for endpoint from router
addr, err := p.getRoute(ctx, p.Endpoint)
2019-06-27 12:56:52 +01:00
if err != nil {
return err
}
// set the address
2019-08-23 14:05:11 +01:00
routes = addr
2019-06-27 12:56:52 +01:00
// set the name
service = p.Endpoint
2019-06-03 18:44:43 +01:00
}
2019-06-27 12:56:52 +01:00
} else {
// no endpoint was specified just lookup the route
// get route for endpoint from router
addr, err := p.getRoute(ctx, service)
2019-06-27 12:56:52 +01:00
if err != nil {
return err
}
2019-08-23 14:05:11 +01:00
routes = addr
2019-06-27 12:56:52 +01:00
}
//nolint:prealloc
opts := []client.CallOption{
// set strategy to round robin
2020-08-05 17:38:41 +01:00
client.WithSelector(p.Selector),
}
2019-08-23 14:05:11 +01:00
// if the address is already set just serve it
// TODO: figure it out if we should know to pick a link
2019-06-27 12:56:52 +01:00
if len(addresses) > 0 {
opts = append(opts,
client.WithAddress(addresses...),
)
2019-08-23 14:05:11 +01:00
// serve the normal way
return p.serveRequest(ctx, p.Client, service, endpoint, req, rsp, opts...)
2019-08-23 14:05:11 +01:00
}
// there's no links e.g we're local routing then just serve it with addresses
if local {
var opts []client.CallOption
// set address if available via routes or specific endpoint
if len(routes) > 0 {
2020-01-02 21:11:25 +00:00
addresses = toNodes(routes)
2019-08-23 14:05:11 +01:00
opts = append(opts, client.WithAddress(addresses...))
}
if logger.V(logger.TraceLevel, logger.DefaultLogger) {
logger.Tracef("Proxy calling %+v\n", addresses)
}
2019-08-23 14:05:11 +01:00
// serve the normal way
return p.serveRequest(ctx, p.Client, service, endpoint, req, rsp, opts...)
2019-06-03 18:44:43 +01:00
}
// we're assuming we need routes to operate on
if len(routes) == 0 {
return errors.InternalServerError("go.micro.proxy", "route not found")
}
2019-08-23 14:05:11 +01:00
var gerr error
// we're routing globally with multiple links
// so we need to pick a link per route
for _, route := range routes {
// pick the link or error out
link, err := p.getLink(route)
if err != nil {
// ok let's try again
gerr = err
continue
}
if logger.V(logger.TraceLevel, logger.DefaultLogger) {
logger.Tracef("Proxy using route %+v\n", route)
}
2019-08-23 14:05:11 +01:00
// set the address to call
addresses := toNodes([]router.Route{route})
// set the address in the options
// disable retries since its one route processing
opts = append(opts,
client.WithAddress(addresses...),
client.WithRetries(0),
)
2019-08-23 14:05:11 +01:00
// do the request with the link
gerr = p.serveRequest(ctx, link, service, endpoint, req, rsp, opts...)
2019-08-23 14:05:11 +01:00
// return on no error since we succeeded
if gerr == nil {
return nil
}
// return where the context deadline was exceeded
if gerr == context.Canceled || gerr == context.DeadlineExceeded {
return err
}
// otherwise attempt to do it all over again
}
// if we got here something went really badly wrong
return gerr
}
func (p *Proxy) serveRequest(ctx context.Context, link client.Client, service, endpoint string, req server.Request, rsp server.Response, opts ...client.CallOption) error {
2019-06-03 18:44:43 +01:00
// read initial request
body, err := req.Read()
if err != nil {
return err
}
// create new request with raw bytes body
2019-12-05 16:10:49 +00:00
creq := link.NewRequest(service, endpoint, &bytes.Frame{Data: body}, client.WithContentType(req.ContentType()))
2019-06-03 18:44:43 +01:00
// not a stream so make a client.Call request
2019-08-16 16:46:29 +01:00
if !req.Stream() {
crsp := new(bytes.Frame)
// make a call to the backend
2019-08-23 14:05:11 +01:00
if err := link.Call(ctx, creq, crsp, opts...); err != nil {
return err
}
// write the response
if err := rsp.Write(crsp.Data); err != nil {
return err
}
return nil
2019-08-16 16:46:29 +01:00
}
// new context with cancel
ctx, cancel := context.WithCancel(ctx)
2019-06-03 18:44:43 +01:00
// create new stream
2019-08-23 14:05:11 +01:00
stream, err := link.Stream(ctx, creq, opts...)
2019-06-03 18:44:43 +01:00
if err != nil {
return err
}
defer stream.Close()
2020-01-02 21:11:25 +00:00
// if we receive a grpc stream we have to refire the initial request
2020-01-03 19:46:14 +00:00
c, ok := req.Codec().(codec.Codec)
if ok && c.String() == "grpc" && link.String() == "grpc" {
2020-01-02 21:11:25 +00:00
// get the header from client
hdr := req.Header()
msg := &codec.Message{
Type: codec.Request,
Header: hdr,
Body: body,
}
// write the raw request
err = stream.Request().Codec().Write(msg, nil)
if err == io.EOF {
return nil
} else if err != nil {
return err
}
}
2019-08-16 16:46:29 +01:00
// create client request read loop if streaming
go func() {
err := readLoop(req, stream)
if err != nil && err != io.EOF {
// cancel the context
cancel()
}
}()
2019-06-03 18:44:43 +01:00
// get raw response
resp := stream.Response()
// create server response write loop
for {
2019-10-25 22:46:43 +01:00
// read backend response body
body, err := resp.Read()
if err != nil {
// when we're done if its a grpc stream we have to set the trailer
if cc, ok := stream.(grpc.ClientStream); ok {
if ss, ok := resp.Codec().(grpc.ServerStream); ok {
ss.SetTrailer(cc.Trailer())
}
}
}
2019-10-25 22:46:43 +01:00
if err == io.EOF {
return nil
} else if err != nil {
return err
}
2019-06-03 18:44:43 +01:00
2019-10-25 22:46:43 +01:00
// read backend response header
hdr := resp.Header()
2019-06-03 18:44:43 +01:00
2019-10-25 22:46:43 +01:00
// write raw response header to client
rsp.WriteHeader(hdr)
2019-06-03 18:44:43 +01:00
2019-10-25 22:46:43 +01:00
// write raw response body to client
err = rsp.Write(body)
if err == io.EOF {
return nil
} else if err != nil {
return err
2019-06-03 18:44:43 +01:00
}
}
}
2019-12-16 15:18:20 +00:00
func (p *Proxy) String() string {
return "mucp"
}
// NewSingleHostProxy returns a proxy which sends requests to a single backend
func NewSingleHostProxy(endpoint string) *Proxy {
return &Proxy{
Endpoint: endpoint,
2019-06-03 18:44:43 +01:00
}
}
// NewProxy returns a new proxy which will route based on mucp headers
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)
}
p := new(Proxy)
2019-08-23 14:05:11 +01:00
p.Links = map[string]client.Client{}
2019-12-16 14:55:47 +00:00
p.Routes = make(map[string]map[uint64]router.Route)
p.options = options
// get endpoint
2019-12-16 14:55:47 +00:00
p.Endpoint = options.Endpoint
// set the client
p.Client = options.Client
// get router
p.Router = options.Router
// set the default client
if p.Client == nil {
p.Client = grpcc.NewClient()
}
// create default router and start it
if p.Router == nil {
p.Router = registry.NewRouter()
}
2020-08-05 17:38:41 +01:00
if p.Selector == nil {
p.Selector = roundrobin.NewSelector()
}
2019-12-16 14:55:47 +00:00
// set the links
if options.Links != nil {
// get client
p.Links = options.Links
}
// TODO: remove this cruft
// skip watching routes if proxy is set
if len(p.Client.Options().Proxy) > 0 {
return p
}
2019-09-20 16:25:29 +01:00
go func() {
// continuously attempt to watch routes
for {
// watch the routes
p.watchRoutes()
// in case of failure just wait a second
time.Sleep(time.Second)
}
}()
2019-07-24 19:03:13 +01:00
2019-10-25 22:46:43 +01:00
go func() {
// TODO: speed up refreshing of metrics
// without this ticking effort e.g stream
t := time.NewTicker(time.Second * 10)
2019-10-25 22:46:43 +01:00
defer t.Stop()
// we must refresh route metrics since they do not trigger new events
2019-12-03 15:25:58 +08:00
for range t.C {
// refresh route metrics
p.refreshMetrics()
2019-10-25 22:46:43 +01:00
}
}()
return p
2019-06-03 18:44:43 +01:00
}