appium_lib_core 4.1.0 → 9.0.0

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 (70) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +325 -270
  3. data/README.md +65 -15
  4. data/Rakefile +5 -22
  5. data/appium_lib_core.gemspec +6 -9
  6. data/bin/console +0 -4
  7. data/lib/appium_lib_core/android/device/auth_finger_print.rb +4 -1
  8. data/lib/appium_lib_core/android/device/network.rb +10 -0
  9. data/lib/appium_lib_core/android/device/performance.rb +3 -0
  10. data/lib/appium_lib_core/android/device/screen.rb +2 -0
  11. data/lib/appium_lib_core/android/device.rb +80 -17
  12. data/lib/appium_lib_core/common/base/bridge.rb +309 -93
  13. data/lib/appium_lib_core/common/base/capabilities.rb +21 -8
  14. data/lib/appium_lib_core/common/{command/mjsonwp.rb → base/device_ime.rb} +33 -12
  15. data/lib/appium_lib_core/common/base/driver.rb +258 -331
  16. data/lib/appium_lib_core/common/base/driver_settings.rb +51 -0
  17. data/lib/appium_lib_core/common/base/has_location.rb +80 -0
  18. data/lib/appium_lib_core/common/base/has_network_connection.rb +56 -0
  19. data/lib/appium_lib_core/common/base/http_default.rb +15 -38
  20. data/lib/appium_lib_core/{ios/uiautomation/bridge.rb → common/base/remote_status.rb} +9 -8
  21. data/lib/appium_lib_core/common/base/rotable.rb +62 -0
  22. data/lib/appium_lib_core/common/base/screenshot.rb +8 -8
  23. data/lib/appium_lib_core/common/base/search_context.rb +13 -17
  24. data/lib/appium_lib_core/common/base.rb +1 -5
  25. data/lib/appium_lib_core/common/command.rb +244 -4
  26. data/lib/appium_lib_core/common/device/app_management.rb +2 -26
  27. data/lib/appium_lib_core/common/device/context.rb +1 -5
  28. data/lib/appium_lib_core/common/device/image_comparison.rb +12 -4
  29. data/lib/appium_lib_core/common/device/keyevent.rb +4 -4
  30. data/lib/appium_lib_core/common/device/{touch_actions.rb → orientation.rb} +6 -10
  31. data/lib/appium_lib_core/common/error.rb +4 -5
  32. data/lib/appium_lib_core/common/log.rb +5 -4
  33. data/lib/appium_lib_core/common/wait.rb +38 -6
  34. data/lib/appium_lib_core/device.rb +3 -9
  35. data/lib/appium_lib_core/driver.rb +189 -158
  36. data/lib/appium_lib_core/{patch.rb → element.rb} +62 -25
  37. data/lib/appium_lib_core/ios/xcuitest/device.rb +2 -0
  38. data/lib/appium_lib_core/{common/base/command.rb → mac2/bridge.rb} +9 -8
  39. data/lib/appium_lib_core/mac2/device/screen.rb +48 -0
  40. data/lib/appium_lib_core/mac2/device.rb +92 -0
  41. data/lib/appium_lib_core/{ios.rb → mac2.rb} +2 -5
  42. data/lib/appium_lib_core/support/event_firing_bridge.rb +57 -0
  43. data/lib/appium_lib_core/version.rb +2 -2
  44. data/lib/appium_lib_core.rb +23 -10
  45. metadata +28 -94
  46. data/.github/ISSUE_TEMPLATE/issue-report.md +0 -29
  47. data/.github/contributing.md +0 -26
  48. data/.github/issue_template.md +0 -20
  49. data/.github/workflows/unittest.yml +0 -68
  50. data/.gitignore +0 -18
  51. data/.rubocop.yml +0 -58
  52. data/azure-pipelines.yml +0 -15
  53. data/ci-jobs/functional/android_setup.yml +0 -3
  54. data/ci-jobs/functional/ios_setup.yml +0 -7
  55. data/ci-jobs/functional/publish_test_result.yml +0 -18
  56. data/ci-jobs/functional/run_appium.yml +0 -25
  57. data/ci-jobs/functional/start-emulator.sh +0 -26
  58. data/ci-jobs/functional_test.yml +0 -298
  59. data/docs/mobile_command.md +0 -34
  60. data/lib/appium_lib_core/common/base/bridge/mjsonwp.rb +0 -81
  61. data/lib/appium_lib_core/common/base/bridge/w3c.rb +0 -252
  62. data/lib/appium_lib_core/common/command/common.rb +0 -110
  63. data/lib/appium_lib_core/common/command/w3c.rb +0 -56
  64. data/lib/appium_lib_core/common/device/value.rb +0 -52
  65. data/lib/appium_lib_core/common/touch_action/multi_touch.rb +0 -56
  66. data/lib/appium_lib_core/common/touch_action/touch_actions.rb +0 -203
  67. data/lib/appium_lib_core/ios/uiautomation/device.rb +0 -44
  68. data/lib/appium_lib_core/ios/uiautomation/patch.rb +0 -34
  69. data/release_notes.md +0 -816
  70. data/script/commands.rb +0 -200
