crossdock/behavior/trace/thrift.go (79 lines of code) (raw):
// Copyright (c) 2016 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 trace
import (
"encoding/json"
"github.com/uber/tchannel-go"
"github.com/uber/tchannel-go/crossdock/log"
"github.com/uber/tchannel-go/thrift"
gen "github.com/uber/tchannel-go/thrift/gen-go/test"
"golang.org/x/net/context"
)
func (b *Behavior) registerThrift(ch *tchannel.Channel) {
handler := &thriftHandler{b: b, ch: ch}
server := thrift.NewServer(ch)
server.Register(gen.NewTChanSimpleServiceServer(handler))
b.thriftCall = handler.callDownstream
}
type thriftHandler struct {
gen.TChanSimpleService // leave nil so calls to unimplemented methods panic.
ch *tchannel.Channel
b *Behavior
}
func (h *thriftHandler) Call(ctx thrift.Context, arg *gen.Data) (*gen.Data, error) {
req, err := requestFromThrift(arg)
if err != nil {
return nil, err
}
res, err := h.b.prepareResponse(ctx, nil, req.Downstream)
if err != nil {
return nil, err
}
return responseToThrift(res)
}
func (h *thriftHandler) callDownstream(ctx context.Context, target *Downstream) (*Response, error) {
req := &Request{
ServerRole: target.ServerRole,
Downstream: target.Downstream,
}
opts := &thrift.ClientOptions{HostPort: target.HostPort}
thriftClient := thrift.NewClient(h.ch, target.ServiceName, opts)
serviceClient := gen.NewTChanSimpleServiceClient(thriftClient)
tReq, err := requestToThrift(req)
if err != nil {
return nil, err
}
log.Printf("Calling Thrift service %s (%s)", target.ServiceName, target.HostPort)
tctx := thrift.Wrap(ctx)
res, err := serviceClient.Call(tctx, tReq)
if err != nil {
return nil, err
}
return responseFromThrift(res)
}
func requestFromThrift(req *gen.Data) (*Request, error) {
var r Request
if err := json.Unmarshal([]byte(req.S2), &r); err != nil {
return nil, err
}
return &r, nil
}
func requestToThrift(r *Request) (*gen.Data, error) {
jsonBytes, err := json.Marshal(r)
if err != nil {
return nil, err
}
return &gen.Data{S2: string(jsonBytes)}, nil
}
func responseFromThrift(res *gen.Data) (*Response, error) {
var r Response
if err := json.Unmarshal([]byte(res.S2), &r); err != nil {
return nil, err
}
return &r, nil
}
func responseToThrift(r *Response) (*gen.Data, error) {
jsonBytes, err := json.Marshal(r)
if err != nil {
return nil, err
}
return &gen.Data{S2: string(jsonBytes)}, nil
}