# JSON-RPC 2.0 ❗️ Main repo: [https://gitrepo.ru/neonxp/jsonrpc2](https://gitrepo.ru/neonxp/jsonrpc2). Github is only mirror. Golang implementation of JSON-RPC 2.0 server with generics. Go 1.18+ required ## Features: - [x] HTTP/HTTPS transport - [x] TCP transport - [ ] WebSocket transport ## Usage (http transport) 1. Create JSON-RPC server with options: ```go import "neonxp.ru/go/jsonrpc2/rpc" ... s := rpc.New( rpc.WithTransport(&transport.HTTP{ Bind: ":8000", // Port to bind CORSOrigin: "*", // CORS origin TLS: &tls.Config{}, // Optional TLS config (default nil) Parallel: true, // Allow parallel run batch methods (default false) }), //Other options like transports/middlewares... ) ``` 2. Add required transport(s): ```go import "neonxp.ru/go/jsonrpc2/transport" ... s.Use( rpc.WithTransport(&transport.TCP{Bind: ":3000"}), //... ) ``` 3. Write handlers: ```go // This handler supports request parameters func Multiply(ctx context.Context, args *Args) (int, error) { return args.A * args.B, nil } // This handler has no request parameters func Hello(ctx context.Context) (string, error) { return "World", nil } ``` A handler must have a context as first parameter and may have a second parameter, representing request paramters (input of any json serializable type). A handler always returns exactly two values (output of any json serializable type and error). 4. Wrap the handler using one of the two functions `rpc.H` (supporting req params) or `rpc.HS` (no params) and register it with the server: ```go // handler has params s.Register("multiply", rpc.H(Multiply)) // handler has no params s.Register("hello", rpc.HS(Hello)) ``` 5. Run RPC server: ```go s.Run(ctx) ``` ## Custom transport Any transport must implement simple interface `transport.Transport`: ```go type Transport interface { Run(ctx context.Context, resolver Resolver) error } ``` ## Complete example [Full code](/example) ```go package main import ( "context" "neonxp.ru/go/jsonrpc2/rpc" "neonxp.ru/go/jsonrpc2/rpc/middleware" "neonxp.ru/go/jsonrpc2/transport" ) func main() { s := rpc.New( rpc.WithLogger(rpc.StdLogger), // Optional logger rpc.WithTransport(&transport.HTTP{Bind: ":8000"}), // HTTP transport ) // Set options after constructor s.Use( rpc.WithTransport(&transport.TCP{Bind: ":3000"}), // TCP transport rpc.WithMiddleware(middleware.Logger(rpc.StdLogger)), // Logger middleware ) s.Register("multiply", rpc.H(Multiply)) s.Register("divide", rpc.H(Divide)) s.Register("hello", rpc.HS(Hello)) s.Run(context.Background()) } func Multiply(ctx context.Context, args *Args) (int, error) { //... } func Divide(ctx context.Context, args *Args) (*Quotient, error) { //... } func Hello(ctx context.Context) (string, error) { // ... } type Args struct { A int `json:"a"` B int `json:"b"` } type Quotient struct { Quo int `json:"quo"` Rem int `json:"rem"` } ``` ## Author Alexander Kiryukhin ## License ![GPL v3](https://www.gnu.org/graphics/gplv3-with-text-136x68.png)