selenium-webdriver 3.142.7 → 4.0.3
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.
- checksums.yaml +5 -5
- data/CHANGES +350 -5
- data/Gemfile +3 -1
- data/LICENSE +1 -1
- data/NOTICE +2 -0
- data/README.md +4 -5
- data/lib/selenium/server.rb +69 -63
- 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/chrome/driver.rb +26 -83
- data/lib/selenium/webdriver/chrome/{bridge.rb → features.rb} +50 -12
- data/lib/selenium/webdriver/chrome/options.rb +129 -58
- data/lib/selenium/webdriver/chrome/profile.rb +6 -3
- data/lib/selenium/webdriver/chrome/service.rb +8 -15
- data/lib/selenium/webdriver/chrome.rb +10 -9
- data/lib/selenium/webdriver/common/action_builder.rb +97 -249
- data/lib/selenium/webdriver/common/driver.rb +112 -23
- 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_casting.rb +77 -0
- data/lib/selenium/webdriver/common/driver_extensions/{has_touch_screen.rb → has_cdp.rb} +10 -8
- data/lib/selenium/webdriver/common/driver_extensions/has_context.rb +45 -0
- data/lib/selenium/webdriver/{firefox/util.rb → common/driver_extensions/has_devtools.rb} +16 -19
- 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_logs.rb +30 -0
- 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/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 +82 -22
- data/lib/selenium/webdriver/common/error.rb +32 -196
- data/lib/selenium/webdriver/common/interactions/interaction.rb +4 -1
- data/lib/selenium/webdriver/common/interactions/key_actions.rb +5 -5
- data/lib/selenium/webdriver/common/interactions/pointer_actions.rb +13 -13
- 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 +15 -15
- data/lib/selenium/webdriver/common/options.rb +154 -23
- data/lib/selenium/webdriver/common/platform.rb +6 -1
- 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 -3
- data/lib/selenium/webdriver/common/service.rb +17 -125
- data/lib/selenium/webdriver/common/service_manager.rb +151 -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 +2 -2
- 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/window.rb +0 -4
- data/lib/selenium/webdriver/common/zipper.rb +1 -9
- data/lib/selenium/webdriver/common.rb +23 -17
- 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 +182 -0
- data/lib/selenium/webdriver/edge/driver.rb +7 -29
- data/lib/selenium/webdriver/edge/features.rb +44 -0
- data/lib/selenium/webdriver/edge/options.rb +11 -48
- data/lib/selenium/webdriver/{common/w3c_manager.rb → edge/profile.rb} +7 -19
- data/lib/selenium/webdriver/edge/service.rb +10 -26
- data/lib/selenium/webdriver/edge.rb +11 -14
- data/lib/selenium/webdriver/firefox/driver.rb +31 -19
- data/lib/selenium/webdriver/firefox/extension.rb +8 -0
- data/lib/selenium/webdriver/firefox/features.rb +66 -0
- data/lib/selenium/webdriver/firefox/options.rb +70 -49
- data/lib/selenium/webdriver/firefox/profile.rb +21 -71
- data/lib/selenium/webdriver/firefox/service.rb +5 -9
- data/lib/selenium/webdriver/firefox.rb +22 -20
- data/lib/selenium/webdriver/ie/driver.rb +1 -47
- data/lib/selenium/webdriver/ie/options.rb +13 -44
- data/lib/selenium/webdriver/ie/service.rb +13 -15
- data/lib/selenium/webdriver/ie.rb +8 -7
- data/lib/selenium/webdriver/remote/bridge.rb +558 -86
- data/lib/selenium/webdriver/remote/capabilities.rb +159 -123
- data/lib/selenium/webdriver/remote/commands.rb +163 -0
- data/lib/selenium/webdriver/remote/driver.rb +22 -12
- data/lib/selenium/webdriver/remote/http/common.rb +0 -5
- data/lib/selenium/webdriver/remote/http/default.rb +17 -20
- data/lib/selenium/webdriver/remote/http/persistent.rb +11 -6
- data/lib/selenium/webdriver/remote/response.rb +16 -47
- data/lib/selenium/webdriver/remote.rb +15 -13
- 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 +17 -9
- 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 +2 -2
- 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 +12 -12
- data/selenium-webdriver.gemspec +28 -12
- metadata +128 -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/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
|
@@ -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,72 @@
|
|
|
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
|
+
get_cast_issue_message: [:get, 'session/:session_id/goog/cast/get_issue_message'],
|
|
31
|
+
stop_casting: [:post, 'session/:session_id/goog/cast/stop_casting'],
|
|
26
32
|
get_network_conditions: [:get, 'session/:session_id/chromium/network_conditions'],
|
|
27
33
|
set_network_conditions: [:post, 'session/:session_id/chromium/network_conditions'],
|
|
34
|
+
delete_network_conditions: [:delete, 'session/:session_id/chromium/network_conditions'],
|
|
35
|
+
set_permission: [:post, 'session/:session_id/permissions'],
|
|
28
36
|
send_command: [:post, 'session/:session_id/goog/cdp/execute'],
|
|
29
37
|
get_available_log_types: [:get, 'session/:session_id/se/log/types'],
|
|
30
38
|
get_log: [:post, 'session/:session_id/se/log']
|
|
31
39
|
}.freeze
|
|
32
40
|
|
|
33
41
|
def commands(command)
|
|
34
|
-
|
|
42
|
+
CHROME_COMMANDS[command] || self.class::COMMANDS[command]
|
|
35
43
|
end
|
|
36
44
|
|
|
37
|
-
def
|
|
38
|
-
execute :
|
|
45
|
+
def launch_app(id)
|
|
46
|
+
execute :launch_app, {}, {id: id}
|
|
39
47
|
end
|
|
40
48
|
|
|
41
|
-
def
|
|
42
|
-
execute :
|
|
49
|
+
def cast_sinks
|
|
50
|
+
execute :get_cast_sinks
|
|
51
|
+
end
|
|
52
|
+
|
|
53
|
+
def cast_sink_to_use=(name)
|
|
54
|
+
execute :set_cast_sink_to_use, {}, {sinkName: name}
|
|
55
|
+
end
|
|
56
|
+
|
|
57
|
+
def cast_issue_message
|
|
58
|
+
execute :cast_issue_message
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
def start_cast_tab_mirroring(name)
|
|
62
|
+
execute :start_cast_tab_mirroring, {}, {sinkName: name}
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
def stop_casting(name)
|
|
66
|
+
execute :stop_casting, {}, {sinkName: name}
|
|
67
|
+
end
|
|
68
|
+
|
|
69
|
+
def set_permission(name, value)
|
|
70
|
+
execute :set_permission, {}, {descriptor: {name: name}, state: value}
|
|
71
|
+
end
|
|
72
|
+
|
|
73
|
+
def network_conditions
|
|
74
|
+
execute :get_network_conditions
|
|
43
75
|
end
|
|
44
76
|
|
|
45
77
|
def network_conditions=(conditions)
|
|
46
78
|
execute :set_network_conditions, {}, {network_conditions: conditions}
|
|
47
79
|
end
|
|
48
80
|
|
|
81
|
+
def delete_network_conditions
|
|
82
|
+
execute :delete_network_conditions
|
|
83
|
+
end
|
|
84
|
+
|
|
85
|
+
def send_command(command_params)
|
|
86
|
+
execute :send_command, {}, command_params
|
|
87
|
+
end
|
|
88
|
+
|
|
49
89
|
def available_log_types
|
|
50
90
|
types = execute :get_available_log_types
|
|
51
91
|
Array(types).map(&:to_sym)
|
|
@@ -55,11 +95,9 @@ module Selenium
|
|
|
55
95
|
data = execute :get_log, {}, {type: type.to_s}
|
|
56
96
|
|
|
57
97
|
Array(data).map do |l|
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
next
|
|
62
|
-
end
|
|
98
|
+
LogEntry.new l.fetch('level', 'UNKNOWN'), l.fetch('timestamp'), l.fetch('message')
|
|
99
|
+
rescue KeyError
|
|
100
|
+
next
|
|
63
101
|
end
|
|
64
102
|
end
|
|
65
103
|
end # Bridge
|
|
@@ -20,19 +20,40 @@
|
|
|
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
53
|
# driver = Selenium::WebDriver.for(:chrome, options: options)
|
|
35
54
|
#
|
|
55
|
+
# @param [Profile] :profile An instance of a Chrome::Profile Class
|
|
56
|
+
# @param [Array] :encoded_extensions List of extensions that do not need to be Base64 encoded
|
|
36
57
|
# @param [Hash] opts the pre-defined options to create the Chrome::Options with
|
|
37
58
|
# @option opts [Array<String>] :args List of command-line arguments to use when starting Chrome
|
|
38
59
|
# @option opts [String] :binary Path to the Chrome executable to use
|
|
@@ -40,18 +61,33 @@ module Selenium
|
|
|
40
61
|
# @option opts [Array<String>] :extensions A list of paths to (.crx) Chrome extensions to install on startup
|
|
41
62
|
# @option opts [Hash] :options A hash for raw options
|
|
42
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
|
|
43
71
|
#
|
|
44
72
|
|
|
45
|
-
def initialize(**opts)
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
@
|
|
49
|
-
|
|
50
|
-
@options =
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
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(&method(:validate_extension))
|
|
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(&method(:validate_extension))
|
|
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(&method(:encode_extension))
|
|
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
|
|
@@ -19,23 +19,26 @@
|
|
|
19
19
|
|
|
20
20
|
require 'net/http'
|
|
21
21
|
|
|
22
|
-
require 'selenium/webdriver/chrome/bridge'
|
|
23
|
-
require 'selenium/webdriver/chrome/driver'
|
|
24
|
-
require 'selenium/webdriver/chrome/profile'
|
|
25
|
-
require 'selenium/webdriver/chrome/options'
|
|
26
|
-
|
|
27
22
|
module Selenium
|
|
28
23
|
module WebDriver
|
|
29
24
|
module Chrome
|
|
25
|
+
autoload :Features, 'selenium/webdriver/chrome/features'
|
|
26
|
+
autoload :Driver, 'selenium/webdriver/chrome/driver'
|
|
27
|
+
autoload :Profile, 'selenium/webdriver/chrome/profile'
|
|
28
|
+
autoload :Options, 'selenium/webdriver/chrome/options'
|
|
29
|
+
autoload :Service, 'selenium/webdriver/chrome/service'
|
|
30
|
+
|
|
30
31
|
def self.driver_path=(path)
|
|
31
32
|
WebDriver.logger.deprecate 'Selenium::WebDriver::Chrome#driver_path=',
|
|
32
|
-
'Selenium::WebDriver::Chrome::Service#driver_path='
|
|
33
|
+
'Selenium::WebDriver::Chrome::Service#driver_path=',
|
|
34
|
+
id: :driver_path
|
|
33
35
|
Selenium::WebDriver::Chrome::Service.driver_path = path
|
|
34
36
|
end
|
|
35
37
|
|
|
36
38
|
def self.driver_path
|
|
37
39
|
WebDriver.logger.deprecate 'Selenium::WebDriver::Chrome#driver_path',
|
|
38
|
-
'Selenium::WebDriver::Chrome::Service#driver_path'
|
|
40
|
+
'Selenium::WebDriver::Chrome::Service#driver_path',
|
|
41
|
+
id: :driver_path
|
|
39
42
|
Selenium::WebDriver::Chrome::Service.driver_path
|
|
40
43
|
end
|
|
41
44
|
|
|
@@ -50,5 +53,3 @@ module Selenium
|
|
|
50
53
|
end # Chrome
|
|
51
54
|
end # WebDriver
|
|
52
55
|
end # Selenium
|
|
53
|
-
|
|
54
|
-
require 'selenium/webdriver/chrome/service'
|