selenium-webdriver 4.4.0 → 4.8.1

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 (123) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGES +98 -1
  3. data/LICENSE +1 -1
  4. data/NOTICE +1 -1
  5. data/bin/linux/selenium-manager +0 -0
  6. data/bin/macos/selenium-manager +0 -0
  7. data/bin/windows/selenium-manager.exe +0 -0
  8. data/lib/selenium/server.rb +20 -16
  9. data/lib/selenium/webdriver/atoms/findElements.js +0 -0
  10. data/lib/selenium/webdriver/atoms/getAttribute.js +0 -0
  11. data/lib/selenium/webdriver/atoms/isDisplayed.js +0 -0
  12. data/lib/selenium/webdriver/atoms/mutationListener.js +0 -0
  13. data/lib/selenium/webdriver/atoms.rb +2 -3
  14. data/lib/selenium/webdriver/bidi/browsing_context.rb +88 -0
  15. data/lib/selenium/webdriver/bidi/browsing_context_info.rb +35 -0
  16. data/lib/selenium/webdriver/bidi/log/base_log_entry.rb +35 -0
  17. data/lib/selenium/webdriver/bidi/log/console_log_entry.rb +35 -0
  18. data/lib/selenium/webdriver/bidi/log/filter_by.rb +40 -0
  19. data/lib/selenium/webdriver/bidi/log/generic_log_entry.rb +33 -0
  20. data/lib/selenium/webdriver/bidi/log/javascript_log_entry.rb +33 -0
  21. data/lib/selenium/webdriver/bidi/log_inspector.rb +143 -0
  22. data/lib/selenium/webdriver/bidi/navigate_result.rb +33 -0
  23. data/lib/selenium/webdriver/bidi/session.rb +13 -0
  24. data/lib/selenium/webdriver/bidi.rb +3 -2
  25. data/lib/selenium/webdriver/chrome/driver.rb +19 -28
  26. data/lib/selenium/webdriver/chrome/features.rb +5 -72
  27. data/lib/selenium/webdriver/chrome/options.rb +3 -237
  28. data/lib/selenium/webdriver/chrome/profile.rb +3 -83
  29. data/lib/selenium/webdriver/chrome/service.rb +4 -19
  30. data/lib/selenium/webdriver/chromium/driver.rb +61 -0
  31. data/lib/selenium/webdriver/chromium/features.rb +103 -0
  32. data/lib/selenium/webdriver/chromium/options.rb +261 -0
  33. data/lib/selenium/webdriver/chromium/profile.rb +113 -0
  34. data/lib/selenium/webdriver/chromium/service.rb +42 -0
  35. data/lib/selenium/webdriver/chromium.rb +32 -0
  36. data/lib/selenium/webdriver/common/action_builder.rb +11 -48
  37. data/lib/selenium/webdriver/common/child_process.rb +124 -0
  38. data/lib/selenium/webdriver/common/driver.rb +39 -26
  39. data/lib/selenium/webdriver/common/driver_extensions/downloads_files.rb +0 -2
  40. data/lib/selenium/webdriver/common/driver_extensions/full_page_screenshot.rb +0 -1
  41. data/lib/selenium/webdriver/common/driver_extensions/has_addons.rb +0 -2
  42. data/lib/selenium/webdriver/common/driver_extensions/has_apple_permissions.rb +0 -2
  43. data/lib/selenium/webdriver/common/driver_extensions/has_authentication.rb +0 -2
  44. data/lib/selenium/webdriver/common/driver_extensions/has_bidi.rb +0 -2
  45. data/lib/selenium/webdriver/common/driver_extensions/has_casting.rb +0 -1
  46. data/lib/selenium/webdriver/common/driver_extensions/has_cdp.rb +0 -2
  47. data/lib/selenium/webdriver/common/driver_extensions/has_context.rb +0 -2
  48. data/lib/selenium/webdriver/common/driver_extensions/has_debugger.rb +0 -2
  49. data/lib/selenium/webdriver/common/driver_extensions/has_devtools.rb +0 -2
  50. data/lib/selenium/webdriver/common/driver_extensions/has_launching.rb +0 -2
  51. data/lib/selenium/webdriver/common/driver_extensions/has_location.rb +1 -2
  52. data/lib/selenium/webdriver/common/driver_extensions/has_log_events.rb +0 -1
  53. data/lib/selenium/webdriver/common/driver_extensions/has_network_conditions.rb +0 -2
  54. data/lib/selenium/webdriver/common/driver_extensions/has_network_interception.rb +0 -2
  55. data/lib/selenium/webdriver/common/driver_extensions/has_permissions.rb +0 -2
  56. data/lib/selenium/webdriver/common/driver_extensions/has_pinned_scripts.rb +0 -2
  57. data/lib/selenium/webdriver/common/element.rb +7 -7
  58. data/lib/selenium/webdriver/common/error.rb +0 -2
  59. data/lib/selenium/webdriver/common/html5/shared_web_storage.rb +2 -2
  60. data/lib/selenium/webdriver/common/interactions/pointer_actions.rb +3 -3
  61. data/lib/selenium/webdriver/common/interactions/pointer_move.rb +2 -2
  62. data/lib/selenium/webdriver/common/interactions/scroll.rb +7 -5
  63. data/lib/selenium/webdriver/common/logger.rb +10 -2
  64. data/lib/selenium/webdriver/common/options.rb +30 -8
  65. data/lib/selenium/webdriver/common/platform.rb +4 -1
  66. data/lib/selenium/webdriver/common/profile_helper.rb +1 -1
  67. data/lib/selenium/webdriver/common/proxy.rb +1 -1
  68. data/lib/selenium/webdriver/common/selenium_manager.rb +89 -0
  69. data/lib/selenium/webdriver/common/service.rb +16 -8
  70. data/lib/selenium/webdriver/common/service_manager.rb +2 -10
  71. data/lib/selenium/webdriver/common/shadow_root.rb +1 -2
  72. data/lib/selenium/webdriver/common/socket_lock.rb +2 -2
  73. data/lib/selenium/webdriver/common/takes_screenshot.rb +2 -3
  74. data/lib/selenium/webdriver/common/target_locator.rb +2 -3
  75. data/lib/selenium/webdriver/common/timeouts.rb +2 -2
  76. data/lib/selenium/webdriver/common/virtual_authenticator/credential.rb +8 -6
  77. data/lib/selenium/webdriver/common/virtual_authenticator/virtual_authenticator.rb +0 -1
  78. data/lib/selenium/webdriver/common/virtual_authenticator/virtual_authenticator_options.rb +16 -16
  79. data/lib/selenium/webdriver/common/websocket_connection.rb +10 -2
  80. data/lib/selenium/webdriver/common.rb +2 -0
  81. data/lib/selenium/webdriver/devtools/console_event.rb +0 -2
  82. data/lib/selenium/webdriver/devtools/exception_event.rb +0 -2
  83. data/lib/selenium/webdriver/devtools/mutation_event.rb +0 -2
  84. data/lib/selenium/webdriver/devtools/network_interceptor.rb +3 -6
  85. data/lib/selenium/webdriver/devtools/pinned_script.rb +0 -2
  86. data/lib/selenium/webdriver/devtools/request.rb +0 -2
  87. data/lib/selenium/webdriver/devtools/response.rb +0 -2
  88. data/lib/selenium/webdriver/devtools.rb +0 -1
  89. data/lib/selenium/webdriver/edge/driver.rb +20 -3
  90. data/lib/selenium/webdriver/edge/features.rb +3 -4
  91. data/lib/selenium/webdriver/edge/options.rb +3 -5
  92. data/lib/selenium/webdriver/edge/profile.rb +2 -2
  93. data/lib/selenium/webdriver/edge/service.rb +2 -2
  94. data/lib/selenium/webdriver/firefox/driver.rb +19 -2
  95. data/lib/selenium/webdriver/firefox/features.rb +5 -2
  96. data/lib/selenium/webdriver/firefox/options.rb +6 -2
  97. data/lib/selenium/webdriver/firefox/profile.rb +6 -6
  98. data/lib/selenium/webdriver/firefox/service.rb +0 -1
  99. data/lib/selenium/webdriver/ie/driver.rb +20 -1
  100. data/lib/selenium/webdriver/ie/service.rb +1 -2
  101. data/lib/selenium/webdriver/remote/{commands.rb → bridge/commands.rb} +0 -2
  102. data/lib/selenium/webdriver/remote/bridge.rb +9 -15
  103. data/lib/selenium/webdriver/remote/capabilities.rb +34 -12
  104. data/lib/selenium/webdriver/remote/driver.rb +13 -13
  105. data/lib/selenium/webdriver/remote/http/curb.rb +0 -2
  106. data/lib/selenium/webdriver/remote/http/default.rb +1 -0
  107. data/lib/selenium/webdriver/remote/response.rb +0 -1
  108. data/lib/selenium/webdriver/remote.rb +0 -1
  109. data/lib/selenium/webdriver/safari/driver.rb +20 -1
  110. data/lib/selenium/webdriver/safari/features.rb +0 -2
  111. data/lib/selenium/webdriver/safari/options.rb +5 -1
  112. data/lib/selenium/webdriver/safari.rb +1 -1
  113. data/lib/selenium/webdriver/support/color.rb +15 -15
  114. data/lib/selenium/webdriver/support/guards/guard.rb +0 -2
  115. data/lib/selenium/webdriver/support/guards/guard_condition.rb +0 -2
  116. data/lib/selenium/webdriver/support/relative_locator.rb +0 -1
  117. data/lib/selenium/webdriver/support/select.rb +3 -1
  118. data/lib/selenium/webdriver/version.rb +1 -1
  119. data/lib/selenium/webdriver.rb +3 -4
  120. data/selenium-webdriver.gemspec +8 -9
  121. metadata +31 -74
  122. data/lib/selenium/webdriver/support/cdp/domain.rb.erb +0 -63
  123. data/lib/selenium/webdriver/support/cdp_client_generator.rb +0 -108