@@ -16,52 +16,69 @@ module Appium
16
16
  module Core
17
17
  class Base
18
18
  class Bridge < ::Selenium::WebDriver::Remote::Bridge
19
+ include Device::DeviceLock
20
+ include Device::Keyboard
21
+ include Device::ImeActions
22
+ include Device::Setting
23
+ include Device::Context
24
+ include Device::FileManagement
25
+ include Device::KeyEvent
26
+ include Device::ImageComparison
27
+ include Device::AppManagement
28
+ include Device::AppState
29
+ include Device::ScreenRecord::Command
30
+ include Device::Device
31
+ include Device::ExecuteDriver
32
+ include Device::Orientation
33
+
19
34
  # Prefix for extra capability defined by W3C
20
35
  APPIUM_PREFIX = 'appium:'
21
36
 
22
- # TODO: Remove the forceMjsonwp after Appium server won't need it
23
- FORCE_MJSONWP = :forceMjsonwp
37
+ # No 'browserName' means the session is native appium connection
38
+ APPIUM_NATIVE_BROWSER_NAME = 'appium'
39
+
40
+ attr_reader :available_commands
41
+
42
+ def browser
43
+ @browser ||= begin
44
+ name = @capabilities&.browser_name
45
+ name ? name.tr(' ', '_').downcase.to_sym : 'unknown'
46
+ rescue KeyError
47
+ APPIUM_NATIVE_BROWSER_NAME
48
+ end
49
+ end
24
50
 
25
- # Almost same as self.handshake in ::Selenium::WebDriver::Remote::Bridge
51
+ # Appium only.
52
+ # Attach to an existing session.
26
53
  #
27
- # Implements protocol handshake which:
54
+ # @param [String] The session id to attach to.
55
+ # @param [String] platform_name The platform name to keep in the dummy capabilities
56
+ # @param [String] platform_name The automation name to keep in the dummy capabilities
57
+ # @return [::Appium::Core::Base::Capabilities]
28
58
  #
29
- # 1. Creates session with driver.
30
- # 2. Sniffs response.
31
- # 3. Based on the response, understands which dialect we should use.
59
+ # @example
32
60
  #
33
- # @return [Bridge::MJSONWP, Bridge::W3C]
61
+ # new_driver = ::Appium::Core::Driver.attach_to(
62
+ # driver.session_id,
63
+ # url: 'http://127.0.0.1:4723/wd/hub', automation_name: 'UiAutomator2', platform_name: 'Android'
64
+ # )
34
65
  #
