selenium-webdriver 4.0.0.alpha5 → 4.0.0.beta3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (87) hide show
  1. checksums.yaml +4 -4
  2. data/lib/selenium/devtools.rb +30 -0
  3. data/lib/selenium/server.rb +18 -26
  4. data/lib/selenium/webdriver.rb +1 -3
  5. data/lib/selenium/webdriver/atoms/findElements.js +93 -93
  6. data/lib/selenium/webdriver/atoms/getAttribute.js +75 -59
  7. data/lib/selenium/webdriver/atoms/isDisplayed.js +72 -72
  8. data/lib/selenium/webdriver/atoms/mutationListener.js +55 -0
  9. data/lib/selenium/webdriver/chrome.rb +1 -1
  10. data/lib/selenium/webdriver/chrome/driver.rb +28 -6
  11. data/lib/selenium/webdriver/chrome/{bridge.rb → features.rb} +6 -8
  12. data/lib/selenium/webdriver/chrome/options.rb +54 -37
  13. data/lib/selenium/webdriver/chrome/profile.rb +6 -3
  14. data/lib/selenium/webdriver/chrome/service.rb +4 -2
  15. data/lib/selenium/webdriver/common.rb +7 -2
  16. data/lib/selenium/webdriver/common/driver.rb +86 -26
  17. data/lib/selenium/webdriver/common/driver_extensions/has_authentication.rb +89 -0
  18. data/lib/selenium/webdriver/common/driver_extensions/has_devtools.rb +6 -1
  19. data/lib/selenium/webdriver/common/driver_extensions/has_location.rb +5 -8
  20. data/lib/selenium/webdriver/common/driver_extensions/has_log_events.rb +149 -0
  21. data/lib/selenium/webdriver/{edge_chrome/bridge.rb → common/driver_extensions/has_logs.rb} +7 -7
  22. data/lib/selenium/webdriver/common/driver_extensions/has_network_connection.rb +6 -27
  23. data/lib/selenium/webdriver/common/driver_extensions/has_network_interception.rb +67 -0
  24. data/lib/selenium/webdriver/common/driver_extensions/has_remote_status.rb +1 -0
  25. data/lib/selenium/webdriver/common/driver_extensions/{rotatable.rb → prints_page.rb} +18 -20
  26. data/lib/selenium/webdriver/common/element.rb +66 -12
  27. data/lib/selenium/webdriver/common/interactions/interaction.rb +4 -1
  28. data/lib/selenium/webdriver/common/logger.rb +6 -3
  29. data/lib/selenium/webdriver/common/manager.rb +11 -1
  30. data/lib/selenium/webdriver/common/options.rb +90 -11
  31. data/lib/selenium/webdriver/common/platform.rb +3 -1
  32. data/lib/selenium/webdriver/common/port_prober.rb +4 -6
  33. data/lib/selenium/webdriver/common/proxy.rb +4 -1
  34. data/lib/selenium/webdriver/common/search_context.rb +4 -1
  35. data/lib/selenium/webdriver/common/service.rb +13 -114
  36. data/lib/selenium/webdriver/common/service_manager.rb +151 -0
  37. data/lib/selenium/webdriver/common/socket_poller.rb +19 -30
  38. data/lib/selenium/webdriver/common/takes_screenshot.rb +63 -0
  39. data/lib/selenium/webdriver/common/target_locator.rb +4 -4
  40. data/lib/selenium/webdriver/devtools.rb +144 -0
  41. data/lib/selenium/webdriver/devtools/console_event.rb +38 -0
  42. data/lib/selenium/webdriver/{edge_html/driver.rb → devtools/exception_event.rb} +10 -13
  43. data/lib/selenium/webdriver/devtools/mutation_event.rb +37 -0
  44. data/lib/selenium/webdriver/devtools/request.rb +57 -0
  45. data/lib/selenium/webdriver/edge.rb +7 -29
  46. data/lib/selenium/webdriver/{edge_chrome → edge}/driver.rb +10 -4
  47. data/lib/selenium/webdriver/edge/features.rb +39 -0
  48. data/lib/selenium/webdriver/{edge_chrome → edge}/options.rb +12 -3
  49. data/lib/selenium/webdriver/{edge_chrome → edge}/profile.rb +2 -2
  50. data/lib/selenium/webdriver/{edge_chrome → edge}/service.rb +2 -2
  51. data/lib/selenium/webdriver/firefox.rb +5 -1
  52. data/lib/selenium/webdriver/firefox/driver.rb +19 -3
  53. data/lib/selenium/webdriver/firefox/{bridge.rb → features.rb} +3 -3
  54. data/lib/selenium/webdriver/firefox/options.rb +25 -31
  55. data/lib/selenium/webdriver/firefox/profile.rb +12 -2
  56. data/lib/selenium/webdriver/firefox/service.rb +1 -1
  57. data/lib/selenium/webdriver/ie/driver.rb +1 -2
  58. data/lib/selenium/webdriver/ie/options.rb +7 -20
  59. data/lib/selenium/webdriver/ie/service.rb +4 -2
  60. data/lib/selenium/webdriver/remote/bridge.rb +50 -42
  61. data/lib/selenium/webdriver/remote/capabilities.rb +127 -71
  62. data/lib/selenium/webdriver/remote/commands.rb +3 -0
  63. data/lib/selenium/webdriver/remote/driver.rb +10 -3
  64. data/lib/selenium/webdriver/remote/http/common.rb +0 -5
  65. data/lib/selenium/webdriver/remote/http/default.rb +8 -7
  66. data/lib/selenium/webdriver/remote/http/persistent.rb +6 -0
  67. data/lib/selenium/webdriver/safari.rb +8 -1
  68. data/lib/selenium/webdriver/safari/driver.rb +3 -4
  69. data/lib/selenium/webdriver/safari/{bridge.rb → features.rb} +3 -3
  70. data/lib/selenium/webdriver/safari/options.rb +1 -33
  71. data/lib/selenium/webdriver/support/block_event_listener.rb +1 -1
  72. data/lib/selenium/webdriver/support/color.rb +2 -2
  73. data/lib/selenium/webdriver/support/event_firing_bridge.rb +1 -1
  74. data/lib/selenium/webdriver/support/guards.rb +95 -0
  75. data/lib/selenium/webdriver/support/guards/guard.rb +89 -0
  76. data/lib/selenium/webdriver/support/guards/guard_condition.rb +52 -0
  77. data/lib/selenium/webdriver/support/select.rb +2 -2
  78. data/lib/selenium/webdriver/version.rb +1 -1
  79. metadata +69 -32
  80. data/CHANGES +0 -1725
  81. data/Gemfile +0 -4
  82. data/LICENSE +0 -202
  83. data/README.md +0 -35
  84. data/lib/selenium/webdriver/common/driver_extensions/takes_screenshot.rb +0 -65
  85. data/lib/selenium/webdriver/edge_html/options.rb +0 -91
  86. data/lib/selenium/webdriver/edge_html/service.rb +0 -47
  87. data/selenium-webdriver.gemspec +0 -48
