ritsudo 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 6fefbc720cfbbde080977f121b0f1c5fb6fe7f1a29ef11d8af1f20b4764b20c6
4
+ data.tar.gz: 5e0cb3a931913cb2a3e6180d32107ec043a37d33c11b6285f7b7858d41339fe4
5
+ SHA512:
6
+ metadata.gz: a1827dcc590467f8e5df1c994933757a651137d22fc2ac5a51a78e8e5167715bdfb5e0f238cc1549223d79b9202dc13519e9cc7f4ef8a06d211cc7fbbb5ea689
7
+ data.tar.gz: 8036fbba03c285161b4e01709b95b6ee42140a2e8c93969d2d80f400d8593357e02e85542f9b53839111c77088ff4b9f441923a71754d729514e2effff7d6eb7
data/.gitignore ADDED
@@ -0,0 +1,6 @@
1
+ **/*.swp
2
+ **/*.log
3
+ .byebug_history
4
+ dev.rb
5
+ Gemfile.lock
6
+
data/Gemfile ADDED
@@ -0,0 +1,10 @@
1
+ source "https://rubygems.org"
2
+
3
+ git_source(:github) {|repo_name| "https://github.com/#{repo_name}" }
4
+
5
+ # Specify your gem's dependencies in ritsudo.gemspec
6
+ gemspec
7
+
8
+ gem 'selenium-webdriver'
9
+ gem 'formatador'
10
+ gem 'thor'
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2019 shingo morita
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,76 @@
1
+ # Ritsudo
2
+
3
+ CLI bnechmark tool based on headless chrome.
4
+ Also measure ajax requests.
5
+
6
+ ## Installation
7
+
8
+ Add this line to your application's Gemfile:
9
+
10
+ ```ruby
11
+ gem 'ritsudo'
12
+ ```
13
+
14
+ And then execute:
15
+
16
+ $ bundle
17
+
18
+ Or install it yourself as:
19
+
20
+ $ gem install ritsudo
21
+
22
+ ## Usage
23
+ ```
24
+ bundle exec bin/ritsudo --url=https://example.com -m "example.com" -u "Mozilla/5.0 (iPhone; CPU iPhone OS 6_0 like Mac OS X) AppleWebKit/536.26 (KHTML, like Gecko) Version/6.0 Mobile/10A403 Safari/8536.25"
25
+ ```
26
+
27
+
28
+ ```ruby
29
+ #!/usr/bin/env ruby
30
+ require "bundler/setup"
31
+ require "ritsudo"
32
+
33
+ ua = "Mozilla/5.0 (iPhone; CPU iPhone OS 6_0 like Mac OS X) AppleWebKit/536.26 (KHTML, like Gecko) Version/6.0 Mobile/10A403 Safari/8536.25"
34
+ collector = Ritsudo::Collector.new(match: /example\.com/)
35
+ benchmark = Ritsudo::Benchmark.new(collector: collector)
36
+ benchmark.do("http:/www.example.com", driver_options: {
37
+ user_agent: ua
38
+ })
39
+
40
+ benchmark.collector.documents.display
41
+ benchmark.collector.xhrs.display
42
+ ```
43
+
44
+ ```
45
+ [Misc]
46
+ +------------------+--------+------+-----+-------+
47
+ | name | avg | max | min | count |
48
+ +------------------+--------+------+-----+-------+
49
+ | DomContentLoaded | 858.6 | 921 | 797 | 5 |
50
+ +------------------+--------+------+-----+-------+
51
+ | Loaded | 1235.6 | 1672 | 975 | 5 |
52
+ +------------------+--------+------+-----+-------+
53
+ [Document]
54
+ +--------------------------+--------+--------+--------+-------+
55
+ | url | avg | max | min | count |
56
+ +--------------------------+--------+--------+--------+-------+
57
+ | http://www.example.com/ | 597.56 | 858.15 | 433.51 | 5 |
58
+ +--------------------------+--------+--------+--------+-------+
59
+ [XHR]
60
+ +------------------------------------------+---------+---------+---------+-------+-------------+
61
+ | url | avg | max | min | count | uncompleted |
62
+ +------------------------------------------+---------+---------+---------+-------+-------------+
63
+ | http://www.example.com/xhr_request | 300.39 | 500.39 | 250.39 | 4 | 1 |
64
+ +------------------------------------------+---------+---------+---------+-------+-------------+
65
+ | http://www.example.com/slow_xhr_request | - | - | - | - | 5 |
66
+ +------------------------------------------+---------+---------+---------+-------+-------------+
67
+ [Script]
68
+ +--------------------------------+-------+-------+-------+-------+-------------+
69
+ | url | avg | max | min | count | uncompleted |
70
+ +--------------------------------+-------+-------+-------+-------+-------------+
71
+ | https//example.com/example.js | 25.88 | 43.15 | 14.96 | 5 | 0 |
72
+ +--------------------------------+-------+-------+-------+-------+-------------+
73
+ ```
74
+ ## License
75
+
76
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,6 @@
1
+ require "bundler/gem_tasks"
2
+ require "rspec/core/rake_task"
3
+
4
+ RSpec::Core::RakeTask.new(:spec)
5
+
6
+ task :default => :spec
data/bin/ritsudo ADDED
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env ruby
2
+ # -*- coding: utf-8 -*-
3
+ # vim:fenc=utf-8 ff=unix ft=ruby ts=4 sw=2 sts=2 si et :
4
+
5
+ require 'ritsudo/cli'
6
+ Ritsudo::Cli.start(ARGV)
@@ -0,0 +1,61 @@
1
+ module Ritsudo
2
+ class Benchmark
3
+ attr_reader :collector
4
+ def initialize(collector: Ritsudo::Collector.new)
5
+ @collector = collector
6
+ end
7
+
8
+ def do(url, count: 5, wait: 1, sub_process_timeout: 3,
9
+ driver_options: {
10
+ timeout: 5,
11
+ wait_time: 1
12
+ })
13
+ puts "Ritsudo requests #{count} times: #{url}"
14
+ count.times do
15
+ print(".")
16
+ collect(url, wait: wait, sub_process_timeout: sub_process_timeout, driver_options: driver_options)
17
+ end
18
+ puts ""
19
+ end
20
+
21
+ def collect_requests_wait_finish(driver, sub_process_timeout:)
22
+ messages = driver.manage.logs.get('performance')
23
+ requests = Ritsudo::Request.grouping(messages)
24
+ Timeout.timeout(sub_process_timeout) do
25
+ while requests.find(&:processing?)
26
+ sleep(0.5)
27
+ messages = driver.manage.logs.get('performance') + messages
28
+ requests = Ritsudo::Request.grouping(messages)
29
+ end
30
+ end
31
+ requests
32
+ rescue Timeout::Error
33
+ requests
34
+ end
35
+
36
+ def drive(driver_options, &block)
37
+ driver = Ritsudo::Driver.new(driver_options)
38
+ yield(driver)
39
+ driver.driver.close()
40
+ end
41
+
42
+ def collect(url, wait:, sub_process_timeout:, driver_options:)
43
+ drive(driver_options) do |driver|
44
+ driver.get(url)
45
+ sleep(wait)
46
+ sleep(1) until driver.driver.execute_script("return window.performance.timing.loadEventEnd")
47
+
48
+ requests = collect_requests_wait_finish(driver, sub_process_timeout: sub_process_timeout)
49
+ @collector.add(requests)
50
+
51
+ dom_content_loaded_script = "return window.performance.timing.domContentLoadedEventEnd - window.performance.timing.requestStart"
52
+ dom_content_loaded = driver.driver.execute_script(dom_content_loaded_script)
53
+ @collector.add_misc("Misc", "DomContentLoaded", dom_content_loaded)
54
+
55
+ loaded_script = "return window.performance.timing.loadEventEnd - window.performance.timing.requestStart"
56
+ loaded = driver.driver.execute_script(loaded_script)
57
+ @collector.add_misc("Misc", "Loaded", loaded)
58
+ end
59
+ end
60
+ end
61
+ end
@@ -0,0 +1,30 @@
1
+ require "ritsudo"
2
+ require 'thor'
3
+ module Ritsudo
4
+ class Cli < Thor
5
+ default_command :benchmark
6
+ option :url, required: true, aliases: ['-a']
7
+ option :count, default: 5, type: :numeric, aliases: ['-c']
8
+ option :sub_process_timeout, default: 5, type: :numeric, aliases: ['-s']
9
+ option :timeout, default: 10, type: :numeric, aliases: ['-t']
10
+ option :wait_time, default: 1, type: :numeric, aliases: ['-w']
11
+ option :ua, type: :string, aliases: ['-u']
12
+ option :match, type: :string, aliases: ['-m']
13
+ desc "benchmark URL", "benchmark"
14
+ def benchmark()
15
+ match = options[:match] ? Regexp.new(options[:match]) : nil
16
+ collector = Ritsudo::Collector.new(match: match)
17
+ benchmark = Ritsudo::Benchmark.new(collector: collector)
18
+ benchmark.do(options[:url],
19
+ count: options[:count],
20
+ sub_process_timeout: options[:sub_process_timeout],
21
+ driver_options: {
22
+ timeout: options[:timeout],
23
+ wait_time: options[:wait_time],
24
+ user_agent: options[:ua]
25
+ }
26
+ )
27
+ benchmark.collector.report
28
+ end
29
+ end
30
+ end
@@ -0,0 +1,43 @@
1
+ module Ritsudo
2
+ class Collector
3
+ attr_reader :documents, :xhrs, :scripts
4
+
5
+ def initialize(documents: Ritsudo::Result::Documents.new,
6
+ xhrs: Ritsudo::Result::Xhrs.new,
7
+ scripts: Ritsudo::Result::Scripts.new,
8
+ misc: Ritsudo::Result::Misc.new,
9
+ match: nil)
10
+ @documents = documents
11
+ @xhrs = xhrs
12
+ @scripts = scripts
13
+ @misc = misc
14
+ @match = match
15
+ end
16
+
17
+ def add(_requests)
18
+ if @match
19
+ requests =_requests.select do |request|
20
+ @match =~ request.url
21
+ end
22
+ else
23
+ requests = _requests
24
+ end
25
+
26
+ types = requests.group_by(&:type)
27
+ documents.add_multiple(types["Document"]) if types["Document"]
28
+ xhrs.add_multiple(types["XHR"]) if types["XHR"]
29
+ scripts.add_multiple(types["Script"]) if types["Script"]
30
+ end
31
+
32
+ def add_misc(group, name, value)
33
+ @misc.add(group, name, value)
34
+ end
35
+
36
+ def report
37
+ @misc.report
38
+ @documents.report
39
+ @xhrs.report
40
+ @scripts.report
41
+ end
42
+ end
43
+ end
@@ -0,0 +1,55 @@
1
+ require 'selenium-webdriver'
2
+ require 'forwardable'
3
+
4
+ module Ritsudo
5
+ class Driver
6
+ extend Forwardable
7
+ def_delegators :@driver, :get, :get_log, :manage
8
+
9
+ def driver
10
+ @driver
11
+ end
12
+
13
+ def initialize(logger_level: :warn, timeout: 60, wait_time: 5,
14
+ logger_output: "./ritsudo.selenium.log",
15
+ user_agent: nil,
16
+ args: ['--headless',
17
+ '--window-size=1920,1080',
18
+ '--ignore-certificate-errors',
19
+ '--disable-popup-blocking',
20
+ '--disable-translate',
21
+ '--blink-settings=imagesEnabled=false'])
22
+
23
+ if user_agent
24
+ args << "--user-agent=#{user_agent}"
25
+ end
26
+
27
+ Selenium::WebDriver.logger.output = logger_output
28
+ Selenium::WebDriver.logger.level = logger_level
29
+ client = Selenium::WebDriver::Remote::Http::Default.new.tap { |c| c.read_timeout = timeout }
30
+ @driver = Selenium::WebDriver.for(:chrome, options: options(args),
31
+ desired_capabilities: caps,
32
+ http_client: client,).tap do |d|
33
+ d.manage.timeouts.implicit_wait = timeout
34
+ end
35
+ #@wait = Selenium::WebDriver::Wait.new(timeout: wait_time)
36
+ end
37
+
38
+ def caps
39
+ Selenium::WebDriver::Remote::Capabilities.chrome(
40
+ loggingPrefs: { performance: 'ALL' },
41
+ chromeOptions: {
42
+ perfLoggingPrefs: {
43
+ enableNetwork: true
44
+ }
45
+ }
46
+ )
47
+ end
48
+ private :caps
49
+
50
+ def options(args)
51
+ Selenium::WebDriver::Chrome::Options.new(args: args)
52
+ end
53
+ private :options
54
+ end
55
+ end
@@ -0,0 +1,38 @@
1
+ module Ritsudo
2
+ class Message
3
+ attr_reader :message
4
+
5
+ def self.wrap(raw_messages)
6
+ messages = raw_messages.map { |raw|
7
+ JSON.parse(raw.message)["message"]
8
+ }.reject { |message|
9
+ message["method"] == "Network.dataReceived" # don't use received info
10
+ }
11
+ messages.map { |message| Ritsudo::Message.new(message) }
12
+ end
13
+
14
+ def initialize(message)
15
+ @message = message
16
+ end
17
+
18
+ def method
19
+ message["method"]
20
+ end
21
+
22
+ def timestamp
23
+ param('timestamp')
24
+ end
25
+
26
+ def url
27
+ param("request")&.[]("url") || 'none'
28
+ end
29
+
30
+ def params
31
+ message["params"]
32
+ end
33
+
34
+ def param(name)
35
+ params&.[](name)
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,58 @@
1
+ module Ritsudo
2
+ class Request
3
+ attr_reader :messages
4
+ attr_reader :request_id
5
+
6
+ def self.grouping(raw_messages)
7
+ messages = Ritsudo::Message.wrap(raw_messages)
8
+ requests = messages.group_by { |message|
9
+ message.param('requestId')
10
+ }.map { |request_id, messages|
11
+ Ritsudo::Request.new(request_id, messages)
12
+ }
13
+ requests
14
+ end
15
+
16
+ def initialize(request_id, messages)
17
+ @request_id = request_id
18
+ @messages = messages
19
+ end
20
+
21
+ def time
22
+ return nil unless received_message && sent_message
23
+ ((received_message.timestamp - sent_message.timestamp) * 1000) # msec
24
+ end
25
+
26
+ def url
27
+ if _url = sent_message&.url
28
+ url = URI.parse(_url)
29
+ "#{url.scheme}//#{url.host}#{url.path}"
30
+ else
31
+ nil
32
+ end
33
+ end
34
+
35
+ def processing?
36
+ sent_message && !finished_message
37
+ end
38
+
39
+ def sent_message
40
+ @messages.find { |message| message.method == "Network.requestWillBeSent" }
41
+ end
42
+ private :sent_message
43
+
44
+ def finished_message
45
+ @messages.find { |event| event.method == "Network.loadingFinished" }
46
+ end
47
+ private :finished_message
48
+
49
+ def received_message
50
+ @messages.find { |event| event.method == "Network.responseReceived" }
51
+ end
52
+ private :received_message
53
+
54
+ def type
55
+ sent_message&.param("type") || 'etc'
56
+ end
57
+ end
58
+ end
@@ -0,0 +1,17 @@
1
+ module Ritsudo
2
+ module Result
3
+ class Base
4
+ def initialize
5
+ @completed = Hash.new { |h,k| h[k] = [] }
6
+ end
7
+
8
+ def add_multiple(requests)
9
+ requests.each { |request| add(request) }
10
+ end
11
+
12
+ def add(request)
13
+ raise NotImplementedError
14
+ end
15
+ end
16
+ end
17
+ end
@@ -0,0 +1,24 @@
1
+ module Ritsudo
2
+ module Result
3
+ class Documents < Base
4
+ def add(request)
5
+ @completed[request.url] << request.time
6
+ end
7
+
8
+ def report
9
+ table_data = []
10
+ puts "[Document]"
11
+ @completed.each do |url, complete|
12
+ table_data << {
13
+ url: url[0..100],
14
+ avg: (complete.sum(0.0) / complete.length).round(2),
15
+ max: complete.max.round(2),
16
+ min: complete.min.round(2),
17
+ count: complete.size
18
+ }
19
+ end
20
+ Formatador.display_table(table_data, [:url, :avg, :max, :min, :count])
21
+ end
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,32 @@
1
+ module Ritsudo
2
+ module Result
3
+ class Misc
4
+ def initialize
5
+ @misc = {}
6
+ end
7
+
8
+ def add(group, name, value)
9
+ @misc[group] ||= {}
10
+ @misc[group][name] ||= []
11
+ @misc[group][name] << value
12
+ end
13
+
14
+ def report
15
+ table_data = []
16
+ @misc.each do |group, name_and_values|
17
+ puts "[#{group}]"
18
+ name_and_values.each do |name, values|
19
+ table_data << {
20
+ name: name,
21
+ avg: (values.sum(0.0) / values.length).round(2),
22
+ max: values.max.round(2),
23
+ min: values.min.round(2),
24
+ count: values.size
25
+ }
26
+ end
27
+ Formatador.display_table(table_data, [:name, :avg, :max, :min, :count])
28
+ end
29
+ end
30
+ end
31
+ end
32
+ end
@@ -0,0 +1,11 @@
1
+ module Ritsudo
2
+ module Result
3
+ class Scripts < Base
4
+ include Ritsudo::Result::Uncompletable
5
+
6
+ def name
7
+ "Script"
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,46 @@
1
+ module Ritsudo
2
+ module Result
3
+ module Uncompletable
4
+ def add(request)
5
+ @requests ||= []
6
+ @uncompleted ||= Hash.new
7
+ @requests << request.url
8
+ @uncompleted[request.url] ||= 0
9
+ if request.time
10
+ @completed[request.url] << request.time
11
+ else
12
+ @uncompleted[request.url] += 1
13
+ end
14
+ end
15
+
16
+ def name
17
+ raise NotImplementedError
18
+ end
19
+
20
+ def report
21
+ puts "[#{name}]"
22
+ table_data = []
23
+ (@requests || []).uniq.each do |url|
24
+ if @completed[url] && !@completed[url].empty?
25
+ avg = (@completed[url].sum(0.0) / @completed[url].length).round(2)
26
+ max = @completed[url].max.round(2)
27
+ min = @completed[url].min.round(2)
28
+ count = @completed[url].size
29
+ else
30
+ avg = max = min = count = "-"
31
+ end
32
+
33
+ table_data << {
34
+ url: url[0..100],
35
+ avg: avg,
36
+ max: max,
37
+ min: min,
38
+ count: count,
39
+ uncompleted: @uncompleted[url]
40
+ }
41
+ end
42
+ Formatador.display_table(table_data, [:url, :avg, :max, :min, :count, :uncompleted])
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,10 @@
1
+ module Ritsudo
2
+ module Result
3
+ class Xhrs < Base
4
+ include Ritsudo::Result::Uncompletable
5
+ def name
6
+ "XHR"
7
+ end
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,4 @@
1
+ module Ritsudo
2
+ module Result
3
+ end
4
+ end
@@ -0,0 +1,3 @@
1
+ module Ritsudo
2
+ VERSION = "0.1.0"
3
+ end
data/lib/ritsudo.rb ADDED
@@ -0,0 +1,21 @@
1
+ require 'json'
2
+ require 'formatador'
3
+
4
+ require 'ritsudo/version'
5
+ require 'ritsudo/driver'
6
+ require 'ritsudo/result'
7
+ require 'ritsudo/message'
8
+ require 'ritsudo/request'
9
+ require 'ritsudo/benchmark'
10
+ require 'ritsudo/collector'
11
+ require 'ritsudo/result'
12
+ require 'ritsudo/result/base'
13
+ require 'ritsudo/result/uncompletable'
14
+ require 'ritsudo/result/misc'
15
+ require 'ritsudo/result/documents'
16
+ require 'ritsudo/result/xhrs'
17
+ require 'ritsudo/result/scripts'
18
+
19
+ module Ritsudo
20
+ class Error < StandardError; end
21
+ end
data/ritsudo.gemspec ADDED
@@ -0,0 +1,44 @@
1
+
2
+ lib = File.expand_path("../lib", __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require "ritsudo/version"
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "ritsudo"
8
+ spec.version = Ritsudo::VERSION
9
+ spec.authors = ["shingo morita"]
10
+ spec.email = ["morita.shingo@gmail.com"]
11
+
12
+ spec.summary = "CLI bnechmark tool based on headless chrome"
13
+ spec.description = "CLI bnechmark tool based on headless chrome. Also measure ajax requests."
14
+ spec.homepage = "https://github.com/eudoxa/ritsudo"
15
+ spec.license = "MIT"
16
+
17
+ # Prevent pushing this gem to RubyGems.org. To allow pushes either set the 'allowed_push_host'
18
+ # to allow pushing to a single host or delete this section to allow pushing to any host.
19
+ if spec.respond_to?(:metadata)
20
+ spec.metadata["homepage_uri"] = spec.homepage
21
+ else
22
+ raise "RubyGems 2.0 or newer is required to protect against " \
23
+ "public gem pushes."
24
+ end
25
+
26
+ # Specify which files should be added to the gem when it is released.
27
+ # The `git ls-files -z` loads the files in the RubyGem that have been added into git.
28
+ spec.files = Dir.chdir(File.expand_path('..', __FILE__)) do
29
+ `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
30
+ end
31
+ spec.bindir = "bin"
32
+ spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
33
+ spec.require_paths = ["lib"]
34
+
35
+ spec.add_runtime_dependency 'selenium-webdriver'
36
+ spec.add_runtime_dependency 'formatador'
37
+ spec.add_runtime_dependency 'thor'
38
+
39
+ spec.add_development_dependency "bundler", "~> 1.17"
40
+ spec.add_development_dependency "rake", "~> 10.0"
41
+ spec.add_development_dependency "rspec", "~> 3.0"
42
+ spec.add_development_dependency "byebug"
43
+ spec.add_development_dependency "formatador"
44
+ end
metadata ADDED
@@ -0,0 +1,179 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: ritsudo
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - shingo morita
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2019-03-21 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: selenium-webdriver
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: formatador
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: thor
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: bundler
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '1.17'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '1.17'
69
+ - !ruby/object:Gem::Dependency
70
+ name: rake
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '10.0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '10.0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: rspec
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - "~>"
88
+ - !ruby/object:Gem::Version
89
+ version: '3.0'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - "~>"
95
+ - !ruby/object:Gem::Version
96
+ version: '3.0'
97
+ - !ruby/object:Gem::Dependency
98
+ name: byebug
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - ">="
102
+ - !ruby/object:Gem::Version
103
+ version: '0'
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - ">="
109
+ - !ruby/object:Gem::Version
110
+ version: '0'
111
+ - !ruby/object:Gem::Dependency
112
+ name: formatador
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - ">="
116
+ - !ruby/object:Gem::Version
117
+ version: '0'
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - ">="
123
+ - !ruby/object:Gem::Version
124
+ version: '0'
125
+ description: CLI bnechmark tool based on headless chrome. Also measure ajax requests.
126
+ email:
127
+ - morita.shingo@gmail.com
128
+ executables:
129
+ - ritsudo
130
+ extensions: []
131
+ extra_rdoc_files: []
132
+ files:
133
+ - ".gitignore"
134
+ - Gemfile
135
+ - LICENSE.txt
136
+ - README.md
137
+ - Rakefile
138
+ - bin/ritsudo
139
+ - lib/ritsudo.rb
140
+ - lib/ritsudo/benchmark.rb
141
+ - lib/ritsudo/cli.rb
142
+ - lib/ritsudo/collector.rb
143
+ - lib/ritsudo/driver.rb
144
+ - lib/ritsudo/message.rb
145
+ - lib/ritsudo/request.rb
146
+ - lib/ritsudo/result.rb
147
+ - lib/ritsudo/result/base.rb
148
+ - lib/ritsudo/result/documents.rb
149
+ - lib/ritsudo/result/misc.rb
150
+ - lib/ritsudo/result/scripts.rb
151
+ - lib/ritsudo/result/uncompletable.rb
152
+ - lib/ritsudo/result/xhrs.rb
153
+ - lib/ritsudo/version.rb
154
+ - ritsudo.gemspec
155
+ homepage: https://github.com/eudoxa/ritsudo
156
+ licenses:
157
+ - MIT
158
+ metadata:
159
+ homepage_uri: https://github.com/eudoxa/ritsudo
160
+ post_install_message:
161
+ rdoc_options: []
162
+ require_paths:
163
+ - lib
164
+ required_ruby_version: !ruby/object:Gem::Requirement
165
+ requirements:
166
+ - - ">="
167
+ - !ruby/object:Gem::Version
168
+ version: '0'
169
+ required_rubygems_version: !ruby/object:Gem::Requirement
170
+ requirements:
171
+ - - ">="
172
+ - !ruby/object:Gem::Version
173
+ version: '0'
174
+ requirements: []
175
+ rubygems_version: 3.0.1
176
+ signing_key:
177
+ specification_version: 4
178
+ summary: CLI bnechmark tool based on headless chrome
179
+ test_files: []