35
- def self.handshake(**opts)
36
- desired_capabilities = opts.delete(:desired_capabilities) { ::Selenium::WebDriver::Remote::Capabilities.new }
37
-
38
- if desired_capabilities.is_a?(Symbol)
39
- unless ::Selenium::WebDriver::Remote::Capabilities.respond_to?(desired_capabilities)
40
- raise ::Selenium::WebDriver::Error::WebDriverError, "invalid desired capability: #{desired_capabilities.inspect}"
41
- end
42
-
43
- desired_capabilities = ::Selenium::WebDriver::Remote::Capabilities.__send__(desired_capabilities)
44
- end
45
-
46
- bridge = new(opts)
47
- capabilities = bridge.create_session(desired_capabilities)
48
-
49
- case bridge.dialect
50
- when :oss # for MJSONWP
51
- Bridge::MJSONWP.new(capabilities, bridge.session_id, **opts)
52
- when :w3c
53
- Bridge::W3C.new(capabilities, bridge.session_id, **opts)
54
- else
55
- raise CoreError, 'cannot understand dialect'
56
- end
66
+ def attach_to(session_id, platform_name, automation_name)
67
+ @available_commands = ::Appium::Core::Commands::COMMANDS.dup
68
+ @session_id = session_id
69
+
70
+ # generate a dummy capabilities instance which only has the given platformName and automationName
71
+ @capabilities = ::Appium::Core::Base::Capabilities.new(
72
+ 'platformName' => platform_name,
73
+ 'automationName' => automation_name
74
+ )
57
75
  end
58
76
 
59
77
  # Override
60
- # Creates session handling both OSS and W3C dialects.
61
- # Copy from Selenium::WebDriver::Remote::Bridge to keep using +merged_capabilities+ for Appium
78
+ # Creates session handling.
62
79
  #
63
- # @param [::Selenium::WebDriver::Remote::W3C::Capabilities, Hash] desired_capabilities A capability
64
- # @return [::Selenium::WebDriver::Remote::Capabilities, ::Selenium::WebDriver::Remote::W3C::Capabilities]
80
+ # @param [::Appium::Core::Base::Capabilities, Hash] capabilities A capability
81
+ # @return [::Appium::Core::Base::Capabilities]
65
82
  #
66
83
  # @example
67
84
  #
@@ -79,44 +96,31 @@ module Appium
79
96
  # }
80
97
  # }
81
98
  # core = ::Appium::Core.for(caps)
82
- # driver = core.start_driver #=> driver.dialect == :w3c if the Appium server support W3C.
99
+ # driver = core.start_driver
83
100
  #
84
- def create_session(desired_capabilities)
85
- response = execute(:new_session, {}, merged_capabilities(desired_capabilities))
86
-
87
- @session_id = response['sessionId']
88
- oss_status = response['status'] # for compatibility with Appium 1.7.1-
89
- value = response['value']
90
-
91
- if value.is_a?(Hash) # include for W3C format
92
- @session_id = value['sessionId'] if value.key?('sessionId')
101
+ def create_session(capabilities)
102
+ @available_commands = ::Appium::Core::Commands::COMMANDS.dup
93
103
 
94
- if value.key?('capabilities')
95
- value = value['capabilities']
96
- elsif value.key?('value')
97
- value = value['value']
98
- end
99
- end
104
+ always_match = add_appium_prefix(capabilities)
105
+ response = execute(:new_session, {}, { capabilities: { alwaysMatch: always_match, firstMatch: [{}] } })
100
106
 
107
+ @session_id = response['sessionId']
101
108
  raise ::Selenium::WebDriver::Error::WebDriverError, 'no sessionId in returned payload' unless @session_id
102
109
 
103
- json_create(oss_status, value)
110
+ @capabilities = json_create(response['capabilities'])
104
111
  end
105
112
 
106
113
  # Append +appium:+ prefix for Appium following W3C spec
107
114
  # https://www.w3.org/TR/webdriver/#dfn-validate-capabilities
108
115
  #
109
- # @param [::Selenium::WebDriver::Remote::W3C::Capabilities, Hash] capabilities A capability
110
- # @return [::Selenium::WebDriver::Remote::W3C::Capabilities]
116
+ # @param [::Appium::Core::Base::Capabilities, Hash] capabilities A capability
117
+ # @return [::Appium::Core::Base::Capabilities]
111
118
  def add_appium_prefix(capabilities)
112
- w3c_capabilities = ::Selenium::WebDriver::Remote::W3C::Capabilities.new
119
+ w3c_capabilities = ::Appium::Core::Base::Capabilities.new
113
120
 