@@ -27,12 +27,11 @@ module Selenium
27
27
  class Profile
28
28
  include ProfileHelper
29
29
 
30
- attr_reader :directory
31
-
32
30
  def initialize(model = nil)
33
31
  @model = verify_model(model)
34
32
  @extensions = []
35
33
  @encoded_extensions = []
34
+ @directory = nil
36
35
  end
37
36
 
38
37
  def add_extension(path)
@@ -45,10 +44,14 @@ module Selenium
45
44
  @encoded_extensions << encoded
46
45
  end
47
46
 
47
+ def directory
48
+ @directory || layout_on_disk
49
+ end
50
+
48
51
  #
49
52
  # Set a preference in the profile.
50
53
  #
51
- # See https://src.chromium.org/svn/trunk/src/chrome/common/pref_names.cc
54
+ # See https://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/pref_names.cc
52
55
  #
53
56
 
54
57
  def []=(key, value)
@@ -32,14 +32,16 @@ module Selenium
32
32
 
33
33
  private
34
34
 
35
- # Note: This processing is deprecated
35
+ # NOTE: This processing is deprecated
36
36
  def extract_service_args(driver_opts)
37
37
  driver_args = super
38
38
  driver_opts = driver_opts.dup
39
39
  driver_args << "--log-path=#{driver_opts.delete(:log_path)}" if driver_opts.key?(:log_path)
