2022-02-06 07:06:32 +00:00
|
|
|
// Copyright (C) 2017 Michael J. Fromberger. All Rights Reserved.
|
|
|
|
|
2021-12-04 16:42:11 +00:00
|
|
|
package jrpc2_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"log"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/creachadair/jrpc2"
|
|
|
|
"github.com/creachadair/jrpc2/code"
|
|
|
|
"github.com/creachadair/jrpc2/handler"
|
|
|
|
"github.com/creachadair/jrpc2/server"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ctx = context.Background()
|
|
|
|
)
|
|
|
|
|
|
|
|
type Msg struct {
|
|
|
|
Text string `json:"msg"`
|
|
|
|
}
|
|
|
|
|
2022-02-06 07:06:32 +00:00
|
|
|
var local = server.NewLocal(handler.Map{
|
|
|
|
"Hello": handler.New(func(ctx context.Context) string {
|
|
|
|
return "Hello, world!"
|
|
|
|
}),
|
|
|
|
"Echo": handler.New(func(_ context.Context, args []json.RawMessage) []json.RawMessage {
|
|
|
|
return args
|
|
|
|
}),
|
|
|
|
"Log": handler.New(func(ctx context.Context, msg Msg) (bool, error) {
|
|
|
|
fmt.Println("Log:", msg.Text)
|
|
|
|
return true, nil
|
|
|
|
}),
|
|
|
|
}, nil)
|
2021-12-04 16:42:11 +00:00
|
|
|
|
|
|
|
func ExampleNewServer() {
|
|
|
|
// We can query the server for its current status information, including a
|
|
|
|
// list of its methods.
|
2022-02-06 07:06:32 +00:00
|
|
|
si := local.Server.ServerInfo()
|
2021-12-04 16:42:11 +00:00
|
|
|
|
|
|
|
fmt.Println(strings.Join(si.Methods, "\n"))
|
|
|
|
// Output:
|
|
|
|
// Echo
|
|
|
|
// Hello
|
|
|
|
// Log
|
|
|
|
}
|
|
|
|
|
|
|
|
func ExampleClient_Call() {
|
2022-02-06 07:06:32 +00:00
|
|
|
rsp, err := local.Client.Call(ctx, "Hello", nil)
|
2021-12-04 16:42:11 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Call: %v", err)
|
|
|
|
}
|
|
|
|
var msg string
|
|
|
|
if err := rsp.UnmarshalResult(&msg); err != nil {
|
|
|
|
log.Fatalf("Decoding result: %v", err)
|
|
|
|
}
|
|
|
|
fmt.Println(msg)
|
|
|
|
// Output:
|
|
|
|
// Hello, world!
|
|
|
|
}
|
|
|
|
|
|
|
|
func ExampleClient_CallResult() {
|
|
|
|
var msg string
|
2022-02-06 07:06:32 +00:00
|
|
|
if err := local.Client.CallResult(ctx, "Hello", nil, &msg); err != nil {
|
2021-12-04 16:42:11 +00:00
|
|
|
log.Fatalf("CallResult: %v", err)
|
|
|
|
}
|
|
|
|
fmt.Println(msg)
|
|
|
|
// Output:
|
|
|
|
// Hello, world!
|
|
|
|
}
|
|
|
|
|
|
|
|
func ExampleClient_Batch() {
|
2022-02-06 07:06:32 +00:00
|
|
|
rsps, err := local.Client.Batch(ctx, []jrpc2.Spec{
|
2021-12-04 16:42:11 +00:00
|
|
|
{Method: "Hello"},
|
|
|
|
{Method: "Log", Params: Msg{"Sing it!"}, Notify: true},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Batch: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Printf("len(rsps) = %d\n", len(rsps))
|
|
|
|
for i, rsp := range rsps {
|
|
|
|
var msg string
|
|
|
|
if err := rsp.UnmarshalResult(&msg); err != nil {
|
|
|
|
log.Fatalf("Invalid result: %v", err)
|
|
|
|
}
|
|
|
|
fmt.Printf("Response #%d: %s\n", i+1, msg)
|
|
|
|
}
|
|
|
|
// Output:
|
|
|
|
// Log: Sing it!
|
|
|
|
// len(rsps) = 1
|
|
|
|
// Response #1: Hello, world!
|
|
|
|
}
|
|
|
|
|
|
|
|
func ExampleRequest_UnmarshalParams() {
|
|
|
|
const msg = `{"jsonrpc":"2.0", "id":101, "method":"M", "params":{"a":1, "b":2, "c":3}}`
|
|
|
|
|
|
|
|
reqs, err := jrpc2.ParseRequests([]byte(msg))
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("ParseRequests: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var t, u struct {
|
|
|
|
A int `json:"a"`
|
|
|
|
B int `json:"b"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// By default, unmarshaling ignores unknown fields (here, "c").
|
|
|
|
if err := reqs[0].UnmarshalParams(&t); err != nil {
|
|
|
|
log.Fatalf("UnmarshalParams: %v", err)
|
|
|
|
}
|
|
|
|
fmt.Printf("t.A=%d, t.B=%d\n", t.A, t.B)
|
|
|
|
|
|
|
|
// To implement strict field checking, there are several options:
|
|
|
|
//
|
|
|
|
// Solution 1: Use the jrpc2.StrictFields helper.
|
|
|
|
err = reqs[0].UnmarshalParams(jrpc2.StrictFields(&t))
|
|
|
|
if code.FromError(err) != code.InvalidParams {
|
|
|
|
log.Fatalf("UnmarshalParams strict: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Solution 2: Implement a DisallowUnknownFields method.
|
|
|
|
var p strictParams
|
|
|
|
err = reqs[0].UnmarshalParams(&p)
|
|
|
|
if code.FromError(err) != code.InvalidParams {
|
|
|
|
log.Fatalf("UnmarshalParams strict: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Solution 3: Decode the raw message separately.
|
|
|
|
var tmp json.RawMessage
|
|
|
|
reqs[0].UnmarshalParams(&tmp) // cannot fail
|
|
|
|
dec := json.NewDecoder(bytes.NewReader(tmp))
|
|
|
|
dec.DisallowUnknownFields()
|
|
|
|
if err := dec.Decode(&u); err == nil {
|
|
|
|
log.Fatal("Decode should have failed for an unknown field")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Output:
|
|
|
|
// t.A=1, t.B=2
|
|
|
|
}
|
|
|
|
|
|
|
|
type strictParams struct {
|
|
|
|
A int `json:"a"`
|
|
|
|
B int `json:"b"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (strictParams) DisallowUnknownFields() {}
|
|
|
|
|
|
|
|
func ExampleResponse_UnmarshalResult() {
|
2022-02-06 07:06:32 +00:00
|
|
|
rsp, err := local.Client.Call(ctx, "Echo", []string{"alpha", "oscar", "kilo"})
|
2021-12-04 16:42:11 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Call: %v", err)
|
|
|
|
}
|
|
|
|
var r1, r3 string
|
|
|
|
|
|
|
|
// Note the nil, which tells the decoder to skip that argument.
|
|
|
|
if err := rsp.UnmarshalResult(&handler.Args{&r1, nil, &r3}); err != nil {
|
|
|
|
log.Fatalf("Decoding result: %v", err)
|
|
|
|
}
|
|
|
|
fmt.Println(r1, r3)
|
|
|
|
// Output:
|
|
|
|
// alpha kilo
|
|
|
|
}
|