selenium-webdriver 3.142.7 → 4.3.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/CHANGES +432 -5
- data/Gemfile +3 -1
- data/LICENSE +1 -1
- data/NOTICE +2 -0
- data/README.md +4 -5
- data/lib/selenium/server.rb +75 -64
- data/lib/selenium/webdriver/atoms/findElements.js +122 -0
- data/lib/selenium/webdriver/atoms/getAttribute.js +100 -7
- data/lib/selenium/webdriver/atoms/isDisplayed.js +76 -78
- data/lib/selenium/webdriver/atoms/mutationListener.js +55 -0
- data/lib/selenium/webdriver/bidi/session.rb +38 -0
- data/lib/selenium/webdriver/bidi.rb +55 -0
- data/lib/selenium/webdriver/chrome/driver.rb +26 -83
- data/lib/selenium/webdriver/chrome/{bridge.rb → features.rb} +55 -12
- data/lib/selenium/webdriver/chrome/options.rb +138 -67
- data/lib/selenium/webdriver/chrome/profile.rb +6 -3
- data/lib/selenium/webdriver/chrome/service.rb +8 -15
- data/lib/selenium/webdriver/chrome.rb +5 -18
- data/lib/selenium/webdriver/common/action_builder.rb +171 -236
- data/lib/selenium/webdriver/common/driver.rb +76 -29
- data/lib/selenium/webdriver/common/driver_extensions/full_page_screenshot.rb +43 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_apple_permissions.rb +51 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_authentication.rb +89 -0
- data/lib/selenium/webdriver/common/driver_extensions/{has_touch_screen.rb → has_bidi.rb} +10 -8
- data/lib/selenium/webdriver/common/driver_extensions/has_casting.rb +87 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_cdp.rb +38 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_context.rb +44 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_devtools.rb +43 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_launching.rb +38 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_location.rb +5 -8
- data/lib/selenium/webdriver/common/driver_extensions/has_log_events.rb +144 -0
- data/lib/selenium/webdriver/common/driver_extensions/{has_remote_status.rb → has_logs.rb} +4 -4
- data/lib/selenium/webdriver/common/driver_extensions/has_network_conditions.rb +17 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_network_connection.rb +6 -27
- data/lib/selenium/webdriver/common/driver_extensions/has_network_interception.rb +136 -0
- data/lib/selenium/webdriver/common/driver_extensions/has_permissions.rb +11 -11
- data/lib/selenium/webdriver/common/driver_extensions/has_pinned_scripts.rb +77 -0
- data/lib/selenium/webdriver/common/driver_extensions/{rotatable.rb → prints_page.rb} +18 -20
- data/lib/selenium/webdriver/common/element.rb +83 -23
- data/lib/selenium/webdriver/common/error.rb +32 -196
- data/lib/selenium/webdriver/common/interactions/input_device.rb +10 -4
- data/lib/selenium/webdriver/common/interactions/interaction.rb +12 -22
- data/lib/selenium/webdriver/common/interactions/interactions.rb +24 -4
- data/lib/selenium/webdriver/common/interactions/key_actions.rb +10 -6
- data/lib/selenium/webdriver/common/interactions/key_input.rb +11 -27
- data/lib/selenium/webdriver/common/interactions/none_input.rb +10 -8
- data/lib/selenium/webdriver/common/interactions/pause.rb +49 -0
- data/lib/selenium/webdriver/common/interactions/pointer_actions.rb +68 -78
- data/lib/selenium/webdriver/common/interactions/pointer_cancel.rb +45 -0
- data/lib/selenium/webdriver/common/interactions/pointer_event_properties.rb +63 -0
- data/lib/selenium/webdriver/common/interactions/pointer_input.rb +15 -84
- data/lib/selenium/webdriver/common/interactions/pointer_move.rb +60 -0
- data/lib/selenium/webdriver/common/interactions/pointer_press.rb +85 -0
- data/lib/selenium/webdriver/common/interactions/scroll.rb +57 -0
- data/lib/selenium/webdriver/common/interactions/scroll_origin.rb +48 -0
- data/lib/selenium/webdriver/common/interactions/typing_interaction.rb +54 -0
- data/lib/selenium/webdriver/common/interactions/wheel_actions.rb +113 -0
- data/lib/selenium/webdriver/common/{w3c_manager.rb → interactions/wheel_input.rb} +14 -17
- data/lib/selenium/webdriver/common/keys.rb +1 -0
- data/lib/selenium/webdriver/common/log_entry.rb +2 -2
- data/lib/selenium/webdriver/common/logger.rb +50 -15
- data/lib/selenium/webdriver/common/manager.rb +11 -38
- data/lib/selenium/webdriver/common/options.rb +147 -23
- data/lib/selenium/webdriver/common/platform.rb +10 -5
- data/lib/selenium/webdriver/common/port_prober.rb +4 -6
- data/lib/selenium/webdriver/common/profile_helper.rb +11 -9
- data/lib/selenium/webdriver/common/proxy.rb +6 -3
- data/lib/selenium/webdriver/common/search_context.rb +7 -9
- data/lib/selenium/webdriver/common/service.rb +17 -125
- data/lib/selenium/webdriver/common/service_manager.rb +150 -0
- data/lib/selenium/webdriver/common/shadow_root.rb +87 -0
- data/lib/selenium/webdriver/common/socket_lock.rb +2 -2
- data/lib/selenium/webdriver/common/socket_poller.rb +3 -3
- data/lib/selenium/webdriver/common/takes_screenshot.rb +66 -0
- data/lib/selenium/webdriver/common/target_locator.rb +32 -4
- data/lib/selenium/webdriver/common/timeouts.rb +31 -4
- data/lib/selenium/webdriver/common/wait.rb +1 -1
- data/lib/selenium/webdriver/common/websocket_connection.rb +149 -0
- data/lib/selenium/webdriver/common/window.rb +0 -4
- data/lib/selenium/webdriver/common/zipper.rb +3 -9
- data/lib/selenium/webdriver/common.rb +35 -18
- data/lib/selenium/webdriver/devtools/console_event.rb +38 -0
- data/lib/selenium/webdriver/devtools/exception_event.rb +36 -0
- data/lib/selenium/webdriver/devtools/mutation_event.rb +37 -0
- data/lib/selenium/webdriver/devtools/pinned_script.rb +59 -0
- data/lib/selenium/webdriver/devtools/request.rb +67 -0
- data/lib/selenium/webdriver/devtools/response.rb +66 -0
- data/lib/selenium/webdriver/devtools.rb +86 -0
- data/lib/selenium/webdriver/edge/driver.rb +7 -29
- data/lib/selenium/webdriver/edge/features.rb +45 -0
- data/lib/selenium/webdriver/edge/options.rb +11 -48
- data/lib/selenium/webdriver/edge/profile.rb +33 -0
- data/lib/selenium/webdriver/edge/service.rb +10 -26
- data/lib/selenium/webdriver/edge.rb +11 -14
- data/lib/selenium/webdriver/firefox/driver.rb +32 -19
- data/lib/selenium/webdriver/firefox/extension.rb +8 -0
- data/lib/selenium/webdriver/firefox/features.rb +63 -0
- data/lib/selenium/webdriver/firefox/options.rb +73 -50
- data/lib/selenium/webdriver/firefox/profile.rb +16 -70
- data/lib/selenium/webdriver/firefox/service.rb +5 -9
- data/lib/selenium/webdriver/firefox/util.rb +1 -1
- data/lib/selenium/webdriver/firefox.rb +17 -28
- data/lib/selenium/webdriver/ie/driver.rb +1 -47
- data/lib/selenium/webdriver/ie/options.rb +15 -46
- data/lib/selenium/webdriver/ie/service.rb +13 -15
- data/lib/selenium/webdriver/ie.rb +3 -16
- data/lib/selenium/webdriver/remote/bridge.rb +563 -86
- data/lib/selenium/webdriver/remote/capabilities.rb +159 -123
- data/lib/selenium/webdriver/remote/commands.rb +158 -0
- data/lib/selenium/webdriver/remote/driver.rb +22 -13
- data/lib/selenium/webdriver/remote/http/common.rb +0 -5
- data/lib/selenium/webdriver/remote/http/default.rb +22 -31
- data/lib/selenium/webdriver/remote/response.rb +18 -49
- data/lib/selenium/webdriver/remote.rb +15 -12
- data/lib/selenium/webdriver/safari/driver.rb +3 -31
- data/lib/selenium/webdriver/safari/{bridge.rb → features.rb} +3 -3
- data/lib/selenium/webdriver/safari/options.rb +10 -29
- data/lib/selenium/webdriver/safari/service.rb +4 -8
- data/lib/selenium/webdriver/safari.rb +13 -19
- data/lib/selenium/webdriver/support/block_event_listener.rb +1 -1
- data/lib/selenium/webdriver/support/cdp/domain.rb.erb +63 -0
- data/lib/selenium/webdriver/support/cdp_client_generator.rb +108 -0
- data/lib/selenium/webdriver/support/color.rb +9 -9
- data/lib/selenium/webdriver/support/event_firing_bridge.rb +4 -4
- data/lib/selenium/webdriver/support/guards/guard.rb +89 -0
- data/lib/selenium/webdriver/{firefox/marionette/bridge.rb → support/guards/guard_condition.rb} +22 -19
- data/lib/selenium/webdriver/support/guards.rb +95 -0
- data/lib/selenium/webdriver/support/relative_locator.rb +51 -0
- data/lib/selenium/webdriver/support/select.rb +3 -3
- data/lib/selenium/webdriver/support.rb +1 -0
- data/lib/selenium/webdriver/version.rb +1 -1
- data/lib/selenium/webdriver.rb +14 -13
- data/selenium-webdriver.gemspec +32 -13
- metadata +176 -69
- data/lib/selenium/webdriver/common/bridge_helper.rb +0 -82
- data/lib/selenium/webdriver/common/driver_extensions/takes_screenshot.rb +0 -64
- data/lib/selenium/webdriver/common/keyboard.rb +0 -70
- data/lib/selenium/webdriver/common/mouse.rb +0 -89
- data/lib/selenium/webdriver/common/touch_action_builder.rb +0 -78
- data/lib/selenium/webdriver/common/touch_screen.rb +0 -123
- data/lib/selenium/webdriver/common/w3c_action_builder.rb +0 -212
- data/lib/selenium/webdriver/edge/bridge.rb +0 -76
- data/lib/selenium/webdriver/firefox/binary.rb +0 -187
- data/lib/selenium/webdriver/firefox/extension/prefs.json +0 -69
- data/lib/selenium/webdriver/firefox/extension/webdriver.xpi +0 -0
- data/lib/selenium/webdriver/firefox/launcher.rb +0 -111
- data/lib/selenium/webdriver/firefox/legacy/driver.rb +0 -83
- data/lib/selenium/webdriver/firefox/marionette/driver.rb +0 -90
- data/lib/selenium/webdriver/firefox/native/linux/amd64/x_ignore_nofocus.so +0 -0
- data/lib/selenium/webdriver/firefox/native/linux/x86/x_ignore_nofocus.so +0 -0
- data/lib/selenium/webdriver/remote/http/persistent.rb +0 -60
- data/lib/selenium/webdriver/remote/oss/bridge.rb +0 -594
- data/lib/selenium/webdriver/remote/oss/commands.rb +0 -223
- data/lib/selenium/webdriver/remote/w3c/bridge.rb +0 -605
- data/lib/selenium/webdriver/remote/w3c/capabilities.rb +0 -310
- data/lib/selenium/webdriver/remote/w3c/commands.rb +0 -157
@@ -0,0 +1,38 @@
|
|
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
|
+
class BiDi
|
23
|
+
class Session
|
24
|
+
Status = Struct.new(:ready, :message)
|
25
|
+
|
26
|
+
def initialize(bidi)
|
27
|
+
@bidi = bidi
|
28
|
+
end
|
29
|
+
|
30
|
+
def status
|
31
|
+
status = @bidi.send_cmd('session.status')
|
32
|
+
Status.new(status['ready'], status['message'])
|
33
|
+
end
|
34
|
+
|
35
|
+
end # Session
|
36
|
+
end # BiDi
|
37
|
+
end # WebDriver
|
38
|
+
end # Selenium
|
@@ -0,0 +1,55 @@
|
|
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
|
+
class BiDi
|
23
|
+
autoload :Session, 'selenium/webdriver/bidi/session'
|
24
|
+
|
25
|
+
def initialize(url:)
|
26
|
+
@ws = WebSocketConnection.new(url: url)
|
27
|
+
end
|
28
|
+
|
29
|
+
def close
|
30
|
+
@ws.close
|
31
|
+
end
|
32
|
+
|
33
|
+
def callbacks
|
34
|
+
@ws.callbacks
|
35
|
+
end
|
36
|
+
|
37
|
+
def session
|
38
|
+
Session.new(self)
|
39
|
+
end
|
40
|
+
|
41
|
+
def send_cmd(method, **params)
|
42
|
+
data = {method: method, params: params.compact}
|
43
|
+
message = @ws.send_cmd(**data)
|
44
|
+
raise Error::WebDriverError, error_message(message) if message['error']
|
45
|
+
|
46
|
+
message['result']
|
47
|
+
end
|
48
|
+
|
49
|
+
def error_message(message)
|
50
|
+
"#{message['error']}: #{message['message']}\n#{message['stacktrace']}"
|
51
|
+
end
|
52
|
+
|
53
|
+
end # BiDi
|
54
|
+
end # WebDriver
|
55
|
+
end # Selenium
|
@@ -27,100 +27,43 @@ module Selenium
|
|
27
27
|
#
|
28
28
|
|
29
29
|
class Driver < WebDriver::Driver
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
super(@bridge, listener: listener)
|
47
|
-
end
|
30
|
+
EXTENSIONS = [DriverExtensions::HasCDP,
|
31
|
+
DriverExtensions::HasCasting,
|
32
|
+
DriverExtensions::HasNetworkConditions,
|
33
|
+
DriverExtensions::HasNetworkInterception,
|
34
|
+
DriverExtensions::HasWebStorage,
|
35
|
+
DriverExtensions::HasLaunching,
|
36
|
+
DriverExtensions::HasLocation,
|
37
|
+
DriverExtensions::HasPermissions,
|
38
|
+
DriverExtensions::DownloadsFiles,
|
39
|
+
DriverExtensions::HasDevTools,
|
40
|
+
DriverExtensions::HasAuthentication,
|
41
|
+
DriverExtensions::HasLogs,
|
42
|
+
DriverExtensions::HasLogEvents,
|
43
|
+
DriverExtensions::HasPinnedScripts,
|
44
|
+
DriverExtensions::PrintsPage].freeze
|
48
45
|
|
49
46
|
def browser
|
50
47
|
:chrome
|
51
48
|
end
|
52
49
|
|
53
|
-
def quit
|
54
|
-
super
|
55
|
-
ensure
|
56
|
-
@service&.stop
|
57
|
-
end
|
58
|
-
|
59
|
-
def execute_cdp(cmd, **params)
|
60
|
-
@bridge.send_command(cmd: cmd, params: params)
|
61
|
-
end
|
62
|
-
|
63
50
|
private
|
64
51
|
|
65
|
-
def
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
args = opts.delete(:args) || opts.delete(:switches)
|
70
|
-
if args
|
71
|
-
WebDriver.logger.deprecate ':args or :switches', 'Selenium::WebDriver::Chrome::Options#add_argument'
|
72
|
-
raise ArgumentError, ':args must be an Array of Strings' unless args.is_a? Array
|
73
|
-
|
74
|
-
args.each { |arg| options.add_argument(arg.to_s) }
|
75
|
-
end
|
52
|
+
def devtools_url
|
53
|
+
uri = URI(devtools_address)
|
54
|
+
response = Net::HTTP.get(uri.hostname, '/json/version', uri.port)
|
76
55
|
|
77
|
-
|
78
|
-
|
79
|
-
WebDriver.logger.deprecate 'Selenium::WebDriver::Chrome::Driver#new with `:profile` parameter',
|
80
|
-
'Selenium::WebDriver::Chrome::Options#profile or Options#add_option'
|
81
|
-
|
82
|
-
profile = profile.as_json
|
83
|
-
|
84
|
-
if options.args.none?(&/user-data-dir/.method(:match?))
|
85
|
-
options.add_argument("--user-data-dir=#{profile['directory']}")
|
86
|
-
end
|
87
|
-
|
88
|
-
if profile['extensions']
|
89
|
-
WebDriver.logger.deprecate 'Selenium::WebDriver::Chrome::Profile#extensions',
|
90
|
-
'Selenium::WebDriver::Chrome::Options#add_extension'
|
91
|
-
profile['extensions'].each do |extension|
|
92
|
-
options.add_encoded_extension(extension)
|
93
|
-
end
|
94
|
-
end
|
95
|
-
end
|
96
|
-
|
97
|
-
if opts.key?(:detach)
|
98
|
-
WebDriver.logger.deprecate 'Selenium::WebDriver::Chrome::Driver#new with `:detach` parameter',
|
99
|
-
'Selenium::WebDriver::Chrome::Options#new or Options#add_option'
|
100
|
-
options.add_option(:detach, opts.delete(:detach))
|
101
|
-
end
|
102
|
-
|
103
|
-
prefs = opts.delete(:prefs)
|
104
|
-
if prefs
|
105
|
-
WebDriver.logger.deprecate ':prefs', 'Selenium::WebDriver::Chrome::Options#add_preference'
|
106
|
-
prefs.each do |key, value|
|
107
|
-
options.add_preference(key, value)
|
108
|
-
end
|
109
|
-
end
|
110
|
-
|
111
|
-
options = options.as_json
|
112
|
-
caps.merge!(options) unless options[Options::KEY].empty?
|
113
|
-
|
114
|
-
if opts.key?(:proxy) || opts.key?('proxy')
|
115
|
-
WebDriver.logger.deprecate 'Selenium::WebDriver::Chrome::Driver#new with `:proxy` parameter',
|
116
|
-
'Selenium::WebDriver::Chrome::Capabilities#proxy='
|
56
|
+
JSON.parse(response)['webSocketDebuggerUrl']
|
57
|
+
end
|
117
58
|
|
118
|
-
|
119
|
-
|
120
|
-
|
59
|
+
def devtools_version
|
60
|
+
Integer(capabilities.browser_version.split('.').first)
|
61
|
+
end
|
121
62
|
|
122
|
-
|
63
|
+
def devtools_address
|
64
|
+
"http://#{capabilities['goog:chromeOptions']['debuggerAddress']}"
|
123
65
|
end
|
66
|
+
|
124
67
|
end # Driver
|
125
68
|
end # Chrome
|
126
69
|
end # WebDriver
|
@@ -20,32 +20,77 @@
|
|
20
20
|
module Selenium
|
21
21
|
module WebDriver
|
22
22
|
module Chrome
|
23
|
-
module
|
23
|
+
module Features
|
24
24
|
|
25
|
-
|
25
|
+
CHROME_COMMANDS = {
|
26
|
+
launch_app: [:post, 'session/:session_id/chromium/launch_app'],
|
27
|
+
get_cast_sinks: [:get, 'session/:session_id/goog/cast/get_sinks'],
|
28
|
+
set_cast_sink_to_use: [:post, 'session/:session_id/goog/cast/set_sink_to_use'],
|
29
|
+
start_cast_tab_mirroring: [:post, 'session/:session_id/goog/cast/start_tab_mirroring'],
|
30
|
+
start_cast_desktop_mirroring: [:post, 'session/:session_id/goog/cast/start_desktop_mirroring'],
|
31
|
+
get_cast_issue_message: [:get, 'session/:session_id/goog/cast/get_issue_message'],
|
32
|
+
stop_casting: [:post, 'session/:session_id/goog/cast/stop_casting'],
|
26
33
|
get_network_conditions: [:get, 'session/:session_id/chromium/network_conditions'],
|
27
34
|
set_network_conditions: [:post, 'session/:session_id/chromium/network_conditions'],
|
35
|
+
delete_network_conditions: [:delete, 'session/:session_id/chromium/network_conditions'],
|
36
|
+
set_permission: [:post, 'session/:session_id/permissions'],
|
28
37
|
send_command: [:post, 'session/:session_id/goog/cdp/execute'],
|
29
38
|
get_available_log_types: [:get, 'session/:session_id/se/log/types'],
|
30
39
|
get_log: [:post, 'session/:session_id/se/log']
|
31
40
|
}.freeze
|
32
41
|
|
33
42
|
def commands(command)
|
34
|
-
|
43
|
+
CHROME_COMMANDS[command] || self.class::COMMANDS[command]
|
35
44
|
end
|
36
45
|
|
37
|
-
def
|
38
|
-
execute :
|
46
|
+
def launch_app(id)
|
47
|
+
execute :launch_app, {}, {id: id}
|
39
48
|
end
|
40
49
|
|
41
|
-
def
|
42
|
-
execute :
|
50
|
+
def cast_sinks
|
51
|
+
execute :get_cast_sinks
|
52
|
+
end
|
53
|
+
|
54
|
+
def cast_sink_to_use=(name)
|
55
|
+
execute :set_cast_sink_to_use, {}, {sinkName: name}
|
56
|
+
end
|
57
|
+
|
58
|
+
def cast_issue_message
|
59
|
+
execute :cast_issue_message
|
60
|
+
end
|
61
|
+
|
62
|
+
def start_cast_tab_mirroring(name)
|
63
|
+
execute :start_cast_tab_mirroring, {}, {sinkName: name}
|
64
|
+
end
|
65
|
+
|
66
|
+
def start_cast_desktop_mirroring(name)
|
67
|
+
execute :start_cast_desktop_mirroring, {}, {sinkName: name}
|
68
|
+
end
|
69
|
+
|
70
|
+
def stop_casting(name)
|
71
|
+
execute :stop_casting, {}, {sinkName: name}
|
72
|
+
end
|
73
|
+
|
74
|
+
def set_permission(name, value)
|
75
|
+
execute :set_permission, {}, {descriptor: {name: name}, state: value}
|
76
|
+
end
|
77
|
+
|
78
|
+
def network_conditions
|
79
|
+
execute :get_network_conditions
|
43
80
|
end
|
44
81
|
|
45
82
|
def network_conditions=(conditions)
|
46
83
|
execute :set_network_conditions, {}, {network_conditions: conditions}
|
47
84
|
end
|
48
85
|
|
86
|
+
def delete_network_conditions
|
87
|
+
execute :delete_network_conditions
|
88
|
+
end
|
89
|
+
|
90
|
+
def send_command(command_params)
|
91
|
+
execute :send_command, {}, command_params
|
92
|
+
end
|
93
|
+
|
49
94
|
def available_log_types
|
50
95
|
types = execute :get_available_log_types
|
51
96
|
Array(types).map(&:to_sym)
|
@@ -55,11 +100,9 @@ module Selenium
|
|
55
100
|
data = execute :get_log, {}, {type: type.to_s}
|
56
101
|
|
57
102
|
Array(data).map do |l|
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
next
|
62
|
-
end
|
103
|
+
LogEntry.new l.fetch('level', 'UNKNOWN'), l.fetch('timestamp'), l.fetch('message')
|
104
|
+
rescue KeyError
|
105
|
+
next
|
63
106
|
end
|
64
107
|
end
|
65
108
|
end # Bridge
|
@@ -20,38 +20,74 @@
|
|
20
20
|
module Selenium
|
21
21
|
module WebDriver
|
22
22
|
module Chrome
|
23
|
-
class Options < WebDriver::
|
24
|
-
|
25
|
-
attr_accessor :binary, :profile, :detach
|
23
|
+
class Options < WebDriver::Options
|
24
|
+
attr_accessor :profile, :logging_prefs
|
26
25
|
|
27
26
|
KEY = 'goog:chromeOptions'
|
27
|
+
BROWSER = 'chrome'
|
28
|
+
|
29
|
+
# see: http://chromedriver.chromium.org/capabilities
|
30
|
+
CAPABILITIES = {args: 'args',
|
31
|
+
binary: 'binary',
|
32
|
+
local_state: 'localState',
|
33
|
+
prefs: 'prefs',
|
34
|
+
detach: 'detach',
|
35
|
+
debugger_address: 'debuggerAddress',
|
36
|
+
exclude_switches: 'excludeSwitches',
|
37
|
+
minidump_path: 'minidumpPath',
|
38
|
+
emulation: 'mobileEmulation',
|
39
|
+
perf_logging_prefs: 'perfLoggingPrefs',
|
40
|
+
window_types: 'windowTypes',
|
41
|
+
android_package: 'androidPackage',
|
42
|
+
android_activity: 'androidActivity',
|
43
|
+
android_device_serial: 'androidDeviceSerial',
|
44
|
+
android_use_running_app: 'androidUseRunningApp'}.freeze
|
45
|
+
|
46
|
+
# NOTE: special handling of 'extensions' to validate when set instead of when used
|
47
|
+
attr_reader :extensions
|
28
48
|
|
29
|
-
#
|
30
49
|
# Create a new Options instance.
|
31
50
|
#
|
32
51
|
# @example
|
33
52
|
# options = Selenium::WebDriver::Chrome::Options.new(args: ['start-maximized', 'user-data-dir=/tmp/temp_profile'])
|
34
|
-
# driver = Selenium::WebDriver.for(:chrome,
|
53
|
+
# driver = Selenium::WebDriver.for(:chrome, capabilities: options)
|
35
54
|
#
|
55
|
+
# @param [Profile] profile An instance of a Chrome::Profile Class
|
36
56
|
# @param [Hash] opts the pre-defined options to create the Chrome::Options with
|
37
|
-
# @option opts [Array
|
38
|
-
# @option opts [String]
|
39
|
-
# @option opts [
|
40
|
-
# @option opts [
|
41
|
-
# @option opts [
|
42
|
-
# @option opts [Hash]
|
43
|
-
#
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
57
|
+
# @option opts [Array] encoded_extensions List of extensions that do not need to be Base64 encoded
|
58
|
+
# @option opts [Array<String>] args List of command-line arguments to use when starting Chrome
|
59
|
+
# @option opts [String] binary Path to the Chrome executable to use
|
60
|
+
# @option opts [Hash] prefs A hash with each entry consisting of the name of the preference and its value
|
61
|
+
# @option opts [Array<String>] extensions A list of paths to (.crx) Chrome extensions to install on startup
|
62
|
+
# @option opts [Hash] options A hash for raw options
|
63
|
+
# @option opts [Hash] emulation A hash for raw emulation options
|
64
|
+
# @option opts [Hash] local_state A hash for the Local State file in the user data folder
|
65
|
+
# @option opts [Boolean] detach whether browser is closed when the driver is sent the quit command
|
66
|
+
# @option opts [String] debugger_address address of a Chrome debugger server to connect to
|
67
|
+
# @option opts [Array<String>] exclude_switches command line switches to exclude
|
68
|
+
# @option opts [String] minidump_path Directory to store Chrome minidumps (linux only)
|
69
|
+
# @option opts [Hash] perf_logging_prefs A hash for performance logging preferences
|
70
|
+
# @option opts [Array<String>] window_types A list of window types to appear in the list of window handles
|
71
|
+
#
|
72
|
+
|
73
|
+
def initialize(profile: nil, **opts)
|
74
|
+
super(**opts)
|
75
|
+
|
76
|
+
@profile = profile
|
77
|
+
|
78
|
+
@options = {args: [],
|
79
|
+
prefs: {},
|
80
|
+
emulation: {},
|
81
|
+
extensions: [],
|
82
|
+
local_state: {},
|
83
|
+
exclude_switches: [],
|
84
|
+
perf_logging_prefs: {},
|
85
|
+
window_types: []}.merge(@options)
|
86
|
+
|
87
|
+
@logging_prefs = options.delete(:logging_prefs) || {}
|
88
|
+
@encoded_extensions = @options.delete(:encoded_extensions) || []
|
89
|
+
@extensions = []
|
90
|
+
@options.delete(:extensions).each { |ext| validate_extension(ext) }
|
55
91
|
end
|
56
92
|
|
57
93
|
#
|
@@ -65,10 +101,22 @@ module Selenium
|
|
65
101
|
#
|
66
102
|
|
67
103
|
def add_extension(path)
|
68
|
-
|
69
|
-
|
104
|
+
validate_extension(path)
|
105
|
+
end
|
70
106
|
|
71
|
-
|
107
|
+
#
|
108
|
+
# Add an extension by local path.
|
109
|
+
#
|
110
|
+
# @example
|
111
|
+
# extensions = ['/path/to/extension.crx', '/path/to/other.crx']
|
112
|
+
# options = Selenium::WebDriver::Chrome::Options.new
|
113
|
+
# options.extensions = extensions
|
114
|
+
#
|
115
|
+
# @param [Array<String>] extensions A list of paths to (.crx) Chrome extensions to install on startup
|
116
|
+
#
|
117
|
+
|
118
|
+
def extensions=(extensions)
|
119
|
+
extensions.each { |ext| validate_extension(ext) }
|
72
120
|
end
|
73
121
|
|
74
122
|
#
|
@@ -96,22 +144,7 @@ module Selenium
|
|
96
144
|
#
|
97
145
|
|
98
146
|
def add_argument(arg)
|
99
|
-
@args << arg
|
100
|
-
end
|
101
|
-
|
102
|
-
#
|
103
|
-
# Add a new option not yet handled by bindings.
|
104
|
-
#
|
105
|
-
# @example Leave Chrome open when chromedriver is killed
|
106
|
-
# options = Selenium::WebDriver::Chrome::Options.new
|
107
|
-
# options.add_option(:detach, true)
|
108
|
-
#
|
109
|
-
# @param [String, Symbol] name Name of the option
|
110
|
-
# @param [Boolean, String, Integer] value Value of the option
|
111
|
-
#
|
112
|
-
|
113
|
-
def add_option(name, value)
|
114
|
-
@options[name] = value
|
147
|
+
@options[:args] << arg
|
115
148
|
end
|
116
149
|
|
117
150
|
#
|
@@ -126,7 +159,7 @@ module Selenium
|
|
126
159
|
#
|
127
160
|
|
128
161
|
def add_preference(name, value)
|
129
|
-
prefs[name] = value
|
162
|
+
@options[:prefs][name] = value
|
130
163
|
end
|
131
164
|
|
132
165
|
#
|
@@ -142,7 +175,9 @@ module Selenium
|
|
142
175
|
end
|
143
176
|
|
144
177
|
#
|
145
|
-
# Add
|
178
|
+
# Add emulation device information
|
179
|
+
#
|
180
|
+
# see: http://chromedriver.chromium.org/mobile-emulation
|
146
181
|
#
|
147
182
|
# @example Start Chrome in mobile emulation mode by device name
|
148
183
|
# options = Selenium::WebDriver::Chrome::Options.new
|
@@ -152,37 +187,73 @@ module Selenium
|
|
152
187
|
# options = Selenium::WebDriver::Chrome::Options.new
|
153
188
|
# options.add_emulation(device_metrics: {width: 400, height: 800, pixelRatio: 1, touch: true})
|
154
189
|
#
|
155
|
-
# @param [
|
156
|
-
# @
|
157
|
-
# @
|
190
|
+
# @param [Hash] opts the pre-defined options for adding mobile emulation values
|
191
|
+
# @option opts [String] :device_name A valid device name from the Chrome DevTools Emulation panel
|
192
|
+
# @option opts [Hash] :device_metrics Hash containing width, height, pixelRatio, touch
|
193
|
+
# @option opts [String] :user_agent Full user agent
|
158
194
|
#
|
159
195
|
|
160
|
-
def add_emulation(
|
161
|
-
@
|
162
|
-
@emulation[:deviceMetrics] = device_metrics if device_metrics
|
163
|
-
@emulation[:userAgent] = user_agent if user_agent
|
196
|
+
def add_emulation(**opts)
|
197
|
+
@options[:emulation] = opts
|
164
198
|
end
|
165
199
|
|
166
200
|
#
|
167
|
-
#
|
201
|
+
# Enables mobile browser use on Android.
|
202
|
+
#
|
203
|
+
# @see https://chromedriver.chromium.org/getting-started/getting-started---android
|
168
204
|
#
|
205
|
+
# @param [String] package The package name of the Chrome or WebView app.
|
206
|
+
# @param [String] serial_number The device serial number on which to launch the Chrome or WebView app.
|
207
|
+
# @param [String] use_running_app When true uses an already-running Chrome or WebView app,
|
208
|
+
# instead of launching the app with a clear data directory.
|
209
|
+
# @param [String] activity Name of the Activity hosting the WebView (Not available on Chrome Apps).
|
210
|
+
#
|
211
|
+
|
212
|
+
def enable_android(package: 'com.android.chrome', serial_number: nil, use_running_app: nil, activity: nil)
|
213
|
+
@options[:android_package] = package
|
214
|
+
@options[:android_activity] = activity unless activity.nil?
|
215
|
+
@options[:android_device_serial] = serial_number unless serial_number.nil?
|
216
|
+
@options[:android_use_running_app] = use_running_app unless use_running_app.nil?
|
217
|
+
end
|
169
218
|
|
170
|
-
|
171
|
-
|
172
|
-
|
219
|
+
private
|
220
|
+
|
221
|
+
def enable_logging(browser_options)
|
222
|
+
browser_options['goog:loggingPrefs'] = @logging_prefs
|
223
|
+
end
|
224
|
+
|
225
|
+
def process_browser_options(browser_options)
|
226
|
+
enable_logging(browser_options) unless @logging_prefs.empty?
|
227
|
+
|
228
|
+
options = browser_options[self.class::KEY]
|
229
|
+
options['binary'] ||= binary_path if binary_path
|
230
|
+
if @profile
|
231
|
+
options['args'] ||= []
|
232
|
+
options['args'] << "--user-data-dir=#{@profile.directory}"
|
173
233
|
end
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
{
|
234
|
+
|
235
|
+
return if (@encoded_extensions + @extensions).empty?
|
236
|
+
|
237
|
+
options['extensions'] = @encoded_extensions + @extensions.map { |ext| encode_extension(ext) }
|
238
|
+
end
|
239
|
+
|
240
|
+
def binary_path
|
241
|
+
Chrome.path
|
242
|
+
end
|
243
|
+
|
244
|
+
def encode_extension(path)
|
245
|
+
File.open(path, 'rb') { |crx_file| Base64.strict_encode64 crx_file.read }
|
246
|
+
end
|
247
|
+
|
248
|
+
def validate_extension(path)
|
249
|
+
raise Error::WebDriverError, "could not find extension at #{path.inspect}" unless File.file?(path)
|
250
|
+
raise Error::WebDriverError, "file was not an extension #{path.inspect}" unless File.extname(path) == '.crx'
|
251
|
+
|
252
|
+
@extensions << path
|
253
|
+
end
|
254
|
+
|
255
|
+
def camelize?(key)
|
256
|
+
!%w[localState prefs].include?(key)
|
186
257
|
end
|
187
258
|
end # Options
|
188
259
|
end # Chrome
|
@@ -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)
|
@@ -20,35 +20,28 @@
|
|
20
20
|
module Selenium
|
21
21
|
module WebDriver
|
22
22
|
module Chrome
|
23
|
-
#
|
24
|
-
# @api private
|
25
|
-
#
|
26
|
-
|
27
23
|
class Service < WebDriver::Service
|
28
|
-
|
29
|
-
|
30
|
-
|
24
|
+
DEFAULT_PORT = 9515
|
25
|
+
EXECUTABLE = 'chromedriver'
|
26
|
+
MISSING_TEXT = <<~ERROR
|
31
27
|
Unable to find chromedriver. Please download the server from
|
32
28
|
https://chromedriver.storage.googleapis.com/index.html and place it somewhere on your PATH.
|
33
29
|
More info at https://github.com/SeleniumHQ/selenium/wiki/ChromeDriver.
|
34
30
|
ERROR
|
35
|
-
|
36
|
-
|
37
|
-
def self.driver_path=(path)
|
38
|
-
Platform.assert_executable path if path.is_a?(String)
|
39
|
-
@driver_path = path
|
40
|
-
end
|
31
|
+
SHUTDOWN_SUPPORTED = true
|
41
32
|
|
42
33
|
private
|
43
34
|
|
44
|
-
#
|
35
|
+
# NOTE: This processing is deprecated
|
45
36
|
def extract_service_args(driver_opts)
|
46
37
|
driver_args = super
|
47
38
|
driver_opts = driver_opts.dup
|
48
39
|
driver_args << "--log-path=#{driver_opts.delete(:log_path)}" if driver_opts.key?(:log_path)
|
49
40
|
driver_args << "--url-base=#{driver_opts.delete(:url_base)}" if driver_opts.key?(:url_base)
|
50
41
|
driver_args << "--port-server=#{driver_opts.delete(:port_server)}" if driver_opts.key?(:port_server)
|
51
|
-
|
42
|
+
if driver_opts.key?(:whitelisted_ips)
|
43
|
+
driver_args << "--whitelisted-ips=#{driver_opts.delete(:whitelisted_ips)}"
|
44
|
+
end
|
52
45
|
driver_args << "--verbose" if driver_opts.key?(:verbose)
|
53
46
|
driver_args << "--silent" if driver_opts.key?(:silent)
|
54
47
|
driver_args
|