40
40
  driver_args << "--url-base=#{driver_opts.delete(:url_base)}" if driver_opts.key?(:url_base)
41
41
  driver_args << "--port-server=#{driver_opts.delete(:port_server)}" if driver_opts.key?(:port_server)
42
- driver_args << "--whitelisted-ips=#{driver_opts.delete(:whitelisted_ips)}" if driver_opts.key?(:whitelisted_ips)
42
+ if driver_opts.key?(:whitelisted_ips)
43
+ driver_args << "--whitelisted-ips=#{driver_opts.delete(:whitelisted_ips)}"
44
+ end
43
45
  driver_args << "--verbose" if driver_opts.key?(:verbose)
44
46
  driver_args << "--silent" if driver_opts.key?(:silent)
45
47
  driver_args
@@ -23,6 +23,7 @@ require 'selenium/webdriver/common/proxy'
23
23
  require 'selenium/webdriver/common/log_entry'
24
24
  require 'selenium/webdriver/common/file_reaper'
25
25
  require 'selenium/webdriver/common/service'
26
+ require 'selenium/webdriver/common/service_manager'
26
27
  require 'selenium/webdriver/common/socket_lock'
27
28
  require 'selenium/webdriver/common/socket_poller'
28
29
  require 'selenium/webdriver/common/port_prober'
@@ -49,8 +50,6 @@ require 'selenium/webdriver/common/action_builder'
49
50
  require 'selenium/webdriver/common/html5/shared_web_storage'
50
51
  require 'selenium/webdriver/common/html5/local_storage'
51
52
  require 'selenium/webdriver/common/html5/session_storage'
52
- require 'selenium/webdriver/common/driver_extensions/takes_screenshot'
53
- require 'selenium/webdriver/common/driver_extensions/rotatable'
54
53
  require 'selenium/webdriver/common/driver_extensions/has_web_storage'
55
54
  require 'selenium/webdriver/common/driver_extensions/downloads_files'
56
55
  require 'selenium/webdriver/common/driver_extensions/has_location'
@@ -58,13 +57,19 @@ require 'selenium/webdriver/common/driver_extensions/has_session_id'
58
57
  require 'selenium/webdriver/common/driver_extensions/has_remote_status'
59
58
  require 'selenium/webdriver/common/driver_extensions/has_network_conditions'
60
59
  require 'selenium/webdriver/common/driver_extensions/has_network_connection'
60
+ require 'selenium/webdriver/common/driver_extensions/has_network_interception'
61
61
  require 'selenium/webdriver/common/driver_extensions/has_permissions'
62
62
  require 'selenium/webdriver/common/driver_extensions/has_debugger'
63
+ require 'selenium/webdriver/common/driver_extensions/prints_page'
63
64
  require 'selenium/webdriver/common/driver_extensions/uploads_files'
64
65
  require 'selenium/webdriver/common/driver_extensions/has_addons'
65
66
  require 'selenium/webdriver/common/driver_extensions/has_devtools'
67
+ require 'selenium/webdriver/common/driver_extensions/has_authentication'
68
+ require 'selenium/webdriver/common/driver_extensions/has_logs'
69
+ require 'selenium/webdriver/common/driver_extensions/has_log_events'
66
70
  require 'selenium/webdriver/common/keys'
67
71
  require 'selenium/webdriver/common/profile_helper'
68
72
  require 'selenium/webdriver/common/options'
73
+ require 'selenium/webdriver/common/takes_screenshot'
69
74
  require 'selenium/webdriver/common/driver'
70
75
  require 'selenium/webdriver/common/element'
@@ -30,6 +30,7 @@ module Selenium
30
30
 
31
31
  class Driver
32
32
  include SearchContext
33
+ include TakesScreenshot
33
34
 
34
35
  class << self
35
36
  #
@@ -43,21 +44,17 @@ module Selenium
43
44
  def for(browser, opts = {})
44
45
  case browser
45
46
  when :chrome
46
- Chrome::Driver.new(opts)
47
+ Chrome::Driver.new(**opts)
47
48
  when :internet_explorer, :ie