114
- capabilities = capabilities.__send__(:capabilities) unless capabilities.is_a?(Hash)
121
+ capabilities = capabilities.send(:capabilities) unless capabilities.is_a?(Hash)
115
122
 
116
123
  capabilities.each do |name, value|
117
- next if value.nil?
118
- next if value.is_a?(String) && value.empty?
119
-
120
124
  capability_name = name.to_s
121
125
  w3c_name = extension_prefix?(capability_name) ? name : "#{APPIUM_PREFIX}#{capability_name}"
122
126
 
@@ -128,61 +132,273 @@ module Appium
128
132
 
129
133
  private
130
134
 
131
- def camel_case(str)
132
- str.gsub(/_([a-z])/) { Regexp.last_match(1).upcase }
135
+ def camel_case(str_or_sym)
136
+ str_or_sym.to_s.gsub(/_([a-z])/) { Regexp.last_match(1).upcase }
133
137
  end
134
138
 
135
139
  def extension_prefix?(capability_name)
136
- snake_cased_capability_names = ::Selenium::WebDriver::Remote::W3C::Capabilities::KNOWN.map(&:to_s)
140
+ snake_cased_capability_names = ::Appium::Core::Base::Capabilities::KNOWN.map(&:to_s)
137
141
  camel_cased_capability_names = snake_cased_capability_names.map { |v| camel_case(v) }
138
142
 
143
+ # Check 'EXTENSION_CAPABILITY_PATTERN'
139
144
  snake_cased_capability_names.include?(capability_name) ||
140
145
  camel_cased_capability_names.include?(capability_name) ||
141
- capability_name.match(::Selenium::WebDriver::Remote::W3C::Capabilities::EXTENSION_CAPABILITY_PATTERN)
146
+ capability_name.match(':')
142
147
  end
143
148
 
144
- def json_create(oss_status, value)
145
- if oss_status
146
- ::Selenium::WebDriver.logger.info 'Detected OSS dialect.'
147
- @dialect = :oss
148
- ::Selenium::WebDriver::Remote::Capabilities.json_create(value)
149
- else
150
- ::Selenium::WebDriver.logger.info 'Detected W3C dialect.'
151
- @dialect = :w3c
152
- ::Selenium::WebDriver::Remote::W3C::Capabilities.json_create(value)
153
- end
149
+ def json_create(value)
150
+ ::Appium::Core::Base::Capabilities.json_create(value)
154
151
  end
155
152
 
156
- def delete_force_mjsonwp(capabilities)
157
- w3c_capabilities = ::Selenium::WebDriver::Remote::W3C::Capabilities.new
153
+ public
158
154
 
159
- capabilities = capabilities.__send__(:capabilities) unless capabilities.is_a?(Hash)
160
- capabilities.each do |name, value|
161
- next if value.nil?
162
- next if value.is_a?(String) && value.empty?
163
- next if name == FORCE_MJSONWP
155
+ # command for Appium 2.0.
156
+ def add_command(method:, url:, name:, &block)
157
+ ::Appium::Logger.info "Overriding the method '#{name}' for '#{url}'" if @available_commands.key? name
158
+
159
+ @available_commands[name] = [method, url]
160
+
161
+ ::Appium::Core::Device.add_endpoint_method name, &block
162
+ end
163
+
164
+ def commands(command)
165
+ @available_commands[command]
166
+ end
167
+
168
+ def status
169
+ execute :status
170
+ end
171
+
172
+ # Perform 'touch' actions for W3C module.
173
+ # Generate +touch+ pointer action here and users can use this via +driver.action+
174
+ # - https://www.selenium.dev/documentation/webdriver/actions_api/
175
+ # - https://www.selenium.dev/selenium/docs/api/rb/Selenium/WebDriver/ActionBuilder.html
176
+ # - https://www.selenium.dev/selenium/docs/api/rb/Selenium/WebDriver/PointerActions.html
177
+ # - https://www.selenium.dev/selenium/docs/api/rb/Selenium/WebDriver/KeyActions.html
178
+ #
179
+ # The pointer type is 'touch' by default in the Appium Ruby client.
180
+ #
181
+ # @example
182
+ #
183
+ # element = @driver.find_element(:id, "some id")
184
+ # @driver.action.click(element).perform # The 'click' is a part of 'PointerActions'
185
+ #
186
+ def action(_deprecated_async = nil, async: false, devices: nil)
187
+ ::Selenium::WebDriver::ActionBuilder.new(
188
+ self,
189
+ devices: devices || [::Selenium::WebDriver::Interactions.pointer(:touch, name: 'touch')],
190
+ async: async,
191
+ duration: 50 # milliseconds
192
+ )
193
+ end
194
+
195
+ # Port from MJSONWP
196
+ def get_timeouts
197
+ execute :get_timeouts
198
+ end
199
+
200
+ # For Appium
201
+ # override
202
+ def element_displayed?(element)
203
+ # For W3C
204
+ # https://github.com/SeleniumHQ/selenium/commit/b618499adcc3a9f667590652c5757c0caa703289
205
+ # execute_atom :isDisplayed, element
206
+ execute :is_element_displayed, id: element.id
207
+ end
208
+
209
+ # For Appium
210
+ # override
211
+ def element_attribute(element, name)
212
+ # For W3C in Selenium Client
213
+ # execute_atom :getAttribute, element, name.
214
+ # 'dom_attribute' in the WebDriver Selenium.
215
+ execute :get_element_attribute, id: element.id, name: name
216
+ end
164
217
 
