benchmark/internal_server.go (91 lines of code) (raw):
// Copyright (c) 2015 Uber Technologies, Inc.
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
package benchmark
import (
"fmt"
"os"
"github.com/uber/tchannel-go"
"github.com/uber/tchannel-go/hyperbahn"
"github.com/uber/tchannel-go/raw"
"github.com/uber/tchannel-go/thrift"
gen "github.com/uber/tchannel-go/thrift/gen-go/test"
"go.uber.org/atomic"
"golang.org/x/net/context"
)
// internalServer represents a benchmark server.
type internalServer struct {
ch *tchannel.Channel
hc *hyperbahn.Client
opts *options
rawCalls atomic.Int64
thriftCalls atomic.Int64
}
// NewServer returns a new Server that can recieve Thrift calls or raw calls.
func NewServer(optFns ...Option) Server {
opts := getOptions(optFns)
if opts.external {
return newExternalServer(opts)
}
ch, err := tchannel.NewChannel(opts.svcName, &tchannel.ChannelOptions{
Logger: tchannel.NewLevelLogger(tchannel.NewLogger(os.Stderr), tchannel.LogLevelWarn),
})
if err != nil {
panic("failed to create channel: " + err.Error())
}
if err := ch.ListenAndServe("127.0.0.1:0"); err != nil {
panic("failed to listen on port 0: " + err.Error())
}
s := &internalServer{
ch: ch,
opts: opts,
}
tServer := thrift.NewServer(ch)
tServer.Register(gen.NewTChanSecondServiceServer(handler{calls: &s.thriftCalls}))
ch.Register(raw.Wrap(rawHandler{calls: &s.rawCalls}), "echo")
if len(opts.advertiseHosts) > 0 {
if err := s.Advertise(opts.advertiseHosts); err != nil {
panic("failed to advertise: " + err.Error())
}
}
return s
}
// HostPort returns the host:port that the server is listening on.
func (s *internalServer) HostPort() string {
return s.ch.PeerInfo().HostPort
}
// Advertise advertises with Hyperbahn.
func (s *internalServer) Advertise(hyperbahnHosts []string) error {
var err error
config := hyperbahn.Configuration{InitialNodes: hyperbahnHosts}
s.hc, err = hyperbahn.NewClient(s.ch, config, nil)
if err != nil {
panic("failed to setup Hyperbahn client: " + err.Error())
}
return s.hc.Advertise()
}
func (s *internalServer) Close() {
s.ch.Close()
if s.hc != nil {
s.hc.Close()
}
}
func (s *internalServer) RawCalls() int {
return int(s.rawCalls.Load())
}
func (s *internalServer) ThriftCalls() int {
return int(s.thriftCalls.Load())
}
type rawHandler struct {
calls *atomic.Int64
}
func (rawHandler) OnError(ctx context.Context, err error) {
fmt.Println("benchmark.Server error:", err)
}
func (h rawHandler) Handle(ctx context.Context, args *raw.Args) (*raw.Res, error) {
h.calls.Inc()
return &raw.Res{
Arg2: args.Arg2,
Arg3: args.Arg3,
}, nil
}
type handler struct {
calls *atomic.Int64
}
func (h handler) Echo(ctx thrift.Context, arg1 string) (string, error) {
h.calls.Inc()
return arg1, nil
}