48
- IE::Driver.new(opts)
49
+ IE::Driver.new(**opts)
49
50
  when :safari
50
- Safari::Driver.new(opts)
51
+ Safari::Driver.new(**opts)
51
52
  when :firefox, :ff
52
- Firefox::Driver.new(opts)
53
+ Firefox::Driver.new(**opts)
53
54
  when :edge
54
- Edge::Driver.new(opts)
55
- when :edge_chrome
56
- EdgeChrome::Driver.new(opts)
57
- when :edge_html
58
- EdgeHtml::Driver.new(opts)
55
+ Edge::Driver.new(**opts)
59
56
  when :remote
60
- Remote::Driver.new(opts)
57
+ Remote::Driver.new(**opts)
61
58
  else
62
59
  raise ArgumentError, "unknown driver: #{browser.inspect}"
63
60
  end
@@ -73,12 +70,24 @@ module Selenium
73
70
 
74
71
  def initialize(bridge: nil, listener: nil, **opts)
75
72
  @service = nil
76
- bridge ||= create_bridge(opts)
73
+ bridge ||= create_bridge(**opts)
74
+ add_extensions(bridge.browser)
77
75
  @bridge = listener ? Support::EventFiringBridge.new(bridge, listener) : bridge
78
76
  end
79
77
 
80
78
  def inspect
81
- format '#<%<class>s:0x%<hash>x browser=%<browser>s>', class: self.class, hash: hash * 2, browser: bridge.browser.inspect
79
+ format '#<%<class>s:0x%<hash>x browser=%<browser>s>', class: self.class, hash: hash * 2,
80
+ browser: bridge.browser.inspect
81
+ end
82
+
83
+ #
84
+ # information about whether a remote end is in a state in which it can create new sessions,
85
+ # and may include additional meta information.
86
+ #
87
+ # @return [Hash]
88
+ #
89
+ def status
90
+ @bridge.status
82
91
  end
83
92
 
84
93
  #
@@ -296,38 +305,89 @@ module Selenium
296
305
 
297
306
  def create_bridge(**opts)
298
307
  opts[:url] ||= service_url(opts)
308
+ caps = opts.delete(:capabilities)
309
+ # NOTE: This is deprecated
310
+ cap_array = caps.is_a?(Hash) ? [caps] : Array(caps)
311
+
312
+ desired_capabilities = opts.delete(:desired_capabilities)
313
+ if desired_capabilities
314
+ WebDriver.logger.deprecate(':desired_capabilities as a parameter for driver initialization',
315
+ ':capabilities with an Array value of capabilities/options if necessary',
316
+ id: :desired_capabilities)
317
+ desired_capabilities = Remote::Capabilities.new(desired_capabilities) if desired_capabilities.is_a?(Hash)
318
+ cap_array << desired_capabilities
319
+ end
299
320
 
300
- desired_capabilities = opts.delete(:desired_capabilities) || Remote::Capabilities.send(browser || :new)
301
321
  options = opts.delete(:options)
322
+ if options
323
+ WebDriver.logger.deprecate(':options as a parameter for driver initialization',
324
+ ':capabilities with an Array of value capabilities/options if necessary',
325
+ id: :browser_options)
326
+ cap_array << options
327
+ end
302
328
 
303
- bridge = Remote::Bridge.new(http_client: opts.delete(:http_client), url: opts.delete(:url))
329
+ capabilities = generate_capabilities(cap_array)
330
+
331
+ bridge_opts = {http_client: opts.delete(:http_client), url: opts.delete(:url)}
304
332
  raise ArgumentError, "Unable to create a driver with parameters: #{opts}" unless opts.empty?
305
333
 
306
- namespacing = self.class.to_s.split('::')
334
+ bridge = Remote::Bridge.new(**bridge_opts)
307
335
 
308
- if Object.const_defined?("#{namespacing[0..-2].join('::')}::Bridge") && !namespacing.include?('Remote')
309
- bridge.extend Object.const_get("#{namespacing[0, namespacing.length - 1].join('::')}::Bridge")
310
- end
311
-
312
- bridge.create_session(desired_capabilities, options)
336
+ bridge.create_session(capabilities)
313
337
  bridge