165
- w3c_capabilities[name] = value
218
+ # For Appium
219
+ # override
220
+ def active_element
221
+ ::Appium::Core::Element.new self, element_id_from(execute(:get_active_element))
222
+ end
223
+ alias switch_to_active_element active_element
224
+
225
+ # For Appium
226
+ # override
227
+ def find_element_by(how, what, parent_ref = [])
228
+ how, what = convert_locator(how, what)
229
+
230
+ return execute_atom(:findElements, Support::RelativeLocator.new(what).as_json).first if how == 'relative'
231
+
232
+ parent_type, parent_id = parent_ref
233
+ id = case parent_type
234
+ when :element
235
+ execute :find_child_element, { id: parent_id }, { using: how, value: what.to_s }
236
+ when :shadow_root
237
+ execute :find_shadow_child_element, { id: parent_id }, { using: how, value: what.to_s }
238
+ else
239
+ execute :find_element, {}, { using: how, value: what.to_s }
240
+ end
241
+
242
+ ::Appium::Core::Element.new self, element_id_from(id)
243
+ end
244
+
245
+ # For Appium
246
+ # override
247
+ def find_elements_by(how, what, parent_ref = [])
248
+ how, what = convert_locator(how, what)
249
+
250
+ return execute_atom :findElements, Support::RelativeLocator.new(what).as_json if how == 'relative'
251
+
252
+ parent_type, parent_id = parent_ref
253
+ ids = case parent_type
254
+ when :element
255
+ execute :find_child_elements, { id: parent_id }, { using: how, value: what.to_s }
256
+ when :shadow_root
257
+ execute :find_shadow_child_elements, { id: parent_id }, { using: how, value: what.to_s }
258
+ else
259
+ execute :find_elements, {}, { using: how, value: what.to_s }
260
+ end
261
+
262
+ ids.map { |id| ::Appium::Core::Element.new self, element_id_from(id) }
263
+ end
264
+
265
+ # For Appium
266
+ # @param [Hash] id The id which can get as a response from server
267
+ # @return [::Appium::Core::Element]
268
+ def convert_to_element(id)
269
+ ::Appium::Core::Element.new self, element_id_from(id)
270
+ end
271
+
272
+ # For Appium
273
+ # override
274
+ # called in 'extend DriverExtensions::HasNetworkConnection'
275
+ def network_connection
276
+ execute :get_network_connection
277
+ end
278
+
279
+ # For Appium
280
+ # override
281
+ # called in 'extend DriverExtensions::HasNetworkConnection'
282
+ def network_connection=(type)
283
+ execute :set_network_connection, {}, { parameters: { type: type } }
284
+ end
285
+
286
+ # For Appium
287
+ # No implementation for W3C webdriver module
288
+ # called in 'extend DriverExtensions::HasLocation'
289
+ def location
290
+ obj = execute(:get_location) || {}
291
+ ::Appium::Location.new obj['latitude'], obj['longitude'], obj['altitude']
292
+ end
293
+
294
+ # For Appium
295
+ # No implementation for W3C webdriver module
296
+ def set_location(lat, lon, alt = 0.0, speed: nil, satellites: nil)
297
+ loc = { latitude: lat, longitude: lon, altitude: alt }
298
+ loc[:speed] = speed unless speed.nil?
299
+ loc[:satellites] = satellites unless satellites.nil?
300
+ execute :set_location, {}, { location: loc }
301
+ end
302
+
303
+ #
304
+ # logs
305
+ #
306
+ # For Appium
307
+ # No implementation for W3C webdriver module
308
+ def available_log_types
309
+ types = execute :get_available_log_types
310
+ Array(types).map(&:to_sym)
311
+ end
312
+
313
+ # For Appium
314
+ # No implementation for W3C webdriver module
315
+ def log(type)
316
+ data = execute :get_log, {}, { type: type.to_s }
317
+
318
+ Array(data).map do |l|
319
+ ::Selenium::WebDriver::LogEntry.new l.fetch('level', 'UNKNOWN'), l.fetch('timestamp'), l.fetch('message')
320
+ rescue KeyError
321
+ next
166
322
  end