@@ -17,6 +17,8 @@
17
17
  # specific language governing permissions and limitations
18
18
  # under the License.
19
19
 
20
+ require 'selenium/webdriver/chromium/profile'
21
+
20
22
  module Selenium
21
23
  module WebDriver
22
24
  module Chrome
@@ -24,89 +26,7 @@ module Selenium
24
26
  # @private
25
27
  #
26
28
 
27
- class Profile
28
- include ProfileHelper
29
-
30
- def initialize(model = nil)
31
- @model = verify_model(model)
32
- @extensions = []
33
- @encoded_extensions = []
34
- @directory = nil
35
- end
36
-
37
- def add_extension(path)
38
- raise Error::WebDriverError, "could not find extension at #{path.inspect}" unless File.file?(path)
39
-
40
- @extensions << path
41
- end
42
-
43
- def add_encoded_extension(encoded)
44
- @encoded_extensions << encoded
45
- end
46
-
47
- def directory
48
- @directory || layout_on_disk
49
- end
50
-
51
- #
52
- # Set a preference in the profile.
53
- #
54
- # See https://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/pref_names.cc
55
- #
56
-
57
- def []=(key, value)
58
- parts = key.split('.')
59
- parts[0..-2].inject(prefs) { |a, e| a[e] ||= {} }[parts.last] = value
60
- end
61
-
62
- def [](key)
63
- parts = key.split('.')
64
- parts.inject(prefs) { |a, e| a.fetch(e) }
65
- end
66
-
67
- def layout_on_disk
68
- @directory = @model ? create_tmp_copy(@model) : Dir.mktmpdir('webdriver-chrome-profile')
69
- FileReaper << @directory
70
-
71
- write_prefs_to @directory
72
-
73
- @directory
74
- end
75
-
76
- def as_json(*)
77
- extensions = @extensions.map do |crx_path|
78
- File.open(crx_path, 'rb') { |crx_file| Base64.strict_encode64 crx_file.read }
79
- end
80
-
81
- extensions.concat(@encoded_extensions)
82
-
83
- opts = {'directory' => directory || layout_on_disk}
84
- opts['extensions'] = extensions if extensions.any?
85
- opts
86
- end
87
-
88
- private
89
-
90
- def write_prefs_to(dir)
91
- prefs_file = prefs_file_for(dir)
92
-
93
- FileUtils.mkdir_p File.dirname(prefs_file)
94
- File.open(prefs_file, 'w') { |file| file << JSON.generate(prefs) }
95
- end
96
-
97
- def prefs
98
- @prefs ||= read_model_prefs
99
- end
100
-
101
- def read_model_prefs
102
- return {} unless @model
103
-
104
- JSON.parse File.read(prefs_file_for(@model))
105
- end
106
-
107
- def prefs_file_for(dir)
108
- File.join dir, 'Default', 'Preferences'
109
- end
29
+ class Profile < Chromium::Profile
110
30
  end # Profile