314
338
  end
315
339
 
340
+ def generate_capabilities(cap_array)
341
+ cap_array.map { |cap|
342
+ if cap.is_a? Symbol
343
+ cap = Remote::Capabilities.send(cap)
344
+ elsif cap.is_a? Hash
345
+ new_message = 'Capabilities instance initialized with the Hash, or build values with Options class'
346
+ WebDriver.logger.deprecate("passing a Hash value to :capabilities",
347
+ new_message,
348
+ id: :capabilities_hash)
349
+ cap = Remote::Capabilities.new(cap)
350
+ elsif !cap.respond_to? :as_json
351
+ msg = ":capabilities parameter only accepts objects responding to #as_json which #{cap.class} does not"
352
+ raise ArgumentError, msg
353
+ end
354
+ cap&.as_json
355
+ }.inject(:merge) || Remote::Capabilities.send(browser || :new)
356
+ end
357
+
316
358
  def service_url(opts)
317
- @service = opts.delete(:service)
359
+ service_config = opts.delete(:service)
318
360
  %i[driver_opts driver_path port].each do |key|
319
361
  next unless opts.key? key
320
362
 
321
363
  WebDriver.logger.deprecate(":#{key}", ':service with an instance of Selenium::WebDriver::Service',
322
364
  id: "service_#{key}".to_sym)
323
365
  end
324
- @service ||= Service.send(browser,
325
- args: opts.delete(:driver_opts),
326
- path: opts.delete(:driver_path),
327
- port: opts.delete(:port))
328
- @service.start
366
+ service_config ||= Service.send(browser,
367
+ args: opts.delete(:driver_opts),
368
+ path: opts.delete(:driver_path),
369
+ port: opts.delete(:port))
370
+ @service = service_config.launch
329
371
  @service.uri
330
372
  end
373
+
374
+ def screenshot
375
+ bridge.screenshot
376
+ end
377
+
378
+ def add_extensions(browser)
379
+ extensions = case browser
380
+ when :chrome, :msedge
381
+ Chrome::Driver::EXTENSIONS
382
+ when :firefox
383
+ Firefox::Driver::EXTENSIONS
384
+ when :safari, :safari_technology_preview
385
+ Safari::Driver::EXTENSIONS
386
+ else
387
+ []
388
+ end
389
+ extensions.each { |extension| extend extension }
390
+ end
331
391
  end # Driver
332
392
  end # WebDriver
333
393
  end # Selenium
