| // Copyright 2009 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 rpc provides access to the exported methods of an object across a |
| network or other I/O connection. A server registers an object, making it visible |
| as a service with the name of the type of the object. After registration, exported |
| methods of the object will be accessible remotely. A server may register multiple |
| objects (services) of different types but it is an error to register multiple |
| objects of the same type. |
| |
| Only methods that satisfy these criteria will be made available for remote access; |
| other methods will be ignored: |
| |
| - the method's type is exported. |
| - the method is exported. |
| - the method has two arguments, both exported (or builtin) types. |
| - the method's second argument is a pointer. |
| - the method has return type error. |
| |
| In effect, the method must look schematically like |
| |
| func (t *T) MethodName(argType T1, replyType *T2) error |
| |
| where T1 and T2 can be marshaled by encoding/gob. |
| These requirements apply even if a different codec is used. |
| (In the future, these requirements may soften for custom codecs.) |
| |
| The method's first argument represents the arguments provided by the caller; the |
| second argument represents the result parameters to be returned to the caller. |
| The method's return value, if non-nil, is passed back as a string that the client |
| sees as if created by [errors.New]. If an error is returned, the reply parameter |
| will not be sent back to the client. |
| |
| The server may handle requests on a single connection by calling [ServeConn]. More |
| typically it will create a network listener and call [Accept] or, for an HTTP |
| listener, [HandleHTTP] and [http.Serve]. |
| |
| A client wishing to use the service establishes a connection and then invokes |
| [NewClient] on the connection. The convenience function [Dial] ([DialHTTP]) performs |
| both steps for a raw network connection (an HTTP connection). The resulting |
| [Client] object has two methods, [Call] and Go, that specify the service and method to |
| call, a pointer containing the arguments, and a pointer to receive the result |
| parameters. |
| |
| The Call method waits for the remote call to complete while the Go method |
| launches the call asynchronously and signals completion using the Call |
| structure's Done channel. |
| |
| Unless an explicit codec is set up, package [encoding/gob] is used to |
| transport the data. |
| |
| Here is a simple example. A server wishes to export an object of type Arith: |
| |
| package server |
| |
| import "errors" |
| |
| type Args struct { |
| A, B int |
| } |
| |
| type Quotient struct { |
| Quo, Rem int |
| } |
| |
| type Arith int |
| |
| func (t *Arith) Multiply(args *Args, reply *int) error { |
| *reply = args.A * args.B |
| return nil |
| } |
| |
| func (t *Arith) Divide(args *Args, quo *Quotient) error { |
| if args.B == 0 { |
| return errors.New("divide by zero") |
| } |
| quo.Quo = args.A / args.B |
| quo.Rem = args.A % args.B |
| return nil |
| } |
| |
| The server calls (for HTTP service): |
| |
| arith := new(Arith) |
| rpc.Register(arith) |
| rpc.HandleHTTP() |
| l, err := net.Listen("tcp", ":1234") |
| if err != nil { |
| log.Fatal("listen error:", err) |
| } |
| go http.Serve(l, nil) |
| |
| At this point, clients can see a service "Arith" with methods "Arith.Multiply" and |
| "Arith.Divide". To invoke one, a client first dials the server: |
| |
| client, err := rpc.DialHTTP("tcp", serverAddress + ":1234") |
| if err != nil { |
| log.Fatal("dialing:", err) |
| } |
| |
| Then it can make a remote call: |
| |
| // Synchronous call |
| args := &server.Args{7,8} |
| var reply int |
| err = client.Call("Arith.Multiply", args, &reply) |
| if err != nil { |
| log.Fatal("arith error:", err) |
| } |
| fmt.Printf("Arith: %d*%d=%d", args.A, args.B, reply) |
| |
| or |
| |
| // Asynchronous call |
| quotient := new(Quotient) |
| divCall := client.Go("Arith.Divide", args, quotient, nil) |
| replyCall := <-divCall.Done // will be equal to divCall |
| // check errors, print, etc. |
| |
| A server implementation will often provide a simple, type-safe wrapper for the |
| client. |
| |
| The net/rpc package is frozen and is not accepting new features. |
| */ |
| package rpc |
| |
| import ( |
| "bufio" |
| "encoding/gob" |
| "errors" |
| "go/token" |
| "io" |
| "log" |
| "net" |
| "net/http" |
| "reflect" |
| "strings" |
| "sync" |
| ) |
| |
| const ( |
| // Defaults used by HandleHTTP |
| DefaultRPCPath = "/_goRPC_" |
| DefaultDebugPath = "/debug/rpc" |
| ) |
| |
| // Precompute the reflect type for error. |
| var typeOfError = reflect.TypeFor[error]() |
| |
| type methodType struct { |
| sync.Mutex // protects counters |
| method reflect.Method |
| ArgType reflect.Type |
| ReplyType reflect.Type |
| numCalls uint |
| } |
| |
| type service struct { |
| name string // name of service |
| rcvr reflect.Value // receiver of methods for the service |
| typ reflect.Type // type of the receiver |
| method map[string]*methodType // registered methods |
| } |
| |
| // Request is a header written before every RPC call. It is used internally |
| // but documented here as an aid to debugging, such as when analyzing |
| // network traffic. |
| type Request struct { |
| ServiceMethod string // format: "Service.Method" |
| Seq uint64 // sequence number chosen by client |
| next *Request // for free list in Server |
| } |
| |
| // Response is a header written before every RPC return. It is used internally |
| // but documented here as an aid to debugging, such as when analyzing |
| // network traffic. |
| type Response struct { |
| ServiceMethod string // echoes that of the Request |
| Seq uint64 // echoes that of the request |
| Error string // error, if any. |
| next *Response // for free list in Server |
| } |
| |
| // Server represents an RPC Server. |
| type Server struct { |
| serviceMap sync.Map // map[string]*service |
| reqLock sync.Mutex // protects freeReq |
| freeReq *Request |
| respLock sync.Mutex // protects freeResp |
| freeResp *Response |
| } |
| |
| // NewServer returns a new [Server]. |
| func NewServer() *Server { |
| return &Server{} |
| } |
| |
| // DefaultServer is the default instance of [*Server]. |
| var DefaultServer = NewServer() |
| |
| // Is this type exported or a builtin? |
| func isExportedOrBuiltinType(t reflect.Type) bool { |
| for t.Kind() == reflect.Pointer { |
| t = t.Elem() |
| } |
| // PkgPath will be non-empty even for an exported type, |
| // so we need to check the type name as well. |
| return token.IsExported(t.Name()) || t.PkgPath() == "" |
| } |
| |
| // Register publishes in the server the set of methods of the |
| // receiver value that satisfy the following conditions: |
| // - exported method of exported type |
| // - two arguments, both of exported type |
| // - the second argument is a pointer |
| // - one return value, of type error |
| // |
| // It returns an error if the receiver is not an exported type or has |
| // no suitable methods. It also logs the error using package log. |
| // The client accesses each method using a string of the form "Type.Method", |
| // where Type is the receiver's concrete type. |
| func (server *Server) Register(rcvr any) error { |
| return server.register(rcvr, "", false) |
| } |
| |
| // RegisterName is like [Register] but uses the provided name for the type |
| // instead of the receiver's concrete type. |
| func (server *Server) RegisterName(name string, rcvr any) error { |
| return server.register(rcvr, name, true) |
| } |
| |
| // logRegisterError specifies whether to log problems during method registration. |
| // To debug registration, recompile the package with this set to true. |
| const logRegisterError = false |
| |
| func (server *Server) register(rcvr any, name string, useName bool) error { |
| s := new(service) |
| s.typ = reflect.TypeOf(rcvr) |
| s.rcvr = reflect.ValueOf(rcvr) |
| sname := name |
| if !useName { |
| sname = reflect.Indirect(s.rcvr).Type().Name() |
| } |
| if sname == "" { |
| s := "rpc.Register: no service name for type " + s.typ.String() |
| log.Print(s) |
| return errors.New(s) |
| } |
| if !useName && !token.IsExported(sname) { |
| s := "rpc.Register: type " + sname + " is not exported" |
| log.Print(s) |
| return errors.New(s) |
| } |
| s.name = sname |
| |
| // Install the methods |
| s.method = suitableMethods(s.typ, logRegisterError) |
| |
| if len(s.method) == 0 { |
| str := "" |
| |
| // To help the user, see if a pointer receiver would work. |
| method := suitableMethods(reflect.PointerTo(s.typ), false) |
| if len(method) != 0 { |
| str = "rpc.Register: type " + sname + " has no exported methods of suitable type (hint: pass a pointer to value of that type)" |
| } else { |
| str = "rpc.Register: type " + sname + " has no exported methods of suitable type" |
| } |
| log.Print(str) |
| return errors.New(str) |
| } |
| |
| if _, dup := server.serviceMap.LoadOrStore(sname, s); dup { |
| return errors.New("rpc: service already defined: " + sname) |
| } |
| return nil |
| } |
| |
| // suitableMethods returns suitable Rpc methods of typ. It will log |
| // errors if logErr is true. |
| func suitableMethods(typ reflect.Type, logErr bool) map[string]*methodType { |
| methods := make(map[string]*methodType) |
| for m := 0; m < typ.NumMethod(); m++ { |
| method := typ.Method(m) |
| mtype := method.Type |
| mname := method.Name |
| // Method must be exported. |
| if !method.IsExported() { |
| continue |
| } |
| // Method needs three ins: receiver, *args, *reply. |
| if mtype.NumIn() != 3 { |
| if logErr { |
| log.Printf("rpc.Register: method %q has %d input parameters; needs exactly three\n", mname, mtype.NumIn()) |
| } |
| continue |
| } |
| // First arg need not be a pointer. |
| argType := mtype.In(1) |
| if !isExportedOrBuiltinType(argType) { |
| if logErr { |
| log.Printf("rpc.Register: argument type of method %q is not exported: %q\n", mname, argType) |
| } |
| continue |
| } |
| // Second arg must be a pointer. |
| replyType := mtype.In(2) |
| if replyType.Kind() != reflect.Pointer { |
| if logErr { |
| log.Printf("rpc.Register: reply type of method %q is not a pointer: %q\n", mname, replyType) |
| } |
| continue |
| } |
| // Reply type must be exported. |
| if !isExportedOrBuiltinType(replyType) { |
| if logErr { |
| log.Printf("rpc.Register: reply type of method %q is not exported: %q\n", mname, replyType) |
| } |
| continue |
| } |
| // Method needs one out. |
| if mtype.NumOut() != 1 { |
| if logErr { |
| log.Printf("rpc.Register: method %q has %d output parameters; needs exactly one\n", mname, mtype.NumOut()) |
| } |
| continue |
| } |
| // The return type of the method must be error. |
| if returnType := mtype.Out(0); returnType != typeOfError { |
| if logErr { |
| log.Printf("rpc.Register: return type of method %q is %q, must be error\n", mname, returnType) |
| } |
| continue |
| } |
| methods[mname] = &methodType{method: method, ArgType: argType, ReplyType: replyType} |
| } |
| return methods |
| } |
| |
| // A value sent as a placeholder for the server's response value when the server |
| // receives an invalid request. It is never decoded by the client since the Response |
| // contains an error when it is used. |
| var invalidRequest = struct{}{} |
| |
| func (server *Server) sendResponse(sending *sync.Mutex, req *Request, reply any, codec ServerCodec, errmsg string) { |
| resp := server.getResponse() |
| // Encode the response header |
| resp.ServiceMethod = req.ServiceMethod |
| if errmsg != "" { |
| resp.Error = errmsg |
| reply = invalidRequest |
| } |
| resp.Seq = req.Seq |
| sending.Lock() |
| err := codec.WriteResponse(resp, reply) |
| if debugLog && err != nil { |
| log.Println("rpc: writing response:", err) |
| } |
| sending.Unlock() |
| server.freeResponse(resp) |
| } |
| |
| func (m *methodType) NumCalls() (n uint) { |
| m.Lock() |
| n = m.numCalls |
| m.Unlock() |
| return n |
| } |
| |
| func (s *service) call(server *Server, sending *sync.Mutex, wg *sync.WaitGroup, mtype *methodType, req *Request, argv, replyv reflect.Value, codec ServerCodec) { |
| if wg != nil { |
| defer wg.Done() |
| } |
| mtype.Lock() |
| mtype.numCalls++ |
| mtype.Unlock() |
| function := mtype.method.Func |
| // Invoke the method, providing a new value for the reply. |
| returnValues := function.Call([]reflect.Value{s.rcvr, argv, replyv}) |
| // The return value for the method is an error. |
| errInter := returnValues[0].Interface() |
| errmsg := "" |
| if errInter != nil { |
| errmsg = errInter.(error).Error() |
| } |
| server.sendResponse(sending, req, replyv.Interface(), codec, errmsg) |
| server.freeRequest(req) |
| } |
| |
| type gobServerCodec struct { |
| rwc io.ReadWriteCloser |
| dec *gob.Decoder |
| enc *gob.Encoder |
| encBuf *bufio.Writer |
| closed bool |
| } |
| |
| func (c *gobServerCodec) ReadRequestHeader(r *Request) error { |
| return c.dec.Decode(r) |
| } |
| |
| func (c *gobServerCodec) ReadRequestBody(body any) error { |
| return c.dec.Decode(body) |
| } |
| |
| func (c *gobServerCodec) WriteResponse(r *Response, body any) (err error) { |
| if err = c.enc.Encode(r); err != nil { |
| if c.encBuf.Flush() == nil { |
| // Gob couldn't encode the header. Should not happen, so if it does, |
| // shut down the connection to signal that the connection is broken. |
| log.Println("rpc: gob error encoding response:", err) |
| c.Close() |
| } |
| return |
| } |
| if err = c.enc.Encode(body); err != nil { |
| if c.encBuf.Flush() == nil { |
| // Was a gob problem encoding the body but the header has been written. |
| // Shut down the connection to signal that the connection is broken. |
| log.Println("rpc: gob error encoding body:", err) |
| c.Close() |
| } |
| return |
| } |
| return c.encBuf.Flush() |
| } |
| |
| func (c *gobServerCodec) Close() error { |
| if c.closed { |
| // Only call c.rwc.Close once; otherwise the semantics are undefined. |
| return nil |
| } |
| c.closed = true |
| return c.rwc.Close() |
| } |
| |
| // ServeConn runs the server on a single connection. |
| // ServeConn blocks, serving the connection until the client hangs up. |
| // The caller typically invokes ServeConn in a go statement. |
| // ServeConn uses the gob wire format (see package gob) on the |
| // connection. To use an alternate codec, use [ServeCodec]. |
| // See [NewClient]'s comment for information about concurrent access. |
| func (server *Server) ServeConn(conn io.ReadWriteCloser) { |
| buf := bufio.NewWriter(conn) |
| srv := &gobServerCodec{ |
| rwc: conn, |
| dec: gob.NewDecoder(conn), |
| enc: gob.NewEncoder(buf), |
| encBuf: buf, |
| } |
| server.ServeCodec(srv) |
| } |
| |
| // ServeCodec is like [ServeConn] but uses the specified codec to |
| // decode requests and encode responses. |
| func (server *Server) ServeCodec(codec ServerCodec) { |
| sending := new(sync.Mutex) |
| wg := new(sync.WaitGroup) |
| for { |
| service, mtype, req, argv, replyv, keepReading, err := server.readRequest(codec) |
| if err != nil { |
| if debugLog && err != io.EOF { |
| log.Println("rpc:", err) |
| } |
| if !keepReading { |
| break |
| } |
| // send a response if we actually managed to read a header. |
| if req != nil { |
| server.sendResponse(sending, req, invalidRequest, codec, err.Error()) |
| server.freeRequest(req) |
| } |
| continue |
| } |
| wg.Add(1) |
| go service.call(server, sending, wg, mtype, req, argv, replyv, codec) |
| } |
| // We've seen that there are no more requests. |
| // Wait for responses to be sent before closing codec. |
| wg.Wait() |
| codec.Close() |
| } |
| |
| // ServeRequest is like [ServeCodec] but synchronously serves a single request. |
| // It does not close the codec upon completion. |
| func (server *Server) ServeRequest(codec ServerCodec) error { |
| sending := new(sync.Mutex) |
| service, mtype, req, argv, replyv, keepReading, err := server.readRequest(codec) |
| if err != nil { |
| if !keepReading { |
| return err |
| } |
| // send a response if we actually managed to read a header. |
| if req != nil { |
| server.sendResponse(sending, req, invalidRequest, codec, err.Error()) |
| server.freeRequest(req) |
| } |
| return err |
| } |
| service.call(server, sending, nil, mtype, req, argv, replyv, codec) |
| return nil |
| } |
| |
| func (server *Server) getRequest() *Request { |
| server.reqLock.Lock() |
| req := server.freeReq |
| if req == nil { |
| req = new(Request) |
| } else { |
| server.freeReq = req.next |
| *req = Request{} |
| } |
| server.reqLock.Unlock() |
| return req |
| } |
| |
| func (server *Server) freeRequest(req *Request) { |
| server.reqLock.Lock() |
| req.next = server.freeReq |
| server.freeReq = req |
| server.reqLock.Unlock() |
| } |
| |
| func (server *Server) getResponse() *Response { |
| server.respLock.Lock() |
| resp := server.freeResp |
| if resp == nil { |
| resp = new(Response) |
| } else { |
| server.freeResp = resp.next |
| *resp = Response{} |
| } |
| server.respLock.Unlock() |
| return resp |
| } |
| |
| func (server *Server) freeResponse(resp *Response) { |
| server.respLock.Lock() |
| resp.next = server.freeResp |
| server.freeResp = resp |
| server.respLock.Unlock() |
| } |
| |
| func (server *Server) readRequest(codec ServerCodec) (service *service, mtype *methodType, req *Request, argv, replyv reflect.Value, keepReading bool, err error) { |
| service, mtype, req, keepReading, err = server.readRequestHeader(codec) |
| if err != nil { |
| if !keepReading { |
| return |
| } |
| // discard body |
| codec.ReadRequestBody(nil) |
| return |
| } |
| |
| // Decode the argument value. |
| argIsValue := false // if true, need to indirect before calling. |
| if mtype.ArgType.Kind() == reflect.Pointer { |
| argv = reflect.New(mtype.ArgType.Elem()) |
| } else { |
| argv = reflect.New(mtype.ArgType) |
| argIsValue = true |
| } |
| // argv guaranteed to be a pointer now. |
| if err = codec.ReadRequestBody(argv.Interface()); err != nil { |
| return |
| } |
| if argIsValue { |
| argv = argv.Elem() |
| } |
| |
| replyv = reflect.New(mtype.ReplyType.Elem()) |
| |
| switch mtype.ReplyType.Elem().Kind() { |
| case reflect.Map: |
| replyv.Elem().Set(reflect.MakeMap(mtype.ReplyType.Elem())) |
| case reflect.Slice: |
| replyv.Elem().Set(reflect.MakeSlice(mtype.ReplyType.Elem(), 0, 0)) |
| } |
| return |
| } |
| |
| func (server *Server) readRequestHeader(codec ServerCodec) (svc *service, mtype *methodType, req *Request, keepReading bool, err error) { |
| // Grab the request header. |
| req = server.getRequest() |
| err = codec.ReadRequestHeader(req) |
| if err != nil { |
| req = nil |
| if err == io.EOF || err == io.ErrUnexpectedEOF { |
| return |
| } |
| err = errors.New("rpc: server cannot decode request: " + err.Error()) |
| return |
| } |
| |
| // We read the header successfully. If we see an error now, |
| // we can still recover and move on to the next request. |
| keepReading = true |
| |
| dot := strings.LastIndex(req.ServiceMethod, ".") |
| if dot < 0 { |
| err = errors.New("rpc: service/method request ill-formed: " + req.ServiceMethod) |
| return |
| } |
| serviceName := req.ServiceMethod[:dot] |
| methodName := req.ServiceMethod[dot+1:] |
| |
| // Look up the request. |
| svci, ok := server.serviceMap.Load(serviceName) |
| if !ok { |
| err = errors.New("rpc: can't find service " + req.ServiceMethod) |
| return |
| } |
| svc = svci.(*service) |
| mtype = svc.method[methodName] |
| if mtype == nil { |
| err = errors.New("rpc: can't find method " + req.ServiceMethod) |
| } |
| return |
| } |
| |
| // Accept accepts connections on the listener and serves requests |
| // for each incoming connection. Accept blocks until the listener |
| // returns a non-nil error. The caller typically invokes Accept in a |
| // go statement. |
| func (server *Server) Accept(lis net.Listener) { |
| for { |
| conn, err := lis.Accept() |
| if err != nil { |
| log.Print("rpc.Serve: accept:", err.Error()) |
| return |
| } |
| go server.ServeConn(conn) |
| } |
| } |
| |
| // Register publishes the receiver's methods in the [DefaultServer]. |
| func Register(rcvr any) error { return DefaultServer.Register(rcvr) } |
| |
| // RegisterName is like [Register] but uses the provided name for the type |
| // instead of the receiver's concrete type. |
| func RegisterName(name string, rcvr any) error { |
| return DefaultServer.RegisterName(name, rcvr) |
| } |
| |
| // A ServerCodec implements reading of RPC requests and writing of |
| // RPC responses for the server side of an RPC session. |
| // The server calls [ServerCodec.ReadRequestHeader] and [ServerCodec.ReadRequestBody] in pairs |
| // to read requests from the connection, and it calls [ServerCodec.WriteResponse] to |
| // write a response back. The server calls [ServerCodec.Close] when finished with the |
| // connection. ReadRequestBody may be called with a nil |
| // argument to force the body of the request to be read and discarded. |
| // See [NewClient]'s comment for information about concurrent access. |
| type ServerCodec interface { |
| ReadRequestHeader(*Request) error |
| ReadRequestBody(any) error |
| WriteResponse(*Response, any) error |
| |
| // Close can be called multiple times and must be idempotent. |
| Close() error |
| } |
| |
| // ServeConn runs the [DefaultServer] on a single connection. |
| // ServeConn blocks, serving the connection until the client hangs up. |
| // The caller typically invokes ServeConn in a go statement. |
| // ServeConn uses the gob wire format (see package gob) on the |
| // connection. To use an alternate codec, use [ServeCodec]. |
| // See [NewClient]'s comment for information about concurrent access. |
| func ServeConn(conn io.ReadWriteCloser) { |
| DefaultServer.ServeConn(conn) |
| } |
| |
| // ServeCodec is like [ServeConn] but uses the specified codec to |
| // decode requests and encode responses. |
| func ServeCodec(codec ServerCodec) { |
| DefaultServer.ServeCodec(codec) |
| } |
| |
| // ServeRequest is like [ServeCodec] but synchronously serves a single request. |
| // It does not close the codec upon completion. |
| func ServeRequest(codec ServerCodec) error { |
| return DefaultServer.ServeRequest(codec) |
| } |
| |
| // Accept accepts connections on the listener and serves requests |
| // to [DefaultServer] for each incoming connection. |
| // Accept blocks; the caller typically invokes it in a go statement. |
| func Accept(lis net.Listener) { DefaultServer.Accept(lis) } |
| |
| // Can connect to RPC service using HTTP CONNECT to rpcPath. |
| var connected = "200 Connected to Go RPC" |
| |
| // ServeHTTP implements an [http.Handler] that answers RPC requests. |
| func (server *Server) ServeHTTP(w http.ResponseWriter, req *http.Request) { |
| if req.Method != "CONNECT" { |
| w.Header().Set("Content-Type", "text/plain; charset=utf-8") |
| w.WriteHeader(http.StatusMethodNotAllowed) |
| io.WriteString(w, "405 must CONNECT\n") |
| return |
| } |
| conn, _, err := w.(http.Hijacker).Hijack() |
| if err != nil { |
| log.Print("rpc hijacking ", req.RemoteAddr, ": ", err.Error()) |
| return |
| } |
| io.WriteString(conn, "HTTP/1.0 "+connected+"\n\n") |
| server.ServeConn(conn) |
| } |
| |
| // HandleHTTP registers an HTTP handler for RPC messages on rpcPath, |
| // and a debugging handler on debugPath. |
| // It is still necessary to invoke [http.Serve](), typically in a go statement. |
| func (server *Server) HandleHTTP(rpcPath, debugPath string) { |
| http.Handle(rpcPath, server) |
| http.Handle(debugPath, debugHTTP{server}) |
| } |
| |
| // HandleHTTP registers an HTTP handler for RPC messages to [DefaultServer] |
| // on [DefaultRPCPath] and a debugging handler on [DefaultDebugPath]. |
| // It is still necessary to invoke [http.Serve](), typically in a go statement. |
| func HandleHTTP() { |
| DefaultServer.HandleHTTP(DefaultRPCPath, DefaultDebugPath) |
| } |