xref: /aosp_15_r20/external/grpc-grpc/src/ruby/qps/client.rb (revision cc02d7e222339f7a4f6ba5f422e6413f4bd931f2)
1*cc02d7e2SAndroid Build Coastguard Worker#!/usr/bin/env ruby
2*cc02d7e2SAndroid Build Coastguard Worker
3*cc02d7e2SAndroid Build Coastguard Worker# Copyright 2016 gRPC authors.
4*cc02d7e2SAndroid Build Coastguard Worker#
5*cc02d7e2SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License");
6*cc02d7e2SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License.
7*cc02d7e2SAndroid Build Coastguard Worker# You may obtain a copy of the License at
8*cc02d7e2SAndroid Build Coastguard Worker#
9*cc02d7e2SAndroid Build Coastguard Worker#     http://www.apache.org/licenses/LICENSE-2.0
10*cc02d7e2SAndroid Build Coastguard Worker#
11*cc02d7e2SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software
12*cc02d7e2SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS,
13*cc02d7e2SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14*cc02d7e2SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and
15*cc02d7e2SAndroid Build Coastguard Worker# limitations under the License.
16*cc02d7e2SAndroid Build Coastguard Worker
17*cc02d7e2SAndroid Build Coastguard Worker# Worker and worker service implementation
18*cc02d7e2SAndroid Build Coastguard Worker
19*cc02d7e2SAndroid Build Coastguard Workerthis_dir = File.expand_path(File.dirname(__FILE__))
20*cc02d7e2SAndroid Build Coastguard Workerlib_dir = File.join(File.dirname(this_dir), 'lib')
21*cc02d7e2SAndroid Build Coastguard Worker$LOAD_PATH.unshift(lib_dir) unless $LOAD_PATH.include?(lib_dir)
22*cc02d7e2SAndroid Build Coastguard Worker$LOAD_PATH.unshift(this_dir) unless $LOAD_PATH.include?(this_dir)
23*cc02d7e2SAndroid Build Coastguard Worker
24*cc02d7e2SAndroid Build Coastguard Workerrequire 'grpc'
25*cc02d7e2SAndroid Build Coastguard Workerrequire 'histogram'
26*cc02d7e2SAndroid Build Coastguard Workerrequire 'src/proto/grpc/testing/benchmark_service_services_pb'
27*cc02d7e2SAndroid Build Coastguard Worker
28*cc02d7e2SAndroid Build Coastguard Workerclass Poisson
29*cc02d7e2SAndroid Build Coastguard Worker  def interarrival
30*cc02d7e2SAndroid Build Coastguard Worker    @lambda_recip * (-Math.log(1.0-rand))
31*cc02d7e2SAndroid Build Coastguard Worker  end
32*cc02d7e2SAndroid Build Coastguard Worker  def advance
33*cc02d7e2SAndroid Build Coastguard Worker    t = @next_time
34*cc02d7e2SAndroid Build Coastguard Worker    @next_time += interarrival
35*cc02d7e2SAndroid Build Coastguard Worker    t
36*cc02d7e2SAndroid Build Coastguard Worker  end
37*cc02d7e2SAndroid Build Coastguard Worker  def initialize(lambda)
38*cc02d7e2SAndroid Build Coastguard Worker    @lambda_recip = 1.0/lambda
39*cc02d7e2SAndroid Build Coastguard Worker    @next_time = Time.now + interarrival
40*cc02d7e2SAndroid Build Coastguard Worker  end
41*cc02d7e2SAndroid Build Coastguard Workerend
42*cc02d7e2SAndroid Build Coastguard Worker
43*cc02d7e2SAndroid Build Coastguard Workerclass BenchmarkClient
44*cc02d7e2SAndroid Build Coastguard Worker  def initialize(config)
45*cc02d7e2SAndroid Build Coastguard Worker    opts = {}
46*cc02d7e2SAndroid Build Coastguard Worker    if config.security_params
47*cc02d7e2SAndroid Build Coastguard Worker      if config.security_params.use_test_ca
48*cc02d7e2SAndroid Build Coastguard Worker        certs = load_test_certs
49*cc02d7e2SAndroid Build Coastguard Worker        cred = GRPC::Core::ChannelCredentials.new(certs[0])
50*cc02d7e2SAndroid Build Coastguard Worker      else
51*cc02d7e2SAndroid Build Coastguard Worker        cred = GRPC::Core::ChannelCredentials.new()
52*cc02d7e2SAndroid Build Coastguard Worker      end
53*cc02d7e2SAndroid Build Coastguard Worker      if config.security_params.server_host_override
54*cc02d7e2SAndroid Build Coastguard Worker        channel_args = {}
55*cc02d7e2SAndroid Build Coastguard Worker        channel_args[GRPC::Core::Channel::SSL_TARGET] =
56*cc02d7e2SAndroid Build Coastguard Worker          config.security_params.server_host_override
57*cc02d7e2SAndroid Build Coastguard Worker        opts[:channel_args] = channel_args
58*cc02d7e2SAndroid Build Coastguard Worker      end
59*cc02d7e2SAndroid Build Coastguard Worker    else
60*cc02d7e2SAndroid Build Coastguard Worker      cred = :this_channel_is_insecure
61*cc02d7e2SAndroid Build Coastguard Worker    end
62*cc02d7e2SAndroid Build Coastguard Worker    @histres = config.histogram_params.resolution
63*cc02d7e2SAndroid Build Coastguard Worker    @histmax = config.histogram_params.max_possible
64*cc02d7e2SAndroid Build Coastguard Worker    @start_time = Time.now
65*cc02d7e2SAndroid Build Coastguard Worker    @histogram = Histogram.new(@histres, @histmax)
66*cc02d7e2SAndroid Build Coastguard Worker    @done = false
67*cc02d7e2SAndroid Build Coastguard Worker
68*cc02d7e2SAndroid Build Coastguard Worker    gtsr = Grpc::Testing::SimpleRequest
69*cc02d7e2SAndroid Build Coastguard Worker    gtpt = Grpc::Testing::PayloadType
70*cc02d7e2SAndroid Build Coastguard Worker    gtp = Grpc::Testing::Payload
71*cc02d7e2SAndroid Build Coastguard Worker    simple_params = config.payload_config.simple_params
72*cc02d7e2SAndroid Build Coastguard Worker    req = gtsr.new(response_type: gtpt::COMPRESSABLE,
73*cc02d7e2SAndroid Build Coastguard Worker                   response_size: simple_params.resp_size,
74*cc02d7e2SAndroid Build Coastguard Worker                   payload: gtp.new(type: gtpt::COMPRESSABLE,
75*cc02d7e2SAndroid Build Coastguard Worker                                    body: nulls(simple_params.req_size)))
76*cc02d7e2SAndroid Build Coastguard Worker
77*cc02d7e2SAndroid Build Coastguard Worker    @child_threads = []
78*cc02d7e2SAndroid Build Coastguard Worker
79*cc02d7e2SAndroid Build Coastguard Worker    (0..config.client_channels-1).each do |chan|
80*cc02d7e2SAndroid Build Coastguard Worker      gtbss = Grpc::Testing::BenchmarkService::Stub
81*cc02d7e2SAndroid Build Coastguard Worker      st = config.server_targets
82*cc02d7e2SAndroid Build Coastguard Worker      stub = gtbss.new(st[chan % st.length], cred, **opts)
83*cc02d7e2SAndroid Build Coastguard Worker      (0..config.outstanding_rpcs_per_channel-1).each do |r|
84*cc02d7e2SAndroid Build Coastguard Worker        @child_threads << Thread.new {
85*cc02d7e2SAndroid Build Coastguard Worker          case config.load_params.load.to_s
86*cc02d7e2SAndroid Build Coastguard Worker          when 'closed_loop'
87*cc02d7e2SAndroid Build Coastguard Worker            waiter = nil
88*cc02d7e2SAndroid Build Coastguard Worker          when 'poisson'
89*cc02d7e2SAndroid Build Coastguard Worker            waiter = Poisson.new(config.load_params.poisson.offered_load /
90*cc02d7e2SAndroid Build Coastguard Worker                                 (config.client_channels *
91*cc02d7e2SAndroid Build Coastguard Worker                                  config.outstanding_rpcs_per_channel))
92*cc02d7e2SAndroid Build Coastguard Worker          end
93*cc02d7e2SAndroid Build Coastguard Worker          case config.rpc_type
94*cc02d7e2SAndroid Build Coastguard Worker          when :UNARY
95*cc02d7e2SAndroid Build Coastguard Worker            unary_ping_ponger(req,stub,config,waiter)
96*cc02d7e2SAndroid Build Coastguard Worker          when :STREAMING
97*cc02d7e2SAndroid Build Coastguard Worker            streaming_ping_ponger(req,stub,config,waiter)
98*cc02d7e2SAndroid Build Coastguard Worker          end
99*cc02d7e2SAndroid Build Coastguard Worker        }
100*cc02d7e2SAndroid Build Coastguard Worker      end
101*cc02d7e2SAndroid Build Coastguard Worker    end
102*cc02d7e2SAndroid Build Coastguard Worker  end
103*cc02d7e2SAndroid Build Coastguard Worker  def wait_to_issue(waiter)
104*cc02d7e2SAndroid Build Coastguard Worker    if waiter
105*cc02d7e2SAndroid Build Coastguard Worker      delay = waiter.advance-Time.now
106*cc02d7e2SAndroid Build Coastguard Worker      sleep delay if delay > 0
107*cc02d7e2SAndroid Build Coastguard Worker    end
108*cc02d7e2SAndroid Build Coastguard Worker  end
109*cc02d7e2SAndroid Build Coastguard Worker  def unary_ping_ponger(req, stub, config,waiter)
110*cc02d7e2SAndroid Build Coastguard Worker    while !@done
111*cc02d7e2SAndroid Build Coastguard Worker      wait_to_issue(waiter)
112*cc02d7e2SAndroid Build Coastguard Worker      start = Time.now
113*cc02d7e2SAndroid Build Coastguard Worker      resp = stub.unary_call(req)
114*cc02d7e2SAndroid Build Coastguard Worker      @histogram.add((Time.now-start)*1e9)
115*cc02d7e2SAndroid Build Coastguard Worker    end
116*cc02d7e2SAndroid Build Coastguard Worker  end
117*cc02d7e2SAndroid Build Coastguard Worker  def streaming_ping_ponger(req, stub, config, waiter)
118*cc02d7e2SAndroid Build Coastguard Worker    q = EnumeratorQueue.new(self)
119*cc02d7e2SAndroid Build Coastguard Worker    resp = stub.streaming_call(q.each_item)
120*cc02d7e2SAndroid Build Coastguard Worker    start = Time.now
121*cc02d7e2SAndroid Build Coastguard Worker    q.push(req)
122*cc02d7e2SAndroid Build Coastguard Worker    pushed_sentinal = false
123*cc02d7e2SAndroid Build Coastguard Worker    resp.each do |r|
124*cc02d7e2SAndroid Build Coastguard Worker      @histogram.add((Time.now-start)*1e9)
125*cc02d7e2SAndroid Build Coastguard Worker      if !@done
126*cc02d7e2SAndroid Build Coastguard Worker        wait_to_issue(waiter)
127*cc02d7e2SAndroid Build Coastguard Worker        start = Time.now
128*cc02d7e2SAndroid Build Coastguard Worker        q.push(req)
129*cc02d7e2SAndroid Build Coastguard Worker      else
130*cc02d7e2SAndroid Build Coastguard Worker        q.push(self) unless pushed_sentinal
131*cc02d7e2SAndroid Build Coastguard Worker	# Continue polling on the responses to consume and release resources
132*cc02d7e2SAndroid Build Coastguard Worker        pushed_sentinal = true
133*cc02d7e2SAndroid Build Coastguard Worker      end
134*cc02d7e2SAndroid Build Coastguard Worker    end
135*cc02d7e2SAndroid Build Coastguard Worker  end
136*cc02d7e2SAndroid Build Coastguard Worker  def mark(reset)
137*cc02d7e2SAndroid Build Coastguard Worker    lat = Grpc::Testing::HistogramData.new(
138*cc02d7e2SAndroid Build Coastguard Worker      bucket: @histogram.contents,
139*cc02d7e2SAndroid Build Coastguard Worker      min_seen: @histogram.minimum,
140*cc02d7e2SAndroid Build Coastguard Worker      max_seen: @histogram.maximum,
141*cc02d7e2SAndroid Build Coastguard Worker      sum: @histogram.sum,
142*cc02d7e2SAndroid Build Coastguard Worker      sum_of_squares: @histogram.sum_of_squares,
143*cc02d7e2SAndroid Build Coastguard Worker      count: @histogram.count
144*cc02d7e2SAndroid Build Coastguard Worker    )
145*cc02d7e2SAndroid Build Coastguard Worker    elapsed = Time.now-@start_time
146*cc02d7e2SAndroid Build Coastguard Worker    if reset
147*cc02d7e2SAndroid Build Coastguard Worker      @start_time = Time.now
148*cc02d7e2SAndroid Build Coastguard Worker      @histogram = Histogram.new(@histres, @histmax)
149*cc02d7e2SAndroid Build Coastguard Worker    end
150*cc02d7e2SAndroid Build Coastguard Worker    Grpc::Testing::ClientStats.new(latencies: lat, time_elapsed: elapsed)
151*cc02d7e2SAndroid Build Coastguard Worker  end
152*cc02d7e2SAndroid Build Coastguard Worker  def shutdown
153*cc02d7e2SAndroid Build Coastguard Worker    @done = true
154*cc02d7e2SAndroid Build Coastguard Worker    @child_threads.each do |thread|
155*cc02d7e2SAndroid Build Coastguard Worker      thread.join
156*cc02d7e2SAndroid Build Coastguard Worker    end
157*cc02d7e2SAndroid Build Coastguard Worker  end
158*cc02d7e2SAndroid Build Coastguard Workerend
159