111
31
  end # Chrome
112
32
  end # WebDriver
@@ -17,35 +17,20 @@
17
17
  # specific language governing permissions and limitations
18
18
  # under the License.
19
19
 
20
+ require 'selenium/webdriver/chromium/service'
21
+
20
22
  module Selenium
21
23
  module WebDriver
22
24
  module Chrome
23
- class Service < WebDriver::Service
25
+ class Service < Chromium::Service
24
26
  DEFAULT_PORT = 9515
25
27
  EXECUTABLE = 'chromedriver'
26
28
  MISSING_TEXT = <<~ERROR
27
29
  Unable to find chromedriver. Please download the server from
28
30
  https://chromedriver.storage.googleapis.com/index.html and place it somewhere on your PATH.
29
- More info at https://github.com/SeleniumHQ/selenium/wiki/ChromeDriver.
31
+ More info at https://www.selenium.dev/documentation/webdriver/getting_started/install_drivers/?language=ruby.
30
32
  ERROR
31
33
  SHUTDOWN_SUPPORTED = true
32
-
33
- private
34
-
35
- # NOTE: This processing is deprecated
36
- def extract_service_args(driver_opts)
37
- driver_args = super
38
- driver_opts = driver_opts.dup
39
- driver_args << "--log-path=#{driver_opts.delete(:log_path)}" if driver_opts.key?(:log_path)
40
- driver_args << "--url-base=#{driver_opts.delete(:url_base)}" if driver_opts.key?(:url_base)
41
- driver_args << "--port-server=#{driver_opts.delete(:port_server)}" if driver_opts.key?(:port_server)
42
- if driver_opts.key?(:whitelisted_ips)
43
- driver_args << "--whitelisted-ips=#{driver_opts.delete(:whitelisted_ips)}"
44
- end
45
- driver_args << "--verbose" if driver_opts.key?(:verbose)
46
- driver_args << "--silent" if driver_opts.key?(:silent)
47
- driver_args
48
- end
49
34
  end # Service
