selenium-webdriver 4.0.0.alpha5 → 4.0.0.beta3
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/lib/selenium/devtools.rb +30 -0
- data/lib/selenium/server.rb +18 -26
- data/lib/selenium/webdriver.rb +1 -3
- data/lib/selenium/webdriver/atoms/findElements.js +93 -93
- data/lib/selenium/webdriver/atoms/getAttribute.js +75 -59
- data/lib/selenium/webdriver/atoms/isDisplayed.js +72 -72
- data/lib/selenium/webdriver/atoms/mutationListener.js +55 -0
- data/lib/selenium/webdriver/chrome.rb +1 -1
- data/lib/selenium/webdriver/chrome/driver.rb +28 -6
- data/lib/selenium/webdriver/chrome/{bridge.rb → features.rb} +6 -8
- data/lib/selenium/webdriver/chrome/options.rb +54 -37
- data/lib/selenium/webdriver/chrome/profile.rb +6 -3
- data/lib/selenium/webdriver/chrome/service.rb +4 -2
- data/lib/selenium/webdriver/common.rb +7 -2
- data/lib/selenium/webdriver/common/driver.rb +86 -26
- data/lib/selenium/webdriver/common/driver_extensions/has_authentication.rb +89 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_devtools.rb +6 -1
- data/lib/selenium/webdriver/common/driver_extensions/has_location.rb +5 -8
- data/lib/selenium/webdriver/common/driver_extensions/has_log_events.rb +149 -0
- data/lib/selenium/webdriver/{edge_chrome/bridge.rb → common/driver_extensions/has_logs.rb} +7 -7
- data/lib/selenium/webdriver/common/driver_extensions/has_network_connection.rb +6 -27
- data/lib/selenium/webdriver/common/driver_extensions/has_network_interception.rb +67 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_remote_status.rb +1 -0
- data/lib/selenium/webdriver/common/driver_extensions/{rotatable.rb → prints_page.rb} +18 -20
- data/lib/selenium/webdriver/common/element.rb +66 -12
- data/lib/selenium/webdriver/common/interactions/interaction.rb +4 -1
- data/lib/selenium/webdriver/common/logger.rb +6 -3
- data/lib/selenium/webdriver/common/manager.rb +11 -1
- data/lib/selenium/webdriver/common/options.rb +90 -11
- data/lib/selenium/webdriver/common/platform.rb +3 -1
- data/lib/selenium/webdriver/common/port_prober.rb +4 -6
- data/lib/selenium/webdriver/common/proxy.rb +4 -1
- data/lib/selenium/webdriver/common/search_context.rb +4 -1
- data/lib/selenium/webdriver/common/service.rb +13 -114
- data/lib/selenium/webdriver/common/service_manager.rb +151 -0
- data/lib/selenium/webdriver/common/socket_poller.rb +19 -30
- data/lib/selenium/webdriver/common/takes_screenshot.rb +63 -0
- data/lib/selenium/webdriver/common/target_locator.rb +4 -4
- data/lib/selenium/webdriver/devtools.rb +144 -0
- data/lib/selenium/webdriver/devtools/console_event.rb +38 -0
- data/lib/selenium/webdriver/{edge_html/driver.rb → devtools/exception_event.rb} +10 -13
- data/lib/selenium/webdriver/devtools/mutation_event.rb +37 -0
- data/lib/selenium/webdriver/devtools/request.rb +57 -0
- data/lib/selenium/webdriver/edge.rb +7 -29
- data/lib/selenium/webdriver/{edge_chrome → edge}/driver.rb +10 -4
- data/lib/selenium/webdriver/edge/features.rb +39 -0
- data/lib/selenium/webdriver/{edge_chrome → edge}/options.rb +12 -3
- data/lib/selenium/webdriver/{edge_chrome → edge}/profile.rb +2 -2
- data/lib/selenium/webdriver/{edge_chrome → edge}/service.rb +2 -2
- data/lib/selenium/webdriver/firefox.rb +5 -1
- data/lib/selenium/webdriver/firefox/driver.rb +19 -3
- data/lib/selenium/webdriver/firefox/{bridge.rb → features.rb} +3 -3
- data/lib/selenium/webdriver/firefox/options.rb +25 -31
- data/lib/selenium/webdriver/firefox/profile.rb +12 -2
- data/lib/selenium/webdriver/firefox/service.rb +1 -1
- data/lib/selenium/webdriver/ie/driver.rb +1 -2
- data/lib/selenium/webdriver/ie/options.rb +7 -20
- data/lib/selenium/webdriver/ie/service.rb +4 -2
- data/lib/selenium/webdriver/remote/bridge.rb +50 -42
- data/lib/selenium/webdriver/remote/capabilities.rb +127 -71
- data/lib/selenium/webdriver/remote/commands.rb +3 -0
- data/lib/selenium/webdriver/remote/driver.rb +10 -3
- data/lib/selenium/webdriver/remote/http/common.rb +0 -5
- data/lib/selenium/webdriver/remote/http/default.rb +8 -7
- data/lib/selenium/webdriver/remote/http/persistent.rb +6 -0
- data/lib/selenium/webdriver/safari.rb +8 -1
- data/lib/selenium/webdriver/safari/driver.rb +3 -4
- data/lib/selenium/webdriver/safari/{bridge.rb → features.rb} +3 -3
- data/lib/selenium/webdriver/safari/options.rb +1 -33
- data/lib/selenium/webdriver/support/block_event_listener.rb +1 -1
- data/lib/selenium/webdriver/support/color.rb +2 -2
- data/lib/selenium/webdriver/support/event_firing_bridge.rb +1 -1
- data/lib/selenium/webdriver/support/guards.rb +95 -0
- data/lib/selenium/webdriver/support/guards/guard.rb +89 -0
- data/lib/selenium/webdriver/support/guards/guard_condition.rb +52 -0
- data/lib/selenium/webdriver/support/select.rb +2 -2
- data/lib/selenium/webdriver/version.rb +1 -1
- metadata +69 -32
- data/CHANGES +0 -1725
- data/Gemfile +0 -4
- data/LICENSE +0 -202
- data/README.md +0 -35
- data/lib/selenium/webdriver/common/driver_extensions/takes_screenshot.rb +0 -65
- data/lib/selenium/webdriver/edge_html/options.rb +0 -91
- data/lib/selenium/webdriver/edge_html/service.rb +0 -47
- 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/
|
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
|
-
#
|
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
|
-
|
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,
|
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
|
-
|
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
|
-
|
334
|
+
bridge = Remote::Bridge.new(**bridge_opts)
|
307
335
|
|
308
|
-
|
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
|
-
|
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
|
-
|
325
|
-
|
326
|
-
|
327
|
-
|
328
|
-
@service.
|
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 ||=
|
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
|
-
|
26
|
+
raise Error::UnsupportedOperationError, 'The W3C standard does not currently support getting location'
|
26
27
|
end
|
27
28
|
|
28
|
-
def location=(
|
29
|
-
raise
|
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
|