323
+ end
167
324
 
168
- w3c_capabilities
325
+ # For Appium
326
+ def log_event(vendor, event)
327
+ execute :post_log_event, {}, { vendor: vendor, event: event }
328
+ end
329
+
330
+ # For Appium
331
+ def log_events(type = nil)
332
+ args = {}
333
+ args['type'] = type unless type.nil?
334
+
335
+ execute :get_log_events, {}, args
336
+ end
337
+
338
+ def viewport_screenshot
339
+ execute_script('mobile: viewportScreenshot')
340
+ end
341
+
342
+ def element_screenshot(element_id)
343
+ execute :take_element_screenshot, id: element_id
344
+ end
345
+
346
+ # for selenium-webdriver compatibility in chrome browser session.
347
+ # This may be needed in selenium-webdriver 4.8 or over? (around the version)
348
+ # when a session starts browserName: 'chrome' for bridge.
349
+ # This method is not only for Android, but also chrome desktop browser as well.
350
+ # So this bridge itself does not restrict the target module.
351
+ def send_command(command_params)
352
+ execute :chrome_send_command, {}, command_params
169
353
  end
170
354
 
171
- def merged_capabilities(desired_capabilities)
172
- force_mjsonwp = desired_capabilities[FORCE_MJSONWP]
173
- desired_capabilities = delete_force_mjsonwp(desired_capabilities) unless force_mjsonwp.nil?
355
+ private
356
+
357
+ def unwrap_script_result(arg)
358
+ case arg
359
+ when Array
360
+ arg.map { |e| unwrap_script_result(e) }
361
+ when Hash
362
+ element_id = element_id_from(arg)
363
+ return ::Appium::Core::Element.new(self, element_id) if element_id
174
364
 
175
- ::Appium::Logger.warn "'forceMjsonwp' no longer works. Sending both W3C and MJSONWP capabilities" if force_mjsonwp
365
+ shadow_root_id = shadow_root_id_from(arg)
366
+ return ::Selenium::WebDriver::Remote::ShadowRoot.new self, shadow_root_id if shadow_root_id
176
367
 
177
- new_caps = add_appium_prefix(desired_capabilities)
178
- w3c_capabilities = ::Selenium::WebDriver::Remote::W3C::Capabilities.from_oss(new_caps)
368
+ arg.each { |k, v| arg[k] = unwrap_script_result(v) }
369
+ else
370
+ arg
371
+ end
372
+ end
373
+
374
+ def element_id_from(id)
375
+ id['ELEMENT'] || id['element-6066-11e4-a52e-4f735466cecf']
376
+ end
179
377
 
180
- {
181
- desiredCapabilities: desired_capabilities,
182
- capabilities: {
183
- firstMatch: [w3c_capabilities]
184
- }
185
- }
378
+ # Don't convert locators for Appium in native context
379
+ def convert_locator(how, what)
380
+ # case how
381
+ # when 'class name'
382
+ # how = 'css selector'
383
+ # what = ".#{escape_css(what)}"
384
+ # when 'id'
385
+ # how = 'css selector'
386
+ # what = "##{escape_css(what)}"
387
+ # when 'name'
388
+ # how = 'css selector'
389
+ # what = "*[name='#{escape_css(what)}']"
390
+ # when 'tag name'
391
+ # how = 'css selector'
392
+ # end
393
+ #
394
+ # if what.is_a?(Hash)
395
+ # what = what.each_with_object({}) do |(h, w), hash|
396
+ # h, w = convert_locator(h.to_s, w)
397
+ # hash[h] = w
398
+ # end
399
+ # end
400
+
401
+ [how, what]
186
402
  end
187
403
  end # class Bridge
188
404
  end # class Base
@@ -15,14 +15,27 @@
15
15
  module Appium
16
16
  module Core
17
17
  class Base
18
- module Capabilities
19
- # @private
20
- # @param [Hash] opts_caps Capabilities for Appium server. All capability keys are converted to lowerCamelCase when
21
- # this client sends capabilities to Appium server as JSON format.
22
- # @return [::Selenium::WebDriver::Remote::W3C::Capabilities] Return instance of Appium::Core::Base::Capabilities
23
- # inherited ::Selenium::WebDriver::Remote::W3C::Capabilities
24
- def self.create_capabilities(opts_caps = {})
25
- ::Selenium::WebDriver::Remote::W3C::Capabilities.new(opts_caps)
18
+ class Capabilities < ::Selenium::WebDriver::Remote::Capabilities
19
+ # TODO: Move to 'Options' way instead of 'Capabilities'.
20
+ # Selenium 5 will have Options instead of 'Capabilities'.
21
+ # https://github.com/SeleniumHQ/selenium/blob/trunk/rb/lib/selenium/webdriver/common/options.rb
22
+ # Then, Ruby client also shoud move to the Options way.
23
+ # Appium's capabilities could change by depending on Appium versions. So it does not have
24
+ # standard options like chrome and firefox etc. So, the implementation should differ from
25
+ # other browsers. But here should inherit `Options` to follow Selenium.
26
+
27
+ # Method override
28
+ # FIXME: when we drop "symbolize_keys", this can be removed.
29
+ def convert_key(key)
30
+ case key
31
+ when String
32
+ key.to_s
33
+ when Symbol
34
+ # here do not convert to camel case
35
+ key.to_s
36
+ else
37
+ raise TypeError, "expected String or Symbol, got #{key.inspect}:#{key.class}"
38
+ end
26
39
  end
27
40
  end
28
41
  end
@@ -14,15 +14,36 @@
14
14
 
15
15
  module Appium
16
16
  module Core
17
- module Commands
18
- module MJSONWP
19
- COMMANDS = ::Appium::Core::Commands::COMMANDS.merge(::Appium::Core::Base::Commands::OSS).merge(
20
- {
21
- # W3C already has.
22
- take_element_screenshot: [:get, 'session/:session_id/element/:id/screenshot']
23
- }
24
- ).freeze
25
- end # module MJSONWP
26
- end # module Commands
27
- end # module Core
28
- end # Appium
17
+ class Base
18
+ #
19
+ # @api private
20
+ #
21
+ class DeviceIME
22
+ # @private this class is private
23
+ def initialize(bridge)
24
+ @bridge = bridge
25
+ end
26
+
27
+ def activate(ime_name)
28
+ @bridge.ime_activate(ime_name)
29
+ end
30
+
31
+ def available_engines
32
+ @bridge.ime_available_engines
33
+ end
34
+
35
+ def active_engine
36
+ @bridge.ime_active_engine
37
+ end
38
+
39
+ def activated?
40
+ @bridge.ime_activated
41
+ end
42
+
43
+ def deactivate
44
+ @bridge.ime_deactivate
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end