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