50
35
  end # Chrome
51
36
  end # WebDriver
@@ -0,0 +1,61 @@
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 Chromium
23
+ #
24
+ # Driver implementation for Chrome.
25
+ # @api private
26
+ #
27
+
28
+ class Driver < WebDriver::Driver
29
+ EXTENSIONS = [DriverExtensions::HasCDP,
30
+ DriverExtensions::HasBiDi,
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
45
+
46
+ protected
47
+
48
+ def devtools_url
49
+ uri = URI(devtools_address)
50
+ response = Net::HTTP.get(uri.hostname, '/json/version', uri.port)
51
+
52
+ JSON.parse(response)['webSocketDebuggerUrl']
53
+ end
54
+
55
+ def devtools_version
56
+ Integer(capabilities.browser_version.split('.').first)
57
+ end
58
+ end # Driver
59
+ end # Chromium
60
+ end # WebDriver
61
+ end # Selenium
@@ -0,0 +1,103 @@
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 Chromium
23
+ module Features
24
+ CHROMIUM_COMMANDS = {
25
+ launch_app: [:post, 'session/:session_id/chromium/launch_app'],
26
+ get_network_conditions: [:get, 'session/:session_id/chromium/network_conditions'],
27
+ set_network_conditions: [:post, 'session/:session_id/chromium/network_conditions'],
28
+ delete_network_conditions: [:delete, 'session/:session_id/chromium/network_conditions'],
29
+ set_permission: [:post, 'session/:session_id/permissions'],
30
+ get_available_log_types: [:get, 'session/:session_id/se/log/types'],
31
+ get_log: [:post, 'session/:session_id/se/log']
32
+ }.freeze
33
+
34
+ def commands(command)
35
+ CHROME_COMMANDS[command] || self.class::COMMANDS[command]
36
+ end
37
+
38
+ def launch_app(id)
39
+ execute :launch_app, {}, {id: id}
40
+ end
41
+
42
+ def cast_sinks
43
+ execute :get_cast_sinks
44
+ end
45
+
46
+ def cast_sink_to_use=(name)
47
+ execute :set_cast_sink_to_use, {}, {sinkName: name}
48
+ end
49
+
50
+ def cast_issue_message
51
+ execute :cast_issue_message
52
+ end
53
+
54
+ def start_cast_tab_mirroring(name)
55
+ execute :start_cast_tab_mirroring, {}, {sinkName: name}
56
+ end
57
+
58
+ def start_cast_desktop_mirroring(name)
59
+ execute :start_cast_desktop_mirroring, {}, {sinkName: name}
60
+ end
61
+
62
+ def stop_casting(name)
63
+ execute :stop_casting, {}, {sinkName: name}
64
+ end
65
+
66
+ def set_permission(name, value)
67
+ execute :set_permission, {}, {descriptor: {name: name}, state: value}
68
+ end
69
+
70
+ def network_conditions
71
+ execute :get_network_conditions
72
+ end
73
+
74
+ def network_conditions=(conditions)
75
+ execute :set_network_conditions, {}, {network_conditions: conditions}
76
+ end
77
+
78
+ def delete_network_conditions
79
+ execute :delete_network_conditions
80
+ end
81
+
82
+ def send_command(command_params)
83
+ execute :send_command, {}, command_params
84
+ end
85
+
86
+ def available_log_types
87
+ types = execute :get_available_log_types
88
+ Array(types).map(&:to_sym)
89
+ end
90
+
91
+ def log(type)
92
+ data = execute :get_log, {}, {type: type.to_s}
93
+
94
+ Array(data).map do |l|
95
+ LogEntry.new l.fetch('level', 'UNKNOWN'), l.fetch('timestamp'), l.fetch('message')
96
+ rescue KeyError
97
+ next
98
+ end
99
+ end
100
+ end # Bridge
101
+ end # Chromium
102
+ end # WebDriver
103
+ end # Selenium
@@ -0,0 +1,261 @@
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 Chromium
23
+ class Options < WebDriver::Options
24
+ attr_accessor :profile, :logging_prefs
25
+
26
+ # see: http://chromedriver.chromium.org/capabilities
27
+ CAPABILITIES = {args: 'args',
28
+ binary: 'binary',
29
+ local_state: 'localState',
30
+ prefs: 'prefs',
31
+ detach: 'detach',
32
+ debugger_address: 'debuggerAddress',
33
+ exclude_switches: 'excludeSwitches',
34
+ minidump_path: 'minidumpPath',
35
+ emulation: 'mobileEmulation',
36
+ perf_logging_prefs: 'perfLoggingPrefs',
37
+ window_types: 'windowTypes',
38
+ android_package: 'androidPackage',
39
+ android_activity: 'androidActivity',
40
+ android_device_serial: 'androidDeviceSerial',
41
+ android_use_running_app: 'androidUseRunningApp'}.freeze
42
+
43
+ # NOTE: special handling of 'extensions' to validate when set instead of when used
44
+ attr_reader :extensions
45
+
46
+ # Create a new Options instance.
47
+ #
48
+ # @example
49
+ # options = Selenium::WebDriver::Chrome::Options.new(args: ['start-maximized', 'user-data-dir=/tmp/temp_profile'])
50
+ # driver = Selenium::WebDriver.for(:chrome, capabilities: options)
51
+ #
52
+ # @param [Profile] profile An instance of a Chrome::Profile Class
53
+ # @param [Hash] opts the pre-defined options to create the Chrome::Options with
54
+ # @option opts [Array] encoded_extensions List of extensions that do not need to be Base64 encoded
55
+ # @option opts [Array<String>] args List of command-line arguments to use when starting Chrome
56
+ # @option opts [String] binary Path to the Chrome executable to use
57
+ # @option opts [Hash] prefs A hash with each entry consisting of the name of the preference and its value
58
+ # @option opts [Array<String>] extensions A list of paths to (.crx) Chrome extensions to install on startup
59
+ # @option opts [Hash] options A hash for raw options
60
+ # @option opts [Hash] emulation A hash for raw emulation options
61
+ # @option opts [Hash] local_state A hash for the Local State file in the user data folder
62
+ # @option opts [Boolean] detach whether browser is closed when the driver is sent the quit command
63
+ # @option opts [String] debugger_address address of a Chrome debugger server to connect to
64
+ # @option opts [Array<String>] exclude_switches command line switches to exclude
65
+ # @option opts [String] minidump_path Directory to store Chrome minidumps (linux only)
66
+ # @option opts [Hash] perf_logging_prefs A hash for performance logging preferences
67
+ # @option opts [Array<String>] window_types A list of window types to appear in the list of window handles
68
+ #
69
+
70
+ def initialize(profile: nil, **opts)
71
+ super(**opts)
72
+
73
+ @profile = profile
74
+
75
+ @options = {args: [],
76
+ prefs: {},
77
+ emulation: {},
78
+ extensions: [],
79
+ local_state: {},
80
+ exclude_switches: [],
81
+ perf_logging_prefs: {},
82
+ window_types: []}.merge(@options)
83
+
84
+ @logging_prefs = options.delete(:logging_prefs) || {}
85
+ @encoded_extensions = @options.delete(:encoded_extensions) || []
86
+ @extensions = []
87
+ @options.delete(:extensions).each { |ext| validate_extension(ext) }
88
+ end
89
+
90
+ #
91
+ # Add an extension by local path.
92
+ #
93
+ # @example
94
+ # options = Selenium::WebDriver::Chrome::Options.new
95
+ # options.add_extension('/path/to/extension.crx')
96
+ #
97
+ # @param [String] path The local path to the .crx file
98
+ #
99
+
100
+ def add_extension(path)
101
+ validate_extension(path)
102
+ end
103
+
104
+ #
105
+ # Add an extension by local path.
106
+ #
107
+ # @example
108
+ # extensions = ['/path/to/extension.crx', '/path/to/other.crx']
109
+ # options = Selenium::WebDriver::Chrome::Options.new
110
+ # options.extensions = extensions
111
+ #
112
+ # @param [Array<String>] extensions A list of paths to (.crx) Chrome extensions to install on startup
113
+ #
114
+
115
+ def extensions=(extensions)
116
+ extensions.each { |ext| validate_extension(ext) }
117
+ end
118
+
119
+ #
120
+ # Add an extension by Base64-encoded string.
121
+ #
122
+ # @example
123
+ # options = Selenium::WebDriver::Chrome::Options.new
124
+ # options.add_encoded_extension(encoded_string)
125
+ #
126
+ # @param [String] encoded The Base64-encoded string of the .crx file
127
+ #
128
+
129
+ def add_encoded_extension(encoded)
130
+ @encoded_extensions << encoded
131
+ end
132
+
133
+ #
134
+ # Add a command-line argument to use when starting Chrome.
135
+ #
136
+ # @example Start Chrome maximized
137
+ # options = Selenium::WebDriver::Chrome::Options.new
138
+ # options.add_argument('start-maximized')
139
+ #
140
+ # @param [String] arg The command-line argument to add
141
+ #
142
+
143
+ def add_argument(arg)
144
+ @options[:args] << arg
145
+ end
146
+
147
+ #
148
+ # Add a preference that is only applied to the user profile in use.
149
+ #
150
+ # @example Set the default homepage
151
+ # options = Selenium::WebDriver::Chrome::Options.new
152
+ # options.add_preference('homepage', 'http://www.seleniumhq.com/')
153
+ #
154
+ # @param [String] name Key of the preference
155
+ # @param [Boolean, String, Integer] value Value of the preference
156
+ #
157
+
158
+ def add_preference(name, value)
159
+ @options[:prefs][name] = value
160
+ end
161
+
162
+ #
163
+ # Run Chrome in headless mode.
164
+ # Old headless uses a non-production browser and is set with `--headless`
165
+ # Native headless from v86 - v108 is set with `--headless=chrome`
166
+ # Native headless from v109+ is set with `--headless=new`
167
+ #
168
+ # @example Enable headless mode
169
+ # options = Selenium::WebDriver::Chrome::Options.new
170
+ # options.headless!
171
+ #
172
+
173
+ def headless!
174
+ WebDriver.logger.deprecate('`Options#headless!`',
175
+ "`Options#add_argument('--headless=new')`",
176
+ id: :headless)
177
+ add_argument '--headless'
178
+ end
179
+
180
+ #
181
+ # Add emulation device information
182
+ #
183
+ # see: http://chromedriver.chromium.org/mobile-emulation
184
+ #
185
+ # @example Start Chrome in mobile emulation mode by device name
186
+ # options = Selenium::WebDriver::Chrome::Options.new
187
+ # options.add_emulation(device_name: 'iPhone 6')
188
+ #
189
+ # @example Start Chrome in mobile emulation mode by device metrics
190
+ # options = Selenium::WebDriver::Chrome::Options.new
191
+ # options.add_emulation(device_metrics: {width: 400, height: 800, pixelRatio: 1, touch: true})
192
+ #
193
+ # @param [Hash] opts the pre-defined options for adding mobile emulation values
194
+ # @option opts [String] :device_name A valid device name from the Chrome DevTools Emulation panel
195
+ # @option opts [Hash] :device_metrics Hash containing width, height, pixelRatio, touch
196
+ # @option opts [String] :user_agent Full user agent
197
+ #
198
+
199
+ def add_emulation(**opts)
200
+ @options[:emulation] = opts
201
+ end
202
+
203
+ #
204
+ # Enables mobile browser use on Android.
205
+ #
206
+ # @see https://chromedriver.chromium.org/getting-started/getting-started---android
207
+ #
208
+ # @param [String] package The package name of the Chrome or WebView app.
209
+ # @param [String] serial_number The device serial number on which to launch the Chrome or WebView app.
210
+ # @param [String] use_running_app When true uses an already-running Chrome or WebView app,
211
+ # instead of launching the app with a clear data directory.
212
+ # @param [String] activity Name of the Activity hosting the WebView (Not available on Chrome Apps).
213
+ #
214
+
215
+ def enable_android(package: 'com.android.chrome', serial_number: nil, use_running_app: nil, activity: nil)
216
+ @options[:android_package] = package
217
+ @options[:android_activity] = activity unless activity.nil?
218
+ @options[:android_device_serial] = serial_number unless serial_number.nil?
219
+ @options[:android_use_running_app] = use_running_app unless use_running_app.nil?
220
+ end
221
+
222
+ protected
223
+
224
+ def process_browser_options(browser_options)
225
+ enable_logging(browser_options) unless @logging_prefs.empty?
226
+
227
+ options = browser_options[self.class::KEY]
228
+ options['binary'] ||= binary_path if binary_path
229
+
230
+ if @profile
231
+ options['args'] ||= []
232
+ options['args'] << "--user-data-dir=#{@profile.directory}"
233
+ end
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)
257
+ end
258
+ end # Options
259
+ end # Chromium
260
+ end # WebDriver
261
+ end # Selenium
@@ -0,0 +1,113 @@
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 Chromium
23
+ #
24
+ # @private
25
+ #
26
+
27
+ class Profile
28
+ include ProfileHelper
29
+
30
+ def initialize(model = nil)
31
+ @model = verify_model(model)
32
+ @extensions = []
33
+ @encoded_extensions = []
34
+ @directory = nil
35
+ end
36
+
37
+ def add_extension(path)
38
+ raise Error::WebDriverError, "could not find extension at #{path.inspect}" unless File.file?(path)
39
+
40
+ @extensions << path
41
+ end
42
+
43
+ def add_encoded_extension(encoded)
44
+ @encoded_extensions << encoded
45
+ end
46
+
47
+ def directory
48
+ @directory || layout_on_disk
49
+ end
50
+
51
+ #
52
+ # Set a preference in the profile.
53
+ #
54
+ # See https://src.chromium.org/viewvc/chrome/trunk/src/chrome/common/pref_names.cc
55
+ #
56
+
57
+ def []=(key, value)
58
+ parts = key.split('.')
59
+ parts[0..-2].inject(prefs) { |a, e| a[e] ||= {} }[parts.last] = value
60
+ end
61
+
62
+ def [](key)
63
+ parts = key.split('.')
64
+ parts.inject(prefs) { |a, e| a.fetch(e) }
65
+ end
66
+
67
+ def layout_on_disk
68
+ @directory = @model ? create_tmp_copy(@model) : Dir.mktmpdir('webdriver-chrome-profile')
69
+ FileReaper << @directory
70
+
71
+ write_prefs_to @directory
72
+
73
+ @directory
74
+ end
75
+
76
+ def as_json(*)
77
+ extensions = @extensions.map do |crx_path|
78
+ File.open(crx_path, 'rb') { |crx_file| Base64.strict_encode64 crx_file.read }
79
+ end
80
+
81
+ extensions.concat(@encoded_extensions)
82
+
83
+ opts = {'directory' => directory || layout_on_disk}
84
+ opts['extensions'] = extensions if extensions.any?
85
+ opts
86
+ end
87
+
88
+ private
89
+
90
+ def write_prefs_to(dir)
91
+ prefs_file = prefs_file_for(dir)
92
+
93
+ FileUtils.mkdir_p File.dirname(prefs_file)
94
+ File.open(prefs_file, 'w') { |file| file << JSON.generate(prefs) }
95
+ end
96
+
97
+ def prefs
98
+ @prefs ||= read_model_prefs
99
+ end
100
+
101
+ def read_model_prefs
102
+ return {} unless @model
103
+
104
+ JSON.parse File.read(prefs_file_for(@model))
105
+ end
106
+
107
+ def prefs_file_for(dir)
108
+ File.join dir, 'Default', 'Preferences'
109
+ end
110
+ end # Profile
111
+ end # Chromium
112
+ end # WebDriver
113
+ end # Selenium