-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathoptions.go
194 lines (165 loc) · 10.1 KB
/
options.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
// 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 main
import (
"errors"
"strconv"
"time"
"github.com/yarpc/yab/encoding"
)
// Options are parsed from flags using go-flags.
type Options struct {
ROpts RequestOptions `group:"request"`
TOpts TransportOptions `group:"transport"`
BOpts BenchmarkOptions `group:"benchmark"`
Verbosity []bool `short:"v" description:"Enable more detailed logging. Repeats increase the verbosity, ie. -vvv"`
DisplayVersion bool `long:"version" description:"Displays the application version"`
ManPage bool `long:"man-page" hidden:"yes" description:"Print yab's man page to stdout"`
}
// RequestOptions are request related options
type RequestOptions struct {
Encoding encoding.Encoding `short:"e" long:"encoding" description:"The encoding of the data, options are: Thrift, proto, JSON, raw. Defaults to Thrift if the method contains '::' or a Thrift file is specified. Defaults to proto if the method contains '/' or a proto filedescriptorset is specified"`
ThriftFile string `short:"t" long:"thrift" description:"Path of the .thrift file"`
FileDescriptorSet []string `short:"F" long:"file-descriptor-set-bin" description:"A binary file containing a compiled protobuf FileDescriptorSet."`
Procedure string `long:"procedure" description:"The full method name to invoke (Thrift: Svc::Method, Proto: package.Service/Method)."`
MethodName stringAlias `short:"m" long:"method" description:"Alias for procedure"`
RequestJSON string `short:"r" long:"request" unquote:"false" description:"The request body, in JSON or YAML format"`
RequestFile string `short:"f" long:"file" description:"Path of a file containing the request body in JSON or YAML"`
Headers map[string]string `short:"H" long:"header" description:"Individual application header as a key:value pair per flag"`
HeadersJSON string `long:"headers" unquote:"false" description:"The headers in JSON or YAML format"`
HeadersFile string `long:"headers-file" description:"Path of a file containing the headers in JSON or YAML"`
Baggage map[string]string `short:"B" long:"baggage" description:"Individual context baggage header as a key:value pair per flag"`
Health bool `long:"health" description:"Hit the health endpoint, Meta::health"`
Timeout timeMillisFlag `long:"timeout" default-mask:"1s" description:"The timeout for each request. E.g., 100ms, 0.5s, 1s. If no unit is specified, milliseconds are assumed."`
YamlTemplate string `short:"y" long:"yaml-template" description:"Send a tchannel request specified by a YAML template"`
TemplateArgs map[string]string `short:"A" long:"arg" description:"A list of key-value template arguments, specified as -A foo:bar -A user:me"`
// Thrift options
ThriftDisableEnvelopes bool `long:"disable-thrift-envelope" description:"Disables Thrift envelopes (disabled by default for TChannel and gRPC)"`
ThriftMultiplexed bool `long:"multiplexed-thrift" description:"Enables the Thrift TMultiplexedProtocol used by services that host multiple Thrift services on a single endpoint."`
// These are aliases for tcurl compatibility.
Aliases struct {
Endpoint stringAlias `long:"endpoint" hidden:"true"`
Arg1 stringAlias `short:"1" long:"arg1" hidden:"true"`
Arg2 stringAlias `short:"2" long:"arg2" unquote:"false" hidden:"true"`
Arg3 stringAlias `short:"3" long:"arg3" unquote:"false" hidden:"true"`
Body stringAlias `long:"body" unquote:"false" hidden:"true"`
JSON bool `long:"json" hidden:"true"`
Raw bool `long:"raw" hidden:"true"`
}
StreamRequestOptions StreamRequestOptions
}
// StreamRequestOptions are stream request related options
type StreamRequestOptions struct {
Interval timeMillisFlag `long:"stream-interval" description:"Interval between consecutive stream message sends, applicable separately to every stream request opened on a connection."`
DelayCloseSendStream timeMillisFlag `long:"stream-delay-close-send" description:"Delay the closure of send stream once all the request messages have been sent."`
}
// TransportOptions are transport related options.
type TransportOptions struct {
ServiceName string `short:"s" long:"service" description:"The TChannel/Hyperbahn service name"`
Peers []string `short:"p" long:"peer" description:"The host:port of the service to call"`
PeerList string `short:"P" long:"peer-list" description:"Path or URL of a JSON, YAML, or flat file containing a list of host:ports. -P? for supported protocols."`
CallerName string `long:"caller" description:"Caller will override the default caller name (which is yab-$USER)."`
RoutingKey string `long:"rk" description:"The routing key overrides the service name traffic group for proxies."`
RoutingDelegate string `long:"rd" description:"The routing delegate overrides the routing key traffic group for proxies."`
ShardKey string `long:"sk" description:"The shard key is a transport header that clues where to send a request within a clustered traffic group."`
Jaeger bool `long:"jaeger" description:"Use the Jaeger tracing client to send Uber style traces and baggage headers"`
TransportHeaders map[string]string `short:"T" long:"topt" description:"Transport options for TChannel, protocol headers for HTTP"`
HTTPMethod string `long:"http-method" description:"The HTTP method to use"`
GRPCMaxResponseSize int `long:"grpc-max-response-size" description:"Maximum response size for gRPC requests. Default value is 4MB"`
// This is a hack to work around go-flags not allowing disabling flags:
// https://github.com/jessevdk/go-flags/issues/191
// Do not specify this value in a defaults.ini file as it is not possible
// to enable Jaeger via CLI.
// Our plan is to change go-flags to support "--no-FLAG" and remove this hack.
NoJaeger bool `long:"no-jaeger" hidden:"true"`
}
// BenchmarkOptions are benchmark-specific options
type BenchmarkOptions struct {
MaxRequests int `short:"n" long:"max-requests" default:"0" description:"The maximum number of requests to make. 0 implies no limit."`
MaxDuration time.Duration `short:"d" long:"max-duration" default:"0s" description:"The maximum amount of time to run the benchmark for. 0 implies no duration limit."`
// NumCPUs is the value for GOMAXPROCS. The default value of 0 will not update GOMAXPROCS.
NumCPUs int `long:"cpus" description:"The number of OS threads"`
Connections int `long:"connections" description:"The number of TCP connections to use"`
WarmupRequests int `long:"warmup" description:"The number of requests to make to warmup each connection" default:"10"`
Concurrency int `long:"concurrency" default:"1" description:"The number of concurrent calls per connection"`
RPS int `long:"rps" default:"0" description:"Limit on the number of requests per second. The default (0) is no limit."`
// Benchmark metrics can optionally be reported via statsd.
StatsdHostPort string `long:"statsd" description:"Optional host:port of a StatsD server to report metrics"`
PerPeerStats bool `long:"per-peer-stats" description:"Whether to emit stats by peer rather than aggregated"`
Format string `long:"format" description:"Prints benchmark output in either text or JSON format. Default is text."`
}
func newOptions() *Options {
var opts Options
// Defaults
opts.ROpts.Timeout = timeMillisFlag(time.Second)
opts.TOpts.HTTPMethod = "POST"
// Set flag aliases
opts.ROpts.MethodName.dest = &opts.ROpts.Procedure
aliases := &opts.ROpts.Aliases
aliases.Arg1.dest = &opts.ROpts.Procedure
aliases.Endpoint.dest = &opts.ROpts.Procedure
aliases.Arg2.dest = &opts.ROpts.HeadersJSON
aliases.Arg3.dest = &opts.ROpts.RequestJSON
aliases.Body.dest = &opts.ROpts.RequestJSON
return &opts
}
type timeMillisFlag time.Duration
func (t *timeMillisFlag) setDuration(d time.Duration) {
*t = timeMillisFlag(d)
}
func (t timeMillisFlag) Duration() time.Duration {
return time.Duration(t)
}
func (t timeMillisFlag) String() string {
return time.Duration(t).String()
}
func (t *timeMillisFlag) UnmarshalFlag(value string) error {
valueInt, err := strconv.Atoi(value)
if err == nil {
// We received a number without a unit, assume milliseconds.
t.setDuration(time.Duration(valueInt) * time.Millisecond)
return nil
}
d, err := time.ParseDuration(value)
if err != nil {
return err
}
t.setDuration(d)
return nil
}
var errStringAliasMissing = errors.New("string alias missing destination")
type stringAlias struct {
dest *string
}
func (s *stringAlias) UnmarshalFlag(value string) error {
if s.dest == nil {
return errStringAliasMissing
}
*s.dest = value
return nil
}
func setEncodingOptions(opts *Options) {
if opts.ROpts.Aliases.JSON {
opts.ROpts.Encoding = encoding.JSON
}
if opts.ROpts.Aliases.Raw {
opts.ROpts.Encoding = encoding.Raw
}
}