2018-09-21 09:17:43 -06:00
|
|
|
// Copyright 2018 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Package jsonrpc2 is a minimal implementation of the JSON RPC 2 spec.
|
|
|
|
// https://www.jsonrpc.org/specification
|
|
|
|
// It is intended to be compatible with other implementations at the wire level.
|
|
|
|
package jsonrpc2
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
2018-10-30 18:12:26 -06:00
|
|
|
"time"
|
2019-06-21 14:26:30 -06:00
|
|
|
|
|
|
|
"golang.org/x/tools/internal/lsp/telemetry"
|
|
|
|
"golang.org/x/tools/internal/lsp/telemetry/stats"
|
|
|
|
"golang.org/x/tools/internal/lsp/telemetry/tag"
|
|
|
|
"golang.org/x/tools/internal/lsp/telemetry/trace"
|
2018-09-21 09:17:43 -06:00
|
|
|
)
|
|
|
|
|
|
|
|
// Conn is a JSON RPC 2 client server connection.
|
|
|
|
// Conn is bidirectional; it does not have a designated server or client end.
|
|
|
|
type Conn struct {
|
2019-03-29 09:13:22 -06:00
|
|
|
seq int64 // must only be accessed using atomic operations
|
|
|
|
Handler Handler
|
|
|
|
Canceler Canceler
|
|
|
|
Logger Logger
|
|
|
|
Capacity int
|
|
|
|
RejectIfOverloaded bool
|
|
|
|
stream Stream
|
|
|
|
err error
|
|
|
|
pendingMu sync.Mutex // protects the pending map
|
2019-06-24 07:18:31 -06:00
|
|
|
pending map[ID]chan *wireResponse
|
2019-03-29 09:13:22 -06:00
|
|
|
handlingMu sync.Mutex // protects the handling map
|
2019-06-24 07:16:28 -06:00
|
|
|
handling map[ID]*Request
|
2019-03-29 09:13:22 -06:00
|
|
|
}
|
|
|
|
|
2019-06-24 07:16:28 -06:00
|
|
|
type requestState int
|
|
|
|
|
|
|
|
const (
|
|
|
|
requestWaiting = requestState(iota)
|
|
|
|
requestSerial
|
|
|
|
requestParallel
|
|
|
|
requestReplied
|
|
|
|
requestDone
|
|
|
|
)
|
|
|
|
|
2019-06-24 07:18:31 -06:00
|
|
|
// Request is sent to a server to represent a Call or Notify operaton.
|
|
|
|
type Request struct {
|
2019-06-24 07:16:28 -06:00
|
|
|
conn *Conn
|
|
|
|
cancel context.CancelFunc
|
|
|
|
start time.Time
|
|
|
|
state requestState
|
|
|
|
nextRequest chan struct{}
|
2019-06-24 07:18:31 -06:00
|
|
|
|
|
|
|
// Method is a string containing the method name to invoke.
|
|
|
|
Method string
|
|
|
|
// Params is either a struct or an array with the parameters of the method.
|
|
|
|
Params *json.RawMessage
|
|
|
|
// The id of this request, used to tie the response back to the request.
|
|
|
|
// Will be either a string or a number. If not set, the request is a notify,
|
|
|
|
// and no response is possible.
|
|
|
|
ID *ID
|
|
|
|
}
|
|
|
|
|
2018-09-21 09:17:43 -06:00
|
|
|
// Handler is an option you can pass to NewConn to handle incoming requests.
|
2018-11-13 20:49:07 -07:00
|
|
|
// If the request returns false from IsNotify then the Handler must eventually
|
|
|
|
// call Reply on the Conn with the supplied request.
|
|
|
|
// Handlers are called synchronously, they should pass the work off to a go
|
|
|
|
// routine if they are going to take a long time.
|
2019-06-24 07:18:31 -06:00
|
|
|
type Handler func(context.Context, *Request)
|
2018-09-21 09:17:43 -06:00
|
|
|
|
|
|
|
// Canceler is an option you can pass to NewConn which is invoked for
|
|
|
|
// cancelled outgoing requests.
|
|
|
|
// It is okay to use the connection to send notifications, but the context will
|
|
|
|
// be in the cancelled state, so you must do it with the background context
|
|
|
|
// instead.
|
2019-06-24 07:18:31 -06:00
|
|
|
type Canceler func(context.Context, *Conn, ID)
|
2018-09-21 09:17:43 -06:00
|
|
|
|
2019-06-21 14:26:30 -06:00
|
|
|
type rpcStats struct {
|
|
|
|
server bool
|
|
|
|
method string
|
|
|
|
span trace.Span
|
|
|
|
start time.Time
|
|
|
|
received int64
|
|
|
|
sent int64
|
|
|
|
}
|
|
|
|
|
|
|
|
type statsKeyType string
|
|
|
|
|
|
|
|
const rpcStatsKey = statsKeyType("rpcStatsKey")
|
|
|
|
|
|
|
|
func start(ctx context.Context, server bool, method string, id *ID) (context.Context, *rpcStats) {
|
2019-06-24 07:16:28 -06:00
|
|
|
if method == "" {
|
|
|
|
panic("no method in rpc stats")
|
|
|
|
}
|
2019-06-21 14:26:30 -06:00
|
|
|
s := &rpcStats{
|
|
|
|
server: server,
|
|
|
|
method: method,
|
|
|
|
start: time.Now(),
|
|
|
|
}
|
2019-06-24 07:16:28 -06:00
|
|
|
ctx = context.WithValue(ctx, rpcStatsKey, s)
|
2019-06-21 14:26:30 -06:00
|
|
|
tags := make([]tag.Mutator, 0, 4)
|
|
|
|
tags = append(tags, tag.Upsert(telemetry.KeyMethod, method))
|
|
|
|
mode := telemetry.Outbound
|
|
|
|
spanKind := trace.SpanKindClient
|
|
|
|
if server {
|
|
|
|
spanKind = trace.SpanKindServer
|
|
|
|
mode = telemetry.Inbound
|
|
|
|
}
|
|
|
|
tags = append(tags, tag.Upsert(telemetry.KeyRPCDirection, mode))
|
|
|
|
if id != nil {
|
|
|
|
tags = append(tags, tag.Upsert(telemetry.KeyRPCID, id.String()))
|
|
|
|
}
|
2019-06-24 07:16:28 -06:00
|
|
|
ctx, s.span = trace.StartSpan(ctx, method, trace.WithSpanKind(spanKind))
|
|
|
|
ctx, _ = tag.New(ctx, tags...)
|
|
|
|
stats.Record(ctx, telemetry.Started.M(1))
|
|
|
|
return ctx, s
|
2019-06-21 14:26:30 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *rpcStats) end(ctx context.Context, err *error) {
|
|
|
|
if err != nil && *err != nil {
|
|
|
|
ctx, _ = tag.New(ctx, tag.Upsert(telemetry.KeyStatus, "ERROR"))
|
|
|
|
} else {
|
|
|
|
ctx, _ = tag.New(ctx, tag.Upsert(telemetry.KeyStatus, "OK"))
|
|
|
|
}
|
|
|
|
elapsedTime := time.Since(s.start)
|
|
|
|
latencyMillis := float64(elapsedTime) / float64(time.Millisecond)
|
|
|
|
|
|
|
|
stats.Record(ctx,
|
|
|
|
telemetry.ReceivedBytes.M(s.received),
|
|
|
|
telemetry.SentBytes.M(s.sent),
|
|
|
|
telemetry.Latency.M(latencyMillis),
|
|
|
|
)
|
|
|
|
|
|
|
|
s.span.End()
|
|
|
|
}
|
|
|
|
|
2018-09-21 09:17:43 -06:00
|
|
|
// NewErrorf builds a Error struct for the suppied message and code.
|
|
|
|
// If args is not empty, message and args will be passed to Sprintf.
|
|
|
|
func NewErrorf(code int64, format string, args ...interface{}) *Error {
|
|
|
|
return &Error{
|
|
|
|
Code: code,
|
|
|
|
Message: fmt.Sprintf(format, args...),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-28 19:06:01 -06:00
|
|
|
// NewConn creates a new connection object around the supplied stream.
|
|
|
|
// You must call Run for the connection to be active.
|
|
|
|
func NewConn(s Stream) *Conn {
|
2018-09-21 09:17:43 -06:00
|
|
|
conn := &Conn{
|
|
|
|
stream: s,
|
2019-06-24 07:18:31 -06:00
|
|
|
pending: make(map[ID]chan *wireResponse),
|
2019-06-24 07:16:28 -06:00
|
|
|
handling: make(map[ID]*Request),
|
2018-09-21 09:17:43 -06:00
|
|
|
}
|
2019-03-28 19:06:01 -06:00
|
|
|
// the default handler reports a method error
|
2019-06-24 07:18:31 -06:00
|
|
|
conn.Handler = func(ctx context.Context, r *Request) {
|
2019-06-24 07:16:28 -06:00
|
|
|
if !r.IsNotify() {
|
2019-06-24 07:18:31 -06:00
|
|
|
r.Reply(ctx, nil, NewErrorf(CodeMethodNotFound, "method %q not found", r.Method))
|
2018-09-21 09:17:43 -06:00
|
|
|
}
|
|
|
|
}
|
2019-06-24 07:18:31 -06:00
|
|
|
// the default canceler does nothing
|
|
|
|
conn.Canceler = func(context.Context, *Conn, ID) {}
|
2019-03-28 19:06:01 -06:00
|
|
|
// the default logger does nothing
|
|
|
|
conn.Logger = func(Direction, *ID, time.Duration, string, *json.RawMessage, *Error) {}
|
2018-09-21 09:17:43 -06:00
|
|
|
return conn
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cancel cancels a pending Call on the server side.
|
|
|
|
// The call is identified by its id.
|
|
|
|
// JSON RPC 2 does not specify a cancel message, so cancellation support is not
|
|
|
|
// directly wired in. This method allows a higher level protocol to choose how
|
|
|
|
// to propagate the cancel.
|
|
|
|
func (c *Conn) Cancel(id ID) {
|
|
|
|
c.handlingMu.Lock()
|
2018-11-13 20:49:07 -07:00
|
|
|
handling, found := c.handling[id]
|
2018-09-21 09:17:43 -06:00
|
|
|
c.handlingMu.Unlock()
|
2018-11-13 20:49:07 -07:00
|
|
|
if found {
|
|
|
|
handling.cancel()
|
2018-09-21 09:17:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notify is called to send a notification request over the connection.
|
|
|
|
// It will return as soon as the notification has been sent, as no response is
|
|
|
|
// possible.
|
2019-06-21 14:26:30 -06:00
|
|
|
func (c *Conn) Notify(ctx context.Context, method string, params interface{}) (err error) {
|
|
|
|
ctx, rpcStats := start(ctx, false, method, nil)
|
|
|
|
defer rpcStats.end(ctx, &err)
|
|
|
|
|
2018-09-21 09:17:43 -06:00
|
|
|
jsonParams, err := marshalToRaw(params)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshalling notify parameters: %v", err)
|
|
|
|
}
|
2019-06-24 07:18:31 -06:00
|
|
|
request := &wireRequest{
|
2018-09-21 09:17:43 -06:00
|
|
|
Method: method,
|
|
|
|
Params: jsonParams,
|
|
|
|
}
|
|
|
|
data, err := json.Marshal(request)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshalling notify request: %v", err)
|
|
|
|
}
|
2019-03-28 19:06:01 -06:00
|
|
|
c.Logger(Send, nil, -1, request.Method, request.Params, nil)
|
2019-06-21 14:26:30 -06:00
|
|
|
n, err := c.stream.Write(ctx, data)
|
|
|
|
rpcStats.sent += n
|
|
|
|
return err
|
2018-09-21 09:17:43 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Call sends a request over the connection and then waits for a response.
|
|
|
|
// If the response is not an error, it will be decoded into result.
|
|
|
|
// result must be of a type you an pass to json.Unmarshal.
|
2019-06-21 14:26:30 -06:00
|
|
|
func (c *Conn) Call(ctx context.Context, method string, params, result interface{}) (err error) {
|
|
|
|
// generate a new request identifier
|
|
|
|
id := ID{Number: atomic.AddInt64(&c.seq, 1)}
|
|
|
|
ctx, rpcStats := start(ctx, false, method, &id)
|
|
|
|
defer rpcStats.end(ctx, &err)
|
2018-09-21 09:17:43 -06:00
|
|
|
jsonParams, err := marshalToRaw(params)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshalling call parameters: %v", err)
|
|
|
|
}
|
2019-06-24 07:18:31 -06:00
|
|
|
request := &wireRequest{
|
2018-09-21 09:17:43 -06:00
|
|
|
ID: &id,
|
|
|
|
Method: method,
|
|
|
|
Params: jsonParams,
|
|
|
|
}
|
|
|
|
// marshal the request now it is complete
|
|
|
|
data, err := json.Marshal(request)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshalling call request: %v", err)
|
|
|
|
}
|
|
|
|
// we have to add ourselves to the pending map before we send, otherwise we
|
|
|
|
// are racing the response
|
2019-06-24 07:18:31 -06:00
|
|
|
rchan := make(chan *wireResponse)
|
2018-09-21 09:17:43 -06:00
|
|
|
c.pendingMu.Lock()
|
|
|
|
c.pending[id] = rchan
|
|
|
|
c.pendingMu.Unlock()
|
|
|
|
defer func() {
|
|
|
|
// clean up the pending response handler on the way out
|
|
|
|
c.pendingMu.Lock()
|
|
|
|
delete(c.pending, id)
|
|
|
|
c.pendingMu.Unlock()
|
|
|
|
}()
|
|
|
|
// now we are ready to send
|
2018-10-30 18:12:26 -06:00
|
|
|
before := time.Now()
|
2019-03-28 19:06:01 -06:00
|
|
|
c.Logger(Send, request.ID, -1, request.Method, request.Params, nil)
|
2019-06-21 14:26:30 -06:00
|
|
|
n, err := c.stream.Write(ctx, data)
|
|
|
|
rpcStats.sent += n
|
|
|
|
if err != nil {
|
2018-09-21 09:17:43 -06:00
|
|
|
// sending failed, we will never get a response, so don't leave it pending
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// now wait for the response
|
|
|
|
select {
|
|
|
|
case response := <-rchan:
|
2018-10-30 18:12:26 -06:00
|
|
|
elapsed := time.Since(before)
|
2019-04-06 10:22:29 -06:00
|
|
|
c.Logger(Receive, response.ID, elapsed, request.Method, response.Result, response.Error)
|
2018-09-21 09:17:43 -06:00
|
|
|
// is it an error response?
|
|
|
|
if response.Error != nil {
|
|
|
|
return response.Error
|
|
|
|
}
|
|
|
|
if result == nil || response.Result == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err := json.Unmarshal(*response.Result, result); err != nil {
|
|
|
|
return fmt.Errorf("unmarshalling result: %v", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
case <-ctx.Done():
|
|
|
|
// allow the handler to propagate the cancel
|
2019-06-24 07:18:31 -06:00
|
|
|
c.Canceler(ctx, c, id)
|
2018-09-21 09:17:43 -06:00
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-24 07:18:31 -06:00
|
|
|
// Conn returns the connection that created this request.
|
|
|
|
func (r *Request) Conn() *Conn { return r.conn }
|
|
|
|
|
|
|
|
// IsNotify returns true if this request is a notification.
|
|
|
|
func (r *Request) IsNotify() bool {
|
|
|
|
return r.ID == nil
|
|
|
|
}
|
|
|
|
|
2019-06-24 07:16:28 -06:00
|
|
|
// Parallel indicates that the system is now allowed to process other requests
|
|
|
|
// in parallel with this one.
|
|
|
|
// It is safe to call any number of times, but must only be called from the
|
|
|
|
// request handling go routine.
|
|
|
|
// It is implied by both reply and by the handler returning.
|
|
|
|
func (r *Request) Parallel() {
|
|
|
|
if r.state >= requestParallel {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
r.state = requestParallel
|
|
|
|
close(r.nextRequest)
|
|
|
|
}
|
|
|
|
|
2018-11-13 20:49:07 -07:00
|
|
|
// Reply sends a reply to the given request.
|
|
|
|
// It is an error to call this if request was not a call.
|
|
|
|
// You must call this exactly once for any given request.
|
2019-06-24 07:16:28 -06:00
|
|
|
// It should only be called from the handler go routine.
|
2018-11-13 20:49:07 -07:00
|
|
|
// If err is set then result will be ignored.
|
2019-06-24 07:18:31 -06:00
|
|
|
func (r *Request) Reply(ctx context.Context, result interface{}, err error) error {
|
2019-06-24 07:16:28 -06:00
|
|
|
if r.state >= requestReplied {
|
|
|
|
return fmt.Errorf("reply invoked more than once")
|
|
|
|
}
|
2019-06-24 07:18:31 -06:00
|
|
|
if r.IsNotify() {
|
2018-11-13 20:49:07 -07:00
|
|
|
return fmt.Errorf("reply not invoked with a valid call")
|
|
|
|
}
|
2019-06-24 07:16:28 -06:00
|
|
|
ctx, st := trace.StartSpan(ctx, r.Method+":reply", trace.WithSpanKind(trace.SpanKindClient))
|
|
|
|
defer st.End()
|
|
|
|
|
|
|
|
r.Parallel()
|
|
|
|
r.state = requestReplied
|
2018-11-13 20:49:07 -07:00
|
|
|
|
2019-06-24 07:16:28 -06:00
|
|
|
elapsed := time.Since(r.start)
|
2018-11-13 20:49:07 -07:00
|
|
|
var raw *json.RawMessage
|
|
|
|
if err == nil {
|
|
|
|
raw, err = marshalToRaw(result)
|
|
|
|
}
|
2019-06-24 07:18:31 -06:00
|
|
|
response := &wireResponse{
|
2018-11-13 20:49:07 -07:00
|
|
|
Result: raw,
|
2019-06-24 07:18:31 -06:00
|
|
|
ID: r.ID,
|
2018-11-13 20:49:07 -07:00
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
if callErr, ok := err.(*Error); ok {
|
|
|
|
response.Error = callErr
|
|
|
|
} else {
|
|
|
|
response.Error = NewErrorf(0, "%s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
data, err := json.Marshal(response)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-06-24 07:18:31 -06:00
|
|
|
r.conn.Logger(Send, response.ID, elapsed, r.Method, response.Result, response.Error)
|
|
|
|
n, err := r.conn.stream.Write(ctx, data)
|
2019-06-21 14:26:30 -06:00
|
|
|
|
|
|
|
v := ctx.Value(rpcStatsKey)
|
|
|
|
if v != nil {
|
2019-06-24 07:16:28 -06:00
|
|
|
v.(*rpcStats).sent += n
|
2019-06-21 14:26:30 -06:00
|
|
|
} else {
|
2019-06-24 07:16:28 -06:00
|
|
|
panic("no stats available in reply")
|
2019-06-21 14:26:30 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2018-11-13 20:49:07 -07:00
|
|
|
// TODO(iancottrell): if a stream write fails, we really need to shut down
|
|
|
|
// the whole stream
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-06-24 07:16:28 -06:00
|
|
|
func (c *Conn) setHandling(r *Request, active bool) {
|
|
|
|
if r.ID == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
r.conn.handlingMu.Lock()
|
|
|
|
defer r.conn.handlingMu.Unlock()
|
|
|
|
if active {
|
|
|
|
r.conn.handling[*r.ID] = r
|
|
|
|
} else {
|
|
|
|
delete(r.conn.handling, *r.ID)
|
|
|
|
}
|
2018-11-13 20:49:07 -07:00
|
|
|
}
|
|
|
|
|
2018-09-21 09:17:43 -06:00
|
|
|
// combined has all the fields of both Request and Response.
|
|
|
|
// We can decode this and then work out which it is.
|
|
|
|
type combined struct {
|
|
|
|
VersionTag VersionTag `json:"jsonrpc"`
|
|
|
|
ID *ID `json:"id,omitempty"`
|
|
|
|
Method string `json:"method"`
|
|
|
|
Params *json.RawMessage `json:"params,omitempty"`
|
|
|
|
Result *json.RawMessage `json:"result,omitempty"`
|
|
|
|
Error *Error `json:"error,omitempty"`
|
|
|
|
}
|
|
|
|
|
2019-03-28 19:06:01 -06:00
|
|
|
// Run blocks until the connection is terminated, and returns any error that
|
|
|
|
// caused the termination.
|
2018-09-21 09:17:43 -06:00
|
|
|
// It must be called exactly once for each Conn.
|
|
|
|
// It returns only when the reader is closed or there is an error in the stream.
|
2019-03-28 19:06:01 -06:00
|
|
|
func (c *Conn) Run(ctx context.Context) error {
|
2019-06-24 07:16:28 -06:00
|
|
|
nextRequest := make(chan struct{})
|
|
|
|
close(nextRequest)
|
2018-09-21 09:17:43 -06:00
|
|
|
for {
|
|
|
|
// get the data for a message
|
2019-06-21 14:26:30 -06:00
|
|
|
data, n, err := c.stream.Read(ctx)
|
2018-09-21 09:17:43 -06:00
|
|
|
if err != nil {
|
|
|
|
// the stream failed, we cannot continue
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// read a combined message
|
|
|
|
msg := &combined{}
|
|
|
|
if err := json.Unmarshal(data, msg); err != nil {
|
|
|
|
// a badly formed message arrived, log it and continue
|
|
|
|
// we trust the stream to have isolated the error to just this message
|
2019-03-28 19:06:01 -06:00
|
|
|
c.Logger(Receive, nil, -1, "", nil, NewErrorf(0, "unmarshal failed: %v", err))
|
2018-09-21 09:17:43 -06:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
// work out which kind of message we have
|
|
|
|
switch {
|
|
|
|
case msg.Method != "":
|
|
|
|
// if method is set it must be a request
|
2019-06-24 07:16:28 -06:00
|
|
|
reqCtx, cancelReq := context.WithCancel(ctx)
|
|
|
|
reqCtx, rpcStats := start(reqCtx, true, msg.Method, msg.ID)
|
|
|
|
rpcStats.received += n
|
|
|
|
thisRequest := nextRequest
|
|
|
|
nextRequest = make(chan struct{})
|
|
|
|
req := &Request{
|
|
|
|
conn: c,
|
|
|
|
cancel: cancelReq,
|
|
|
|
nextRequest: nextRequest,
|
|
|
|
start: time.Now(),
|
|
|
|
Method: msg.Method,
|
|
|
|
Params: msg.Params,
|
|
|
|
ID: msg.ID,
|
2018-09-21 09:17:43 -06:00
|
|
|
}
|
2019-06-24 07:16:28 -06:00
|
|
|
c.setHandling(req, true)
|
|
|
|
go func() {
|
|
|
|
<-thisRequest
|
|
|
|
req.state = requestSerial
|
|
|
|
defer func() {
|
|
|
|
c.setHandling(req, false)
|
|
|
|
if !req.IsNotify() && req.state < requestReplied {
|
|
|
|
req.Reply(reqCtx, nil, NewErrorf(CodeInternalError, "method %q did not reply", req.Method))
|
|
|
|
}
|
|
|
|
req.Parallel()
|
|
|
|
rpcStats.end(reqCtx, nil)
|
|
|
|
cancelReq()
|
|
|
|
}()
|
|
|
|
c.Logger(Receive, req.ID, -1, req.Method, req.Params, nil)
|
|
|
|
c.Handler(reqCtx, req)
|
|
|
|
}()
|
2018-09-21 09:17:43 -06:00
|
|
|
case msg.ID != nil:
|
|
|
|
// we have a response, get the pending entry from the map
|
|
|
|
c.pendingMu.Lock()
|
|
|
|
rchan := c.pending[*msg.ID]
|
|
|
|
if rchan != nil {
|
|
|
|
delete(c.pending, *msg.ID)
|
|
|
|
}
|
|
|
|
c.pendingMu.Unlock()
|
|
|
|
// and send the reply to the channel
|
2019-06-24 07:18:31 -06:00
|
|
|
response := &wireResponse{
|
2018-09-21 09:17:43 -06:00
|
|
|
Result: msg.Result,
|
|
|
|
Error: msg.Error,
|
|
|
|
ID: msg.ID,
|
|
|
|
}
|
|
|
|
rchan <- response
|
|
|
|
close(rchan)
|
|
|
|
default:
|
2019-03-28 19:06:01 -06:00
|
|
|
c.Logger(Receive, nil, -1, "", nil, NewErrorf(0, "message not a call, notify or response, ignoring"))
|
2018-09-21 09:17:43 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func marshalToRaw(obj interface{}) (*json.RawMessage, error) {
|
|
|
|
data, err := json.Marshal(obj)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
raw := json.RawMessage(data)
|
|
|
|
return &raw, nil
|
|
|
|
}
|