mirror of
https://github.com/golang/go
synced 2024-11-18 14:14:46 -07:00
434f7a8fef
The previous implementation was exposing the details of the wire format and resulted in non idomatic go, detecting the presence of absence of values in fields to deterimine the message type. Now the messages are distinct types and we use type switches instead. Request still exists as an interface to expose the shared behaviour of Call and Notification, as this is the type accepted by handlers. The set of messages is deliberately closed by using a private methods on the interfaces. Change-Id: I2cf15ee3923ef4688670c62896f81f760c77fe04 Reviewed-on: https://go-review.googlesource.com/c/tools/+/228719 Run-TryBot: Ian Cottrell <iancottrell@google.com> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Robert Findley <rfindley@google.com>
228 lines
6.4 KiB
Go
228 lines
6.4 KiB
Go
// 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
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
)
|
|
|
|
// Message is the interface to all jsonrpc2 message types.
|
|
// They share no common functionality, but are a closed set of concrete types
|
|
// that are allowed to implement this interface. The message types are *Call,
|
|
// *Notification and *Response.
|
|
type Message interface {
|
|
// isJSONRPC2Message is used to make the set of message implementations a
|
|
// closed set.
|
|
isJSONRPC2Message()
|
|
}
|
|
|
|
// Request is the shared interface to jsonrpc2 messages that request
|
|
// a method be invoked.
|
|
// The request types are a closed set of *Call and *Notification.
|
|
type Request interface {
|
|
Message
|
|
// 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
|
|
// isJSONRPC2Request is used to make the set of request implementations closed.
|
|
isJSONRPC2Request()
|
|
}
|
|
|
|
// Notification is a request for which a response cannot occur, and as such
|
|
// it has not ID.
|
|
type Notification struct {
|
|
// Method is a string containing the method name to invoke.
|
|
method string
|
|
params json.RawMessage
|
|
}
|
|
|
|
// Call is a request that expects a response.
|
|
// The response will have a matching ID.
|
|
type Call struct {
|
|
// 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
|
|
// id of this request, used to tie the Response back to the request.
|
|
id ID
|
|
}
|
|
|
|
// Response is a reply to a Call.
|
|
// It will have the same ID as the call it is a response to.
|
|
type Response struct {
|
|
// result is the content of the response.
|
|
result json.RawMessage
|
|
// err is set only if the call failed.
|
|
err error
|
|
// ID of the request this is a response to.
|
|
id ID
|
|
}
|
|
|
|
// NewNotification constructs a new Notification message for the supplied
|
|
// method and parameters.
|
|
func NewNotification(method string, params interface{}) (*Notification, error) {
|
|
p, merr := marshalToRaw(params)
|
|
return &Notification{method: method, params: p}, merr
|
|
}
|
|
|
|
func (msg *Notification) Method() string { return msg.method }
|
|
func (msg *Notification) Params() json.RawMessage { return msg.params }
|
|
func (msg *Notification) isJSONRPC2Message() {}
|
|
func (msg *Notification) isJSONRPC2Request() {}
|
|
|
|
func (n *Notification) MarshalJSON() ([]byte, error) {
|
|
msg := wireRequest{Method: n.method, Params: &n.params}
|
|
data, err := json.Marshal(msg)
|
|
if err != nil {
|
|
return data, fmt.Errorf("marshaling notification: %w", err)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
func (n *Notification) UnmarshalJSON(data []byte) error {
|
|
msg := wireRequest{}
|
|
if err := json.Unmarshal(data, &msg); err != nil {
|
|
return fmt.Errorf("unmarshaling notification: %w", err)
|
|
}
|
|
n.method = msg.Method
|
|
if msg.Params != nil {
|
|
n.params = *msg.Params
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// NewCall constructs a new Call message for the supplied ID, method and
|
|
// parameters.
|
|
func NewCall(id ID, method string, params interface{}) (*Call, error) {
|
|
p, merr := marshalToRaw(params)
|
|
return &Call{id: id, method: method, params: p}, merr
|
|
}
|
|
|
|
func (msg *Call) Method() string { return msg.method }
|
|
func (msg *Call) Params() json.RawMessage { return msg.params }
|
|
func (msg *Call) ID() ID { return msg.id }
|
|
func (msg *Call) isJSONRPC2Message() {}
|
|
func (msg *Call) isJSONRPC2Request() {}
|
|
|
|
func (c *Call) MarshalJSON() ([]byte, error) {
|
|
msg := wireRequest{Method: c.method, Params: &c.params, ID: &c.id}
|
|
data, err := json.Marshal(msg)
|
|
if err != nil {
|
|
return data, fmt.Errorf("marshaling call: %w", err)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
func (c *Call) UnmarshalJSON(data []byte) error {
|
|
msg := wireRequest{}
|
|
if err := json.Unmarshal(data, &msg); err != nil {
|
|
return fmt.Errorf("unmarshaling call: %w", err)
|
|
}
|
|
c.method = msg.Method
|
|
if msg.Params != nil {
|
|
c.params = *msg.Params
|
|
}
|
|
if msg.ID != nil {
|
|
c.id = *msg.ID
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// NewResponse constructs a new Response message that is a reply to the
|
|
// supplied. If err is set result may be ignored.
|
|
func NewResponse(id ID, result interface{}, err error) (*Response, error) {
|
|
r, merr := marshalToRaw(result)
|
|
return &Response{id: id, result: r, err: err}, merr
|
|
}
|
|
|
|
func (msg *Response) ID() ID { return msg.id }
|
|
func (msg *Response) Result() json.RawMessage { return msg.result }
|
|
func (msg *Response) Err() error { return msg.err }
|
|
func (msg *Response) isJSONRPC2Message() {}
|
|
|
|
func (r *Response) MarshalJSON() ([]byte, error) {
|
|
msg := &wireResponse{Result: &r.result, Error: toWireError(r.err), ID: &r.id}
|
|
data, err := json.Marshal(msg)
|
|
if err != nil {
|
|
return data, fmt.Errorf("marshaling notification: %w", err)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
func toWireError(err error) *wireError {
|
|
if err == nil {
|
|
// no error, the response is complete
|
|
return nil
|
|
}
|
|
if err, ok := err.(*wireError); ok {
|
|
// already a wire error, just use it
|
|
return err
|
|
}
|
|
result := &wireError{Message: err.Error()}
|
|
var wrapped *wireError
|
|
if errors.As(err, &wrapped) {
|
|
// if we wrapped a wire error, keep the code from the wrapped error
|
|
// but the message from the outer error
|
|
result.Code = wrapped.Code
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (r *Response) UnmarshalJSON(data []byte) error {
|
|
msg := wireResponse{}
|
|
if err := json.Unmarshal(data, &msg); err != nil {
|
|
return fmt.Errorf("unmarshaling jsonrpc response: %w", err)
|
|
}
|
|
if msg.Result != nil {
|
|
r.result = *msg.Result
|
|
}
|
|
if msg.Error != nil {
|
|
r.err = msg.Error
|
|
}
|
|
if msg.ID != nil {
|
|
r.id = *msg.ID
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func DecodeMessage(data []byte) (Message, error) {
|
|
msg := wireCombined{}
|
|
if err := json.Unmarshal(data, &msg); err != nil {
|
|
return nil, fmt.Errorf("unmarshaling jsonrpc message: %w", err)
|
|
}
|
|
if msg.Method == "" {
|
|
// no method, should be a response
|
|
if msg.ID == nil {
|
|
return nil, ErrInvalidRequest
|
|
}
|
|
response := &Response{id: *msg.ID}
|
|
if msg.Error != nil {
|
|
response.err = msg.Error
|
|
}
|
|
if msg.Result != nil {
|
|
response.result = *msg.Result
|
|
}
|
|
return response, nil
|
|
}
|
|
// has a method, must be a request
|
|
if msg.ID == nil {
|
|
// request with no ID is a notify
|
|
notify := &Notification{method: msg.Method}
|
|
if msg.Params != nil {
|
|
notify.params = *msg.Params
|
|
}
|
|
return notify, nil
|
|
}
|
|
// request with an ID, must be a call
|
|
call := &Call{method: msg.Method, id: *msg.ID}
|
|
if msg.Params != nil {
|
|
call.params = *msg.Params
|
|
}
|
|
return call, nil
|
|
}
|