consul: Avoid network for server RPC. Fixes #148.

This commit is contained in:
Armon Dadgar 2014-06-10 19:12:26 -07:00
parent 1b15415616
commit ab4e3de185

View File

@ -2,6 +2,7 @@ package consul
import ( import (
"crypto/tls" "crypto/tls"
"errors"
"fmt" "fmt"
"github.com/hashicorp/raft" "github.com/hashicorp/raft"
"github.com/hashicorp/raft-mdb" "github.com/hashicorp/raft-mdb"
@ -11,6 +12,7 @@ import (
"net/rpc" "net/rpc"
"os" "os"
"path/filepath" "path/filepath"
"reflect"
"runtime" "runtime"
"strconv" "strconv"
"sync" "sync"
@ -489,11 +491,52 @@ func (s *Server) IsLeader() bool {
return s.raft.State() == raft.Leader return s.raft.State() == raft.Leader
} }
// inmemCodec is used to do an RPC call without going over a network
type inmemCodec struct {
method string
args interface{}
reply interface{}
err error
}
func (i *inmemCodec) ReadRequestHeader(req *rpc.Request) error {
req.ServiceMethod = i.method
return nil
}
func (i *inmemCodec) ReadRequestBody(args interface{}) error {
sourceValue := reflect.Indirect(reflect.Indirect(reflect.ValueOf(i.args)))
dst := reflect.Indirect(reflect.Indirect(reflect.ValueOf(args)))
dst.Set(sourceValue)
return nil
}
func (i *inmemCodec) WriteResponse(resp *rpc.Response, reply interface{}) error {
if resp.Error != "" {
i.err = errors.New(resp.Error)
return nil
}
sourceValue := reflect.Indirect(reflect.Indirect(reflect.ValueOf(reply)))
dst := reflect.Indirect(reflect.Indirect(reflect.ValueOf(i.reply)))
dst.Set(sourceValue)
return nil
}
func (i *inmemCodec) Close() error {
return nil
}
// RPC is used to make a local RPC call // RPC is used to make a local RPC call
func (s *Server) RPC(method string, args interface{}, reply interface{}) error { func (s *Server) RPC(method string, args interface{}, reply interface{}) error {
addr := s.rpcListener.Addr() codec := &inmemCodec{
version := int(s.config.ProtocolVersion) method: method,
return s.connPool.RPC(addr, version, method, args, reply) args: args,
reply: reply,
}
if err := s.rpcServer.ServeRequest(codec); err != nil {
return err
}
return codec.err
} }
// Stats is used to return statistics for debugging and insight // Stats is used to return statistics for debugging and insight