micro-client-http/stream.go

146 lines
2.8 KiB
Go
Raw Normal View History

2017-01-01 21:39:05 +03:00
package http
import (
"bufio"
2018-03-03 15:28:44 +03:00
"context"
"fmt"
"io"
2017-01-01 21:39:05 +03:00
"net"
"net/http"
"sync"
"go.unistack.org/micro/v3/client"
"go.unistack.org/micro/v3/codec"
"go.unistack.org/micro/v3/errors"
2017-01-01 21:39:05 +03:00
)
// Implements the streamer interface
type httpStream struct {
err error
conn net.Conn
cf codec.Codec
2017-01-01 21:39:05 +03:00
context context.Context
request client.Request
2017-01-01 21:39:05 +03:00
closed chan bool
reader *bufio.Reader
address string
ct string
opts client.CallOptions
sync.RWMutex
2017-01-01 21:39:05 +03:00
}
var errShutdown = fmt.Errorf("connection is shut down")
2017-01-01 21:39:05 +03:00
func (h *httpStream) isClosed() bool {
select {
case <-h.closed:
return true
default:
return false
}
}
func (h *httpStream) Context() context.Context {
return h.context
}
func (h *httpStream) Request() client.Request {
return h.request
}
2019-01-15 00:32:12 +03:00
func (h *httpStream) Response() client.Response {
return nil
}
2017-01-01 21:39:05 +03:00
func (h *httpStream) Send(msg interface{}) error {
h.Lock()
defer h.Unlock()
if h.isClosed() {
h.err = errShutdown
return errShutdown
}
hreq, err := newRequest(h.context, h.address, h.request, h.ct, h.cf, msg, h.opts)
2017-01-01 21:39:05 +03:00
if err != nil {
return err
}
return hreq.Write(h.conn)
2017-01-01 21:39:05 +03:00
}
func (h *httpStream) Recv(msg interface{}) error {
h.Lock()
defer h.Unlock()
if h.isClosed() {
h.err = errShutdown
return errShutdown
}
hrsp, err := http.ReadResponse(h.reader, new(http.Request))
2017-01-01 21:39:05 +03:00
if err != nil {
return errors.InternalServerError("go.micro.client", err.Error())
2017-01-01 21:39:05 +03:00
}
defer hrsp.Body.Close()
2017-01-01 21:39:05 +03:00
return h.parseRsp(h.context, hrsp, h.cf, msg, h.opts)
2017-01-01 21:39:05 +03:00
}
func (h *httpStream) Error() error {
h.RLock()
defer h.RUnlock()
return h.err
}
func (h *httpStream) Close() error {
select {
case <-h.closed:
return nil
default:
close(h.closed)
return h.conn.Close()
}
}
func (h *httpStream) parseRsp(ctx context.Context, hrsp *http.Response, cf codec.Codec, rsp interface{}, opts client.CallOptions) error {
var err error
select {
case <-ctx.Done():
err = ctx.Err()
default:
// fast path return
if hrsp.StatusCode == http.StatusNoContent {
return nil
}
if hrsp.StatusCode < 400 {
if err = cf.ReadBody(hrsp.Body, rsp); err != nil {
return errors.InternalServerError("go.micro.client", err.Error())
}
return nil
}
errmap, ok := opts.Context.Value(errorMapKey{}).(map[string]interface{})
if ok && errmap != nil {
if err, ok = errmap[fmt.Sprintf("%d", hrsp.StatusCode)].(error); !ok {
err, ok = errmap["default"].(error)
}
}
if !ok || err == nil {
buf, cerr := io.ReadAll(hrsp.Body)
if cerr != nil {
return errors.InternalServerError("go.micro.client", cerr.Error())
}
return errors.New("go.micro.client", string(buf), int32(hrsp.StatusCode))
}
if cerr := cf.ReadBody(hrsp.Body, err); cerr != nil {
err = errors.InternalServerError("go.micro.client", cerr.Error())
}
}
return err
}