@@ -0,0 +1,89 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Licensed to the Software Freedom Conservancy (SFC) under one
4
+ # or more contributor license agreements. See the NOTICE file
5
+ # distributed with this work for additional information
6
+ # regarding copyright ownership. The SFC licenses this file
7
+ # to you under the Apache License, Version 2.0 (the
8
+ # "License"); you may not use this file except in compliance
9
+ # with the License. You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing,
14
+ # software distributed under the License is distributed on an
15
+ # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16
+ # KIND, either express or implied. See the License for the
17
+ # specific language governing permissions and limitations
18
+ # under the License.
19
+
20
+ module Selenium
21
+ module WebDriver
22
+ module DriverExtensions
23
+ module HasAuthentication
24
+
25
+ #
26
+ # Registers basic authentication handler which is automatically
27
+ # used whenever browser gets an authentication required response.
28
+ # This currently relies on DevTools so is only supported in
29
+ # Chromium browsers.
30
+ #
31
+ # @example Authenticate any request
32
+ # driver.register(username: 'admin', password: '123456')
33
+ #
34
+ # @example Authenticate based on URL
35
+ # driver.register(username: 'admin1', password: '123456', uri: /mysite1\.com/)
36
+ # driver.register(username: 'admin2', password: '123456', uri: /mysite2\.com/)
37
+ #
38
+ # @param [String] username
39
+ # @param [String] password
40
+ # @param [Regexp] uri to associate the credentials with
41
+ #
42
+
43
+ def register(username:, password:, uri: //)
44
+ auth_handlers << {username: username, password: password, uri: uri}
45
+
46
+ devtools.network.set_cache_disabled(cache_disabled: true)
47
+ devtools.fetch.on(:auth_required) do |params|
48
+ authenticate(params['requestId'], params.dig('request', 'url'))
49
+ end
50
+ devtools.fetch.on(:request_paused) do |params|
51
+ devtools.fetch.continue_request(request_id: params['requestId'])
52
+ end
53
+ devtools.fetch.enable(handle_auth_requests: true)
54
+ end
55
+
56
+ private
57
+
58
+ def auth_handlers
59
+ @auth_handlers ||= []
60
+ end
61
+
62
+ def authenticate(request_id, url)
63
+ credentials = auth_handlers.find do |handler|
64
+ url.match?(handler[:uri])
65
+ end
66
+
67
+ if credentials
68
+ devtools.fetch.continue_with_auth(
69
+ request_id: request_id,
70
+ auth_challenge_response: {
71
+ response: 'ProvideCredentials',
72
+ username: credentials[:username],
73
+ password: credentials[:password]
74
+ }
75
+ )
76
+ else
77
+ devtools.fetch.continue_with_auth(
78
+ request_id: request_id,
79
+ auth_challenge_response: {
80
+ response: 'CancelAuth'
81
+ }
82
+ )
83
+ end
84
+ end
85
+
86
+ end # HasAuthentication
87
+ end # DriverExtensions
88
+ end # WebDriver
89
+ end # Selenium
@@ -29,7 +29,12 @@ module Selenium
29
29
  #
30
30
 
31
31
  def devtools
32
- @devtools ||= DevTools.new(capabilities['goog:chromeOptions']['debuggerAddress'])
32
+ @devtools ||= begin
33
+ require 'selenium/devtools'
34
+ Selenium::DevTools.version ||= devtools_version
35
+ Selenium::DevTools.load_version
36
+ Selenium::WebDriver::DevTools.new(url: devtools_url)
37
+ end
33
38
  end
34
39
 
35
40
  end # HasDevTools
@@ -17,23 +17,20 @@
17
17
  # specific language governing permissions and limitations
18
18
  # under the License.
19
19
 
20
+ # TODO: Deprecated; Delete after 4.0 release
20
21
  module Selenium
21
22
  module WebDriver
22
23
  module DriverExtensions
23
24
  module HasLocation
24
25
  def location
25
- @bridge.location
26
+ raise Error::UnsupportedOperationError, 'The W3C standard does not currently support getting location'
26
27
  end
27
28
 
28
- def location=(loc)
29
- raise TypeError, "expected #{Location}, got #{loc.inspect}:#{loc.class}" unless loc.is_a?(Location)
30
-
31
- @bridge.set_location loc.latitude, loc.longitude, loc.altitude
29
+ def location=(*)
30
+ raise Error::UnsupportedOperationError, 'The W3C standard does not currently support setting location'
32
31
  end
32
+ alias_method :set_location, :location
33
33
 
34
- def set_location(lat, lon, alt)
35
- self.location = Location.new(Float(lat), Float(lon), Float(alt))
36
- end
37
34
  end # HasLocation
38
35
  end # DriverExtensions
39
36
  end # WebDriver
@@ -0,0 +1,149 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Licensed to the Software Freedom Conservancy (SFC) under one
4
+ # or more contributor license agreements. See the NOTICE file
5
+ # distributed with this work for additional information
6
+ # regarding copyright ownership. The SFC licenses this file
7
+ # to you under the Apache License, Version 2.0 (the
8
+ # "License"); you may not use this file except in compliance
9
+ # with the License. You may obtain a copy of the License at
10
+ #
11
+ # http://www.apache.org/licenses/LICENSE-2.0
12
+ #
13
+ # Unless required by applicable law or agreed to in writing,
14
+ # software distributed under the License is distributed on an
15
+ # "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16
+ # KIND, either express or implied. See the License for the
17
+ # specific language governing permissions and limitations
18
+ # under the License.
19
+
20
+ module Selenium
21
+ module WebDriver
22
+ module DriverExtensions
23
+ module HasLogEvents
24
+ include Atoms
25
+
26
+ KINDS = %i[console exception mutation].freeze
27
+
28
+ #
29
+ # Registers listener to be called whenever browser receives
30
+ # a new Console API message such as console.log() or an unhandled
31
+ # exception.
32
+ #
33
+ # This currently relies on DevTools so is only supported in
34
+ # Chromium browsers.
35
+ #
36
+ # @example Collect console messages
37
+ # logs = []
38
+ # driver.on_log_event(:console) do |event|
39
+ # logs.push(event)
40
+ # end
41
+ #
42
+ # @example Collect JavaScript exceptions
43
+ # exceptions = []
44
+ # driver.on_log_event(:exception) do |event|
45
+ # exceptions.push(event)
46
+ # end
47
+ #
48
+ # @example Collect DOM mutations
49
+ # mutations = []
50
+ # driver.on_log_event(:mutation) do |event|
51
+ # mutations.push(event)
52
+ # end
53
+ #
54
+ # @param [Symbol] kind :console, :exception or :mutation
55
+ # @param [#call] block which is called when event happens
56
+ # @yieldparam [DevTools::ConsoleEvent, DevTools::ExceptionEvent, DevTools::MutationEvent]
57
+ #
58
+
59
+ def on_log_event(kind, &block)
60
+ raise Error::WebDriverError, "Don't know how to handle #{kind} events" unless KINDS.include?(kind)
61
+
62
+ enabled = log_listeners[kind].any?
63
+ log_listeners[kind] << block
64
+ return if enabled
65
+
66
+ devtools.runtime.enable
67
+ __send__("log_#{kind}_events")
68
+ end
69
+
70
+ private
71
+
72
+ def log_listeners
73
+ @log_listeners ||= Hash.new { |listeners, kind| listeners[kind] = [] }
74
+ end
75
+
76
+ def log_console_events
77
+ devtools.runtime.on(:console_api_called) do |params|
78
+ event = DevTools::ConsoleEvent.new(
79
+ type: params['type'],
80
+ timestamp: params['timestamp'],
81
+ args: params['args']
82
+ )
83
+
84
+ log_listeners[:console].each do |listener|
85
+ listener.call(event)
86
+ end
87
+ end
88
+ end
89
+
90
+ def log_exception_events
91
+ devtools.runtime.on(:exception_thrown) do |params|
92
+ description = if params.dig('exceptionDetails', 'exception')
93
+ params.dig('exceptionDetails', 'exception', 'description')
94
+ else
95
+ params.dig('exceptionDetails', 'text')
96
+ end
97
+
98
+ event = DevTools::ExceptionEvent.new(
99
+ description: description,
100
+ timestamp: params['timestamp'],
101
+ stacktrace: params.dig('exceptionDetails', 'stackTrace', 'callFrames')
102
+ )
103
+
104
+ log_listeners[:exception].each do |listener|
105
+ listener.call(event)
106
+ end
107
+ end
108
+ end
109
+
110
+ def log_mutation_events
111
+ devtools.page.enable
112
+
113
+ devtools.runtime.add_binding(name: '__webdriver_attribute')
114
+ execute_script(mutation_listener)
115
+ script = devtools.page.add_script_to_evaluate_on_new_document(source: mutation_listener)
116
+ pinned_scripts[mutation_listener] = script['identifier']
117
+
118
+ devtools.runtime.on(:binding_called, &method(:log_mutation_event))
119
+ end
120
+
121
+ def log_mutation_event(params)
122
+ payload = JSON.parse(params['payload'])
123
+ elements = find_elements(css: "*[data-__webdriver_id='#{payload['target']}']")
124
+ return if elements.empty?
125
+
126
+ event = DevTools::MutationEvent.new(
127
+ element: elements.first,
128
+ attribute_name: payload['name'],
129
+ current_value: payload['value'],
130
+ old_value: payload['oldValue']
131
+ )
132
+
133
+ log_listeners[:mutation].each do |log_listener|
134
+ log_listener.call(event)
135
+ end
136
+ end
137
+
138
+ def mutation_listener
139
+ @mutation_listener ||= read_atom(:mutationListener)
140
+ end
141
+
142
+ def pinned_scripts
143
+ @pinned_scripts ||= {}
144
+ end
145
+
146
+ end # HasLogEvents
147
+ end # DriverExtensions
148
+ end # WebDriver
149
+ end # Selenium