# frozen_string_literal: true # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. require 'base64' require_relative 'device_ime' require_relative 'driver_settings' require_relative 'search_context' require_relative 'screenshot' require_relative 'rotable' require_relative 'remote_status' require_relative 'has_location' require_relative 'has_network_connection' require_relative '../wait' module Appium module Core class Base class Driver < ::Selenium::WebDriver::Driver include ::Selenium::WebDriver::DriverExtensions::UploadsFiles include ::Selenium::WebDriver::DriverExtensions::HasSessionId include ::Selenium::WebDriver::DriverExtensions::HasWebStorage include ::Appium::Core::Base::Rotatable include ::Appium::Core::Base::SearchContext include ::Appium::Core::Base::TakesScreenshot include ::Appium::Core::Base::HasRemoteStatus include ::Appium::Core::Base::HasLocation include ::Appium::Core::Base::HasNetworkConnection include ::Appium::Core::Waitable # Private API. # Do not use this for general use. Used by flutter driver to get bridge for creating a new element attr_reader :bridge private def initialize(bridge: nil, listener: nil, **opts) # For ::Appium::Core::Waitable @wait_timeout = opts.delete(:wait_timeout) @wait_interval = opts.delete(:wait_interval) # For logging. # TODO: Remove when appium core no longer uses this in this bridge. @automation_name = opts.delete(:automation_name) super end # Create a proper bridge instance. # # @return [::Appium::Core::Base::Bridge] # def create_bridge(**opts) capabilities = opts.delete(:capabilities) bridge_opts = { http_client: opts.delete(:http_client), url: opts.delete(:url) } raise ::Appium::Core::Error::ArgumentError, "Unable to create a driver with parameters: #{opts}" unless opts.empty? bridge = ::Appium::Core::Base::Bridge.new(**bridge_opts) bridge.create_session(capabilities) bridge end public # Update +server_url+ and HTTP clients following this arguments, protocol, host, port and path. # After this method, +@bridge.http+ will be a new instance following them instead of +server_url+ which is # set before creating session. # If +@bridge.http+ did not have +update_sending_request_to+ method, this method returns immediately. # # @example # # driver = core.start_driver server_url: 'http://example1.com:8000/wd/hub # @bridge.http is for 'http://example1.com:8000/wd/hub/' # driver.update_sending_request_to protocol: 'https', host: 'example2.com', port: 9000, path: '/wd/hub' # driver.manage.timeouts.implicit_wait = 10 # @bridge.http is for 'https://example2.com:9000/wd/hub/' # def update_sending_request_to(protocol:, host:, port:, path:) unless @bridge.http&.class&.method_defined? :update_sending_request_to ::Appium::Logger.warn "#{@bridge.http&.class} has no 'update_sending_request_to'. " \ 'It keeps current connection target.' return end @bridge.http&.update_sending_request_to(scheme: protocol, host: host, port: port, path: path) end AVAILABLE_METHODS = [ :get, :head, :post, :put, :delete, :connect, :options, :trace, :patch ].freeze # Define a new custom method to the driver so that you can define your own method for # drivers/plugins in Appium 2.0. Appium 2.0 and its custom drivers/plugins allow you # to define custom commands that are not part of W3C spec. # # @param [Symbol] method HTTP request method as https://www.w3.org/TR/webdriver/#endpoints # @param [string] url The url to URL template as https://www.w3.org/TR/webdriver/#endpoints. # +:session_id+ is the placeholder of 'session id'. # Other place holders can be specified with +:+ prefix like +:id+. # Then, the +:id+ will be replaced with a given value as the seconds argument of +execute+ # @param [Symbol] name The name of method that is called as the driver instance method. # @param [Proc] block The block to involve as the method. # Please define a method that has the same +name+ with arguments you want. # The method must has +execute+ method. tHe +execute+ is calls the +url+ # with the given parameters. # The first argument should be +name+ as symbol. # The second argument should be hash. If keys in the hash matches +:+ prefix # string in the given url, the matched string in the given url will be # values in the hash. # The third argument should be hash. The hash will be the request body. # Please read examples below for more details. # @raise [ArgumentError] If the given +method+ is invalid value. # # @example # # @driver.add_command( # method: :get, # url: 'session/:session_id/path/to/custom/url', # name: :test_command # ) # # Send a GET request to 'session//path/to/custom/url' # @driver.test_command # # # @driver.add_command( # method: :post, # url: 'session/:session_id/path/to/custom/url', # name: :test_command # ) do # def test_command(argument) # execute(:test_command, {}, { dummy: argument }) # end # end # # Send a POST request to 'session//path/to/custom/url' # # with body "{ dummy: 1 }" as JSON object. "1" is the argument. # # ':session_id' in the given 'url' is replaced with current 'session id'. # @driver.test_command(1) # # # @driver.add_command( # method: :post, # url: 'session/:session_id/element/:id/custom/action', # name: :test_action_command # ) do # def test_action_command(element_id, action) # execute(:test_action_command, {id: element_id}, { dummy_action: action }) # end # end # # Send a POST request to 'session//element//custom/action' # # with body "{ dummy_action: #{action} }" as JSON object. "action" is the seconds argument. # # ':session_id' in the given url is replaced with current 'session id'. # # ':id' in the given url is replaced with the given 'element_id'. # e = @driver.find_element :accessibility_id, 'an element' # @driver.test_action_command(e.id, 'action') # def add_command(method:, url:, name:, &block) unless AVAILABLE_METHODS.include? method raise ::Appium::Core::Error::ArgumentError, "Available method is either #{AVAILABLE_METHODS}" end # TODO: Remove this logger before Appium 2.0 release ::Appium::Logger.info '[Experimental] this method is experimental for Appium 2.0. This interface may change.' @bridge.add_command method: method, url: url, name: name, &block end ### Methods for Appium # Perform 'key' actions for W3C module. # Generate +key+ pointer action here and users can use this via +driver.key_action+ # - https://seleniumhq.github.io/selenium/docs/api/rb/Selenium/WebDriver/W3CActionBuilder.html # - https://seleniumhq.github.io/selenium/docs/api/rb/Selenium/WebDriver/KeyActions.html # # The pointer type is 'key' by default in the Appium Ruby client. # +driver.action+ in Appium Ruby client has 'pointer' action by default. # This method is a shortcut to set 'key' type. # Hense this method is equal to +driver.action(devices: [::Selenium::WebDriver::Interactions.key('keyboard')])+ # as below example. # # @example # # element = @driver.find_element(:id, "some id") # @driver.key_action.send_key('hiあ').perform # The 'send_key' is a part of 'KeyActions' # # is equal to: # # @driver.action(devices: [::Selenium::WebDriver::Interactions.key('keyboard')]).send_keys('hiあ').perform # def key_action(async: false) @bridge.action( async: async, devices: [::Selenium::WebDriver::Interactions.key('keyboard')] ) end # Lock the device # @return [String] # # @example # # @driver.lock #=> Lock the device # @driver.lock(5) #=> Lock the device in 5 sec and unlock the device after 5 sec. # # Block other commands during locking the device. # def lock(duration = nil) @bridge.lock(duration) end # Check current device status is weather locked or not # # @example # # @driver.device_locked? # @driver.locked? # def locked? @bridge.device_locked? end alias device_locked? locked? # Unlock the device # # @example # # @driver.unlock # def unlock @bridge.unlock end # Hide the onscreen keyboard # @param [String] close_key The name of the key which closes the keyboard. # Defaults to 'Done' for iOS(except for XCUITest). # @param [Symbol] strategy The symbol of the strategy which closes the keyboard. # XCUITest ignore this argument. # Default for iOS is +:pressKey+. Default for Android is +:tapOutside+. # # @example # # @driver.hide_keyboard # Close a keyboard with the 'Done' button # @driver.hide_keyboard('Finished') # Close a keyboard with the 'Finished' button # @driver.hide_keyboard(nil, :tapOutside) # Close a keyboard with tapping out side of keyboard # def hide_keyboard(close_key = nil, strategy = nil) @bridge.hide_keyboard close_key, strategy end # Get whether keyboard is displayed or not. # @return [Boolean] Return true if keyboard is shown. Return false if keyboard is hidden. # # @example # @driver.is_keyboard_shown # false # @driver.keyboard_shown? # true # def keyboard_shown? @bridge.is_keyboard_shown end alias is_keyboard_shown keyboard_shown? # Returns an instance of DriverSettings to call get/update. # # @example # # @driver.settings.get # @driver.settings.update('allowInvisibleElements': true) # def settings @settings ||= DriverSettings.new(@bridge) end # Get appium Settings for current test session. # Alias of @driver.settings.get # # @example # # @driver.get_settings # @driver.settings.get # def get_settings settings.get end # Update Appium Settings for current test session # Alias of @driver.settings#update # # @param [Hash] value Settings to update, keys are settings, values to value to set each setting to # # @example # # @driver.update_settings({ 'allowInvisibleElements': true }) # @driver.settings.update({ 'allowInvisibleElements': true }) # @driver.settings = { 'allowInvisibleElements': true } # def settings=(value) settings.update(value) end alias update_settings settings= # Returns an instance of DeviceIME # # @return [Appium::Core::Base::Driver::DeviceIME] # # @example # # @driver.ime.activate engine: 'com.android.inputmethod.latin/.LatinIME' # @driver.ime.available_engines #=> Get the list of IME installed in the target device # @driver.ime.active_engine #=> Get the current active IME such as 'com.android.inputmethod.latin/.LatinIME' # @driver.ime.activated #=> True if IME is activated # @driver.ime.deactivate #=> Deactivate current IME engine # def ime @ime ||= DeviceIME.new(@bridge) end # Android only. Make an engine that is available active. # # @param [String] ime_name The IME owning the activity [required] # # @example # # @driver.ime_activate engine: 'com.android.inputmethod.latin/.LatinIME' # @driver.ime.activate engine: 'com.android.inputmethod.latin/.LatinIME' # def ime_activate(ime_name) ime.activate(ime_name) end # Android only. List all available input engines on the machine. # # @example # # @driver.ime_available_engines #=> Get the list of IME installed in the target device # @driver.ime.available_engines #=> Get the list of IME installed in the target device # def ime_available_engines ime.available_engines end # Android only. Get the name of the active IME engine. # # @example # # @driver.ime_active_engine #=> Get the current active IME such as 'com.android.inputmethod.latin/.LatinIME' # @driver.ime.active_engine #=> Get the current active IME such as 'com.android.inputmethod.latin/.LatinIME' # def ime_active_engine ime.active_engine end # @!method ime_activated # Android only. Indicates whether IME input is active at the moment (not if it is available). # # @return [Boolean] # # @example # # @driver.ime_activated #=> True if IME is activated # @driver.ime.activated #=> True if IME is activated # def ime_activated ime.activated? end # Android only. De-activates the currently-active IME engine. # # @example # # @driver.ime_deactivate #=> Deactivate current IME engine # @driver.ime.deactivate #=> Deactivate current IME engine # def ime_deactivate ime.deactivate end # Perform a block within the given context, then switch back to the starting context. # @param [String] context The context to switch to for the duration of the block. # @param [Proc] block The block to involve within the context # # @example # # result = @driver.within_context('NATIVE_APP') do # @driver.find_element :tag, "button" # end # The result of 'find_element :tag, "button"' # def within_context(context, &block) block_given? ? @bridge.within_context(context, &block) : @bridge.within_context(context) end # Change to the default context. This is equivalent to +set_context nil+. # # @example # # @driver.switch_to_default_context # def switch_to_default_context @bridge.switch_to_default_context end # @return [String] The context currently being used. # # @example # # @driver.current_context # def current_context @bridge.current_context end # @return [Array] All usable contexts, as an array of strings. # # @example # # @driver.available_contexts # def available_contexts @bridge.available_contexts end # Change the context to the given context. # @param [String] context The context to change to # # @example # # @driver.set_context "NATIVE_APP" # @driver.context = "NATIVE_APP" # def context=(context = null) @bridge.set_context(context) end alias set_context context= # Place a file in a specific location on the device. # On Android, the application under test should be built with debuggable flag enabled in order to get access to # its container on the internal file system. # # @param [String] path Either an absolute path OR, for iOS devices, a path relative to the app, as described. # If the path starts with application id prefix, then the file will be pushed to the root of # the corresponding application container. # @param [String] filedata Raw file data to be sent to the device. Converted to base64 in the method. # # @example # # @driver.push_file "/file/to/path", "data" # # file = File.read "your/path/to/test_image.png" # @driver.push_file "/sdcard/Pictures", file # Push a file binary to /sdcard/Pictures path in Android # def push_file(path, filedata) @bridge.push_file(path, filedata) end # Pull a file from the remote device. # On Android the application under test should be built with debuggable flag enabled in order to get access # to its container on the internal file system. # # @param [String] path Either an absolute path OR, for iOS devices, a path relative to the app, as described. # If the path starts with application id prefix, then the file will be pulled from the root # of the corresponding application container. # Otherwise the root folder is considered as / on Android and on iOS it is a media folder root # (real devices only). # Only pulling files from application containers is supported for iOS Simulator. # Provide the remote path in format # @bundle.identifier:container_type/relative_path_in_container # # @return [Base64-decoded] Base64 decoded data # # @example # # decoded_file = @driver.pull_file '/local/data/some/path' #=> Get the file at that path # decoded_file = @driver.pull_file 'Shenanigans.app/some/file' # #=> Get 'some/file' from the install location of Shenanigans.app # decoded_file = @driver.pull_file '@com.appium.example/Documents/file.txt' # #=> Get 'file.txt' in @com.appium.example/Documents # File.open('proper_filename', 'wb') { |f| f<< decoded_file } # def pull_file(path) @bridge.pull_file(path) end # Pull a folder content from the remote device. # On Android the application under test should be built with debuggable flag enabled in order to get access to # its container on the internal file system. # # @param [String] path Absolute path to the folder. # If the path starts with @applicationId/ prefix, then the folder will be pulled # from the root of the corresponding application container. # Otherwise the root folder is considered as / on Android and on iOS it is a media folder root # (real devices only). # Only pulling files from application containers is supported for iOS Simulator. # Provide the remote path in format # @bundle.identifier:container_type/relative_path_in_container # # @return [Base64-decoded] Base64 decoded data which is zip archived # # @example # # decoded_file = @driver.pull_folder '/data/local/tmp' #=> Get the folder at that path # decoded_file = @driver.pull_file '@com.appium.example/Documents' #=> Get 'Documents' in @com.appium.example # File.open('proper_filename', 'wb') { |f| f<< decoded_file } # def pull_folder(path) @bridge.pull_folder(path) end # Press keycode on the device. # http://developer.android.com/reference/android/view/KeyEvent.html # @param [Integer] key The key to press. The values which have +KEYCODE_+ prefix in http://developer.android.com/reference/android/view/KeyEvent.html # e.g.: KEYCODE_HOME is +3+ or +0x00000003+ # @param [[Integer]] metastate The state the metakeys should be in when pressing the key. Default is empty Array. # Metastate have +META_+ prefix in https://developer.android.com/reference/android/view/KeyEvent.html # e.g.: META_SHIFT_ON is +1+ or +0x00000001+ # @param [[Integer]] flags Native Android flag value. Several flags can be combined into a single key event. # Default is empty Array. Can set multiple flags as Array. # Flags have +FLAG_+ prefix in http://developer.android.com/reference/android/view/KeyEvent.html # e.g.: FLAG_CANCELED is +32+ or +0x00000020+ # # @example # # @driver.press_keycode 66 # @driver.press_keycode 66, flags: [0x02] # @driver.press_keycode 66, metastate: [1], flags: [32] # def press_keycode(key, metastate: [], flags: []) @bridge.press_keycode(key, metastate: metastate, flags: flags) end # Long press keycode on the device. # http://developer.android.com/reference/android/view/KeyEvent.html # @param [Integer] key The key to long press. The values which have +KEYCODE_+ prefix in http://developer.android.com/reference/android/view/KeyEvent.html # e.g.: KEYCODE_HOME is +3+ or +0x00000003+ # @param [[Integer]] metastate The state the metakeys should be in when pressing the key. Default is empty Array. # Metastate have +META_+ prefix in https://developer.android.com/reference/android/view/KeyEvent.html # e.g.: META_SHIFT_ON is +1+ or +0x00000001+ # @param [[Integer]] flags Native Android flag value. Several flags can be combined into a single key event. # Default is empty Array. Can set multiple flags as Array. # Flags have +FLAG_+ prefix in http://developer.android.com/reference/android/view/KeyEvent.html # e.g.: FLAG_CANCELED is +32+ or +0x00000020+ # # @example # # @driver.long_press_keycode 66 # @driver.long_press_keycode 66, flags: [0x20, 0x2000] # @driver.long_press_keycode 66, metastate: [1], flags: [32, 8192] # def long_press_keycode(key, metastate: [], flags: []) @bridge.long_press_keycode(key, metastate: metastate, flags: flags) end # @deprecated Except for Windows # Start the simulator and application configured with desired capabilities # # @example # # @driver.launch_app # def launch_app # TODO: Define only in Windows module when ruby_lib_core removes this method if @automation_name != :windows ::Appium::Logger.warn( '[DEPRECATION] launch_app is deprecated. Please use activate_app instead.' ) end @bridge.launch_app end # @deprecated Except for Windows # Close an app on device # # @example # # @driver.close_app # def close_app # TODO: Define only in Windows module when ruby_lib_core removes this method if @automation_name != :windows ::Appium::Logger.warn( '[DEPRECATION] close_app is deprecated. Please use terminate_app instead.' ) end @bridge.close_app end # @deprecated # Reset the device, relaunching the application. # # @example # # @driver.reset # def reset ::Appium::Logger.warn( '[DEPRECATION] reset is deprecated. Please use terminate_app and activate_app, ' \ 'or quit and create a new session instead.' ) @bridge.reset end # Return the hash of all localization strings. # @return [Hash] # # @example # # @driver.app_strings #=> "TransitionsTitle"=>"Transitions", "WebTitle"=>"Web" # def app_strings(language = nil) @bridge.app_strings(language) end # Backgrounds the app for a set number of seconds. # This is a blocking application # @param [Integer] duration How many seconds to background the app for. # @return [String] # # @example # # @driver.background_app # @driver.background_app(5) # @driver.background_app(-1) #=> the app never come back. https://github.com/appium/appium/issues/7741 # def background_app(duration = 0) @bridge.background_app(duration) end # Install the given app onto the device. # Each options can be snake-case or camel-case. Snake-cases will be converted to camel-case # as options value. # # @param [String] path The absolute local path or remote http URL to an .ipa or .apk file, # or a .zip containing one of these. # @param [Boolean] replace Only for Android. Whether to reinstall/upgrade the package if it is already present # on the device under test. +true+ by default # @param [Integer] timeout Only for Android. How much time to wait for the installation to complete. # 60000ms by default. # @param [Boolean] allow_test_packages Only for Android. Whether to allow installation of packages marked as test # in the manifest. +false+ by default # @param [Boolean] use_sdcard Only for Android. Whether to use the SD card to install the app. +false+ by default # @param [Boolean] grant_permissions Only for Android. whether to automatically grant application permissions # on Android 6+ after the installation completes. +false+ by default # # Other parameters such as https://github.com/appium/appium-xcuitest-driver#mobile-installapp also can be set. # Then, arguments in snake case will be camel case as its request parameters. # # @example # # @driver.install_app("/path/to/test.apk") # @driver.install_app("/path/to/test.apk", replace: true, timeout: 20000, allow_test_packages: true, # use_sdcard: false, grant_permissions: false) # @driver.install_app("/path/to/test.ipa", timeoutMs: 20000) # def install_app(path, **options) options = options.transform_keys { |key| key.to_s.gsub(/_./) { |v| v[1].upcase } } unless options.nil? @bridge.install_app(path, options) end # def capitalize(s) # chars = # chars[1:].map(&:capitalize).join # end # @param [Strong] app_id BundleId for iOS or package name for Android # @param [Boolean] keep_data Only for Android. Whether to keep application data and caches after it is uninstalled. # +false+ by default # @param [Integer] timeout Only for Android. How much time to wait for the uninstall to complete. 20000ms by default. # # @example # # @driver.remove_app("io.appium.bundle") # @driver.remove_app("io.appium.bundle", keep_data: false, timeout, 10000) # def remove_app(app_id, keep_data: nil, timeout: nil) @bridge.remove_app(app_id, keep_data: keep_data, timeout: timeout) end # Check whether the specified app is installed on the device # @return [Boolean] # # @example # # @driver.app_installed?("io.appium.bundle") # def app_installed?(app_id) @bridge.app_installed?(app_id) end # Activate(Launch) the specified app. # @return [Hash] # # @example # # @driver.activate_app("io.appium.bundle") #=> {} # def activate_app(app_id) @bridge.activate_app(app_id) end # Terminate the specified app. # # @param [Strong] app_id BundleId for iOS or package name for Android # @param [Integer] timeout Only for Android. How much time to wait for the application termination to complete. # 500ms by default. # @return [Boolean] # # @example # # @driver.terminate_app("io.appium.bundle") # true # @driver.terminate_app("io.appium.bundle", timeout: 500) # def terminate_app(app_id, timeout: nil) @bridge.terminate_app(app_id, timeout: timeout) end # Get the status of an existing application on the device. # State: # :not_installed : The current application state cannot be determined/is unknown # :not_running : The application is not running # :running_in_background_suspended : The application is running in the background and is suspended # :running_in_background : The application is running in the background and is not suspended # :running_in_foreground : The application is running in the foreground # # For more details: https://developer.apple.com/documentation/xctest/xcuiapplicationstate # # @param [String] app_id A target app's bundle id # @return [AppState::STATUS] A number of the state # # @example # # @driver.app_state("io.appium.bundle") #=> :not_running # # Compatibility for other clients # @driver.query_app_state("io.appium.bundle") #=> :not_running # def app_state(app_id) @bridge.app_state(app_id) end alias query_app_state app_state # @param [String] remote_path The path to the remote location, where the resulting video should be uploaded. # The following protocols are supported: http/https, ftp. # Null or empty string value (the default setting) means the content of resulting # file should be encoded as Base64 and passed as the endpoint response value. # An exception will be thrown if the generated media file is too big to # fit into the available process memory. # @param [String] user The name of the user for the remote authentication. # @param [String] pass The password for the remote authentication. # @param [String] method The http multipart upload method name. The 'PUT' one is used by default. # # @example # # @driver.stop_recording_screen # @driver.stop_recording_screen remote_path: 'https://example.com', user: 'example', pass: 'pass', method: 'POST' # def stop_recording_screen(remote_path: nil, user: nil, pass: nil, method: 'PUT') @bridge.stop_recording_screen(remote_path: remote_path, user: user, pass: pass, method: method) end # @param [String] file_path The path to save video decoded from base64 from Appium server. # # @example # # # iOS # @driver.start_recording_screen video_type: 'libx264' # @driver.stop_and_save_recording_screen 'example.mp4' # Video type 'libx264' can be play as '.mp4' video # # # Android # @driver.start_recording_screen # @driver.stop_and_save_recording_screen 'example.mp4' # def stop_and_save_recording_screen(file_path) @bridge.stop_and_save_recording_screen(file_path) end # Cause the device to shake # # @example # # @driver.shake # def shake @bridge.shake end # Get the time on the device # # @param [String] format The set of format specifiers. Read https://momentjs.com/docs/ to get # the full list of supported datetime format specifiers. # The default format is +YYYY-MM-DDTHH:mm:ssZ+, which complies to ISO-8601 # @return [String] Formatted datetime string or the raw command output if formatting fails # # @example # # @driver.device_time #=> "2018-06-12T11:13:31+02:00" # @driver.device_time "YYYY-MM-DD" #=> "2018-06-12" # def device_time(format = nil) @bridge.device_time(format) end # touch actions def touch_actions(actions) @bridge.touch_actions(actions) end def multi_touch(actions) @bridge.multi_touch(actions) end # # Send multiple W3C action chains to server. Use +@driver.action+ for single action chain. # # @param [Array] data Array of actions # @return nil|error # # @example: Zoom # # f1 = ::Selenium::WebDriver::Interactions.pointer(:touch, name: 'finger1') # f1.create_pointer_move(duration: 1, x: 200, y: 500, # origin: ::Selenium::WebDriver::Interactions::PointerMove::VIEWPORT) # f1.create_pointer_down(:left) # f1.create_pointer_move(duration: 1, x: 200, y: 200, # origin: ::Selenium::WebDriver::Interactions::PointerMove::VIEWPORT) # f1.create_pointer_up(:left) # # f2 = ::Selenium::WebDriver::Interactions.pointer(:touch, name: 'finger2') # f2.create_pointer_move(duration: 1, x: 200, y: 500, # origin: ::Selenium::WebDriver::Interactions::PointerMove::VIEWPORT) # f2.create_pointer_down(:left) # f2.create_pointer_move(duration: 1, x: 200, y: 800, # origin: ::Selenium::WebDriver::Interactions::PointerMove::VIEWPORT) # f2.create_pointer_up(:left) # # @driver.perform_actions [f1, f2] #=> 'nil' if the action succeed # def perform_actions(data) raise ::Appium::Core::Error::ArgumentError, "'#{data}' must be Array" unless data.is_a? Array # NOTE: 'add_input' in Selenium Ruby implementation has additional 'pause'. # This implementation is to avoid the additional pause. # https://github.com/SeleniumHQ/selenium/blob/64447d4b03f6986337d1ca8d8b6476653570bcc1/rb/lib/selenium/webdriver/common/action_builder.rb#L207 @bridge.send_actions data.map(&:encode).compact data.each(&:clear_actions) nil end # Get the device window's size. # @return [Selenium::WebDriver::Dimension] # # @example # size = @driver.window_size # size.width #=> Integer # size.height #=> Integer # def window_size manage.window.size end # Get the device window's rect. # @return [Selenium::WebDriver::Rectangle] # # @example # size = @driver.window_rect # size.width #=> Integer # size.height #=> Integer # size.x #=> 0 # size.y #=> 0 # def window_rect manage.window.rect end # Get the device window's size. # @return [String] # # @example # @driver.back # back to the previous view # def back navigate.back end # Get the device window's logs. # @return [Appium::Core::Logs] # # @example # # @driver.logs.available_types # [:syslog, :crashlog, :performance] # @driver.logs.get :syslog # [] # def logs @logs ||= Logs.new(@bridge) end # For W3C. # Get the timeout related settings on the server side. # # @return [Hash] # # @example # @driver.get_timeouts # def get_timeouts @bridge.get_timeouts end # Retrieve the capabilities of the specified session. # It's almost same as +@driver.capabilities+ but you can get more details. # # @return [Selenium::WebDriver::Remote::Capabilities, Selenium::WebDriver::Remote::Capabilities] # # @example # @driver.session_capabilities # # #=> uiautomator2 # # nil, # # :browser_version=>nil, # # :platform_name=>"android", # # :page_load_strategy=>nil, # # :remote_session_id=>nil, # # :accessibility_checks=>nil, # # :profile=>nil, # # :rotatable=>nil, # # :device=>nil, # # "platform"=>"LINUX", # # "webStorageEnabled"=>false, # # "takesScreenshot"=>true, # # "javascriptEnabled"=>true, # # "databaseEnabled"=>false, # # "networkConnectionEnabled"=>true, # # "locationContextEnabled"=>false, # # "warnings"=>{}, # # "desired"=> # # {"platformName"=>"android", # # "automationName"=>"uiautomator2", # # "app"=>"/path/to/app/api.apk.zip", # # "platformVersion"=>"8.1.0", # # "deviceName"=>"Android Emulator", # # "appPackage"=>"io.appium.android.apis", # # "appActivity"=>"io.appium.android.apis.ApiDemos", # # "someCapability"=>"some_capability", # # "unicodeKeyboard"=>true, # # "resetKeyboard"=>true}, # # "automationName"=>"uiautomator2", # # "app"=>"/path/to/app/api.apk.zip", # # "platformVersion"=>"8.1.0", # # "deviceName"=>"emulator-5554", # # "appPackage"=>"io.appium.android.apis", # # "appActivity"=>"io.appium.android.apis.ApiDemos", # # "someCapability"=>"some_capability", # # "unicodeKeyboard"=>true, # # "resetKeyboard"=>true, # # "deviceUDID"=>"emulator-5554", # # "deviceScreenSize"=>"1080x1920", # # "deviceScreenDensity"=>420, # # "deviceModel"=>"Android SDK built for x86", # # "deviceManufacturer"=>"Google", # # "pixelRatio"=>2.625, # # "statBarHeight"=>63, # # "viewportRect"=>{"left"=>0, "top"=>63, "width"=>1080, "height"=>1731}}> # # # #=> XCUITest # # "UICatalog", # # :browser_version=>nil, # # :platform_name=>"ios", # # :page_load_strategy=>nil, # # :remote_session_id=>nil, # # :accessibility_checks=>nil, # # :profile=>nil, # # :rotatable=>nil, # # :device=>"iphone", # # "udid"=>"DED4DBAD-8E5E-4AD6-BDC4-E75CF9AD84D8", # # "automationName"=>"XCUITest", # # "app"=>"/path/to/app/UICatalog.app", # # "platformVersion"=>"11.4", # # "deviceName"=>"iPhone Simulator", # # "useNewWDA"=>true, # # "useJSONSource"=>true, # # "someCapability"=>"some_capability", # # "sdkVersion"=>"11.4", # # "CFBundleIdentifier"=>"com.example.apple-samplecode.UICatalog", # # "pixelRatio"=>2, # # "statBarHeight"=>23.4375, # # "viewportRect"=>{"left"=>0, "top"=>47, "width"=>750, "height"=>1287}}> # def session_capabilities @bridge.session_capabilities end # Returns available sessions on the Appium server # # @return [[Hash]] # # @example # # @driver.sessions #=> [{'id' => 'c363add8-a7ca-4455-b9e3-9ac4d69e95b3', 'capabilities' => { capabilities as Hash }}] # def sessions @bridge.sessions end # Image Comparison def match_images_features(first_image:, second_image:, detector_name: 'ORB', match_func: 'BruteForce', good_matches_factor: nil, visualize: false) @bridge.match_images_features(first_image: first_image, second_image: second_image, detector_name: detector_name, match_func: match_func, good_matches_factor: good_matches_factor, visualize: visualize) end def find_image_occurrence(full_image:, partial_image:, visualize: false, threshold: nil, multiple: nil, match_neighbour_threshold: nil) @bridge.find_image_occurrence(full_image: full_image, partial_image: partial_image, visualize: visualize, threshold: threshold, multiple: multiple, match_neighbour_threshold: match_neighbour_threshold) end def get_images_similarity(first_image:, second_image:, visualize: false) @bridge.get_images_similarity(first_image: first_image, second_image: second_image, visualize: visualize) end def compare_images(mode: :matchFeatures, first_image:, second_image:, options: nil) @bridge.compare_images(mode: mode, first_image: first_image, second_image: second_image, options: options) end # @since Appium 1.8.2 # Return an element if current view has a partial image. The logic depends on template matching by OpenCV. # {https://github.com/appium/appium/blob/1.x/docs/en/writing-running-appium/image-comparison.md image-comparison} # # You can handle settings for the comparision following below. # {https://github.com/appium/appium-base-driver/blob/master/lib/basedriver/device-settings.js#L6 device-settings} # # @param [String] img_path A path to a partial image you'd like to find # # @return [::Appium::Core::Element] # # @example # # @@driver.update_settings({ fixImageFindScreenshotDims: false, fixImageTemplateSize: true, # autoUpdateImageElementPosition: true }) # e = @@driver.find_element_by_image './test/functional/data/test_element_image.png' # def find_element_by_image(img_path) template = Base64.strict_encode64 File.read img_path find_element :image, template end # @since Appium 1.8.2 # Return elements if current view has a partial image. The logic depends on template matching by OpenCV. # {https://github.com/appium/appium/blob/1.x/docs/en/writing-running-appium/image-comparison.md image-comparison} # # You can handle settings for the comparision following below. # {https://github.com/appium/appium-base-driver/blob/master/lib/basedriver/device-settings.js#L6 device-settings} # # @param [String] img_path A path to a partial image you'd like to find # # @return [Array] # # @example # # @@driver.update_settings({ fixImageFindScreenshotDims: false, fixImageTemplateSize: true, # autoUpdateImageElementPosition: true }) # e = @@driver.find_elements_by_image ['./test/functional/data/test_element_image.png'] # e == [] # if the 'e' is empty # def find_elements_by_image(img_path) template = Base64.strict_encode64 File.read img_path find_elements :image, template end # @since Appium 1.14.0 # # Run a set of script against the current session, allowing execution of many commands in one Appium request. # Supports {https://webdriver.io/docs/api.html WebdriverIO} API so far. # Please read {http://appium.io/docs/en/commands/session/execute-driver command API} for more details # about acceptable scripts and the output. # # @param [String] script The string consisting of the script itself # @param [String] type The name of the script type. # Defaults to 'webdriverio'. Depends on server implementation which type is supported. # @param [Integer] timeout_ms The number of +ms+ Appium should wait for the script to finish # before killing it due to timeout. # # @return [Appium::Core::Base::Device::ExecuteDriver::Result] The script result parsed by # Appium::Core::Base::Device::ExecuteDriver::Result. # # @raise [::Selenium::WebDriver::Error::UnknownError] If something error happens in the script. # It has the original message. # # @example # script = <<~SCRIPT # const status = await driver.status(); # console.warn('warning message'); # return [status]; # SCRIPT # r = @@driver.execute_driver(script: script, type: 'webdriverio', timeout: 10_000) # r #=> An instance of Appium::Core::Base::Device::ExecuteDriver::Result # r.result #=> The 'result' key part as the result of the script # r.logs #=> The 'logs' key part as '{'log' => [], 'warn' => [], 'error' => []}' # def execute_driver(script: '', type: 'webdriverio', timeout_ms: nil) @bridge.execute_driver(script: script, type: type, timeout_ms: timeout_ms) end # Convert vanilla element response to ::Appium::Core::Element # # @param [Hash] id The id which can get as a response from server # @return [::Appium::Core::Element] # # @example # response = {"element-6066-11e4-a52e-4f735466cecf"=>"xxxx", "ELEMENT"=>"xxxx"} # ele = @driver.convert_to_element(response) #=> ::Appium::Core::Element # ele.rect #=> Can get the rect of the element # def convert_to_element(id) @bridge.convert_to_element id end end # class Driver end # class Base end # module Core end # module Appium