4976 lines
126 KiB
Ruby
4976 lines
126 KiB
Ruby
# Copyright (C) 2012-2021 Zammad Foundation, http://zammad-foundation.org/
|
|
|
|
ENV['RAILS_ENV'] = 'test'
|
|
# rubocop:disable Lint/NonLocalExitFromIterator, Style/GuardClause, Lint/MissingCopEnableDirective
|
|
require File.expand_path('../config/environment', __dir__)
|
|
require 'selenium-webdriver'
|
|
require 'json'
|
|
require 'net/http'
|
|
require 'uri'
|
|
|
|
# This is a workaround for running the browser test suite
|
|
# in an alphabetical order
|
|
# because `test/browser/aaa_*` tests are required to run first
|
|
require 'minitest'
|
|
module Minitest
|
|
def self.__run(reporter, options)
|
|
Runnable.runnables
|
|
.reject { |s| s.runnable_methods.empty? }
|
|
.map { |suite| suite.run reporter, options }
|
|
end
|
|
end
|
|
|
|
class TestCase < ActiveSupport::TestCase
|
|
|
|
DEBUG = true
|
|
|
|
setup do
|
|
# print current test case to STDOUT
|
|
# for status reasoning and debugging purposes
|
|
source_location = self.class.instance_method(method_name).source_location
|
|
test_file_path = source_location[0].remove("#{Rails.root}/") # rubocop:disable Rails/FilePath
|
|
test_method_line = source_location[1]
|
|
puts <<~HTML
|
|
|
|
Performing test #{self.class.name}##{method_name} (#{test_file_path}:#{test_method_line}):
|
|
|
|
HTML
|
|
end
|
|
|
|
def browser
|
|
ENV['BROWSER'] || 'firefox'
|
|
end
|
|
|
|
def profile
|
|
browser_profile = nil
|
|
case browser
|
|
when 'firefox'
|
|
browser_profile = Selenium::WebDriver::Firefox::Profile.new
|
|
|
|
browser_profile['intl.locale.matchOS'] = false
|
|
browser_profile['intl.accept_languages'] = 'en-US'
|
|
browser_profile['general.useragent.locale'] = 'en-US'
|
|
# currently console log not working for firefox
|
|
# https://github.com/SeleniumHQ/selenium/issues/1161
|
|
# browser_profile['loggingPref'] = { browser: :all }
|
|
when 'chrome'
|
|
|
|
# profile are only working on remote selenium
|
|
if ENV['REMOTE_URL']
|
|
browser_profile = Selenium::WebDriver::Chrome::Profile.new
|
|
browser_profile['intl.accept_languages'] = 'en'
|
|
browser_profile['loggingPref'] = { browser: :all }
|
|
end
|
|
end
|
|
browser_profile
|
|
end
|
|
|
|
def browser_support_cookies
|
|
if browser.match?(%r{(internet_explorer|ie)}i)
|
|
return false
|
|
end
|
|
|
|
true
|
|
end
|
|
|
|
def browser_url
|
|
return ENV['BROWSER_URL'] if ENV['BROWSER_URL'].present?
|
|
|
|
"http://#{host}:3000"
|
|
end
|
|
|
|
def host
|
|
return 'localhost' if ENV['CI'].blank?
|
|
|
|
Socket.ip_address_list.detect(&:ipv4_private?).ip_address
|
|
end
|
|
|
|
def browser_instance
|
|
@browsers ||= {}
|
|
if ENV['REMOTE_URL'].blank?
|
|
params = {
|
|
profile: profile,
|
|
}
|
|
if ENV['BROWSER_HEADLESS'].present?
|
|
case browser
|
|
when 'firefox'
|
|
params[:options] = Selenium::WebDriver::Firefox::Options.new
|
|
params[:options].add_argument('-headless')
|
|
when 'chrome'
|
|
params[:options] = Selenium::WebDriver::Chrome::Options.new
|
|
params[:options].add_argument('-headless')
|
|
end
|
|
end
|
|
local_browser = Selenium::WebDriver.for(browser.to_sym, params)
|
|
@browsers[local_browser.hash] = local_browser
|
|
browser_instance_preferences(local_browser)
|
|
return local_browser
|
|
end
|
|
|
|
# avoid "Cannot read property 'get_Current' of undefined" issues
|
|
(1..5).each do |count|
|
|
|
|
local_browser = browser_instance_remote
|
|
break
|
|
rescue => e
|
|
wait_until_ready = rand(5..13) # rubocop:disable Zammad/ForbidRand
|
|
log('browser_instance', { rescure: true, count: count, sleep: wait_until_ready, exception: e })
|
|
sleep wait_until_ready
|
|
|
|
end
|
|
|
|
local_browser
|
|
end
|
|
|
|
def browser_instance_remote
|
|
caps = Selenium::WebDriver::Remote::Capabilities.send(browser)
|
|
if ENV['BROWSER_OS']
|
|
caps.platform = ENV['BROWSER_OS']
|
|
end
|
|
if ENV['BROWSER_VERSION']
|
|
caps.version = ENV['BROWSER_VERSION']
|
|
end
|
|
|
|
# (ironically) required for timeout checks
|
|
# https://github.com/zalando/zalenium/issues/469#issuecomment-371417340
|
|
# https://opensource.zalando.com/zalenium/#usage
|
|
caps['idleTimeout'] = 300
|
|
|
|
http_client = Selenium::WebDriver::Remote::Http::Default.new(
|
|
open_timeout: 120,
|
|
read_timeout: 120
|
|
)
|
|
|
|
local_browser = Selenium::WebDriver.for(
|
|
:remote,
|
|
url: ENV['REMOTE_URL'],
|
|
desired_capabilities: caps,
|
|
http_client: http_client,
|
|
)
|
|
@browsers[local_browser.hash] = local_browser
|
|
browser_instance_preferences(local_browser)
|
|
|
|
# upload files from remote dir
|
|
local_browser.file_detector = lambda do |args|
|
|
str = args.first.to_s
|
|
str if File.file?(str)
|
|
end
|
|
|
|
local_browser
|
|
end
|
|
|
|
def browser_instance_close(local_browser)
|
|
return if !@browsers[local_browser.hash]
|
|
|
|
@browsers.delete(local_browser.hash)
|
|
local_browser.quit
|
|
end
|
|
|
|
def browser_instance_preferences(local_browser)
|
|
browser_width = ENV['BROWSER_WIDTH'] || 1024
|
|
browser_height = ENV['BROWSER_HEIGHT'] || 800
|
|
local_browser.manage.window.resize_to(browser_width, browser_height)
|
|
if !ENV['REMOTE_URL']&.match?(%r{saucelabs|(grid|ci)\.(zammad\.org|znuny\.com)}i)
|
|
if @browsers.count == 1
|
|
local_browser.manage.window.move_to(0, 0)
|
|
else
|
|
local_browser.manage.window.move_to(browser_width, 0)
|
|
end
|
|
end
|
|
local_browser.manage.timeouts.implicit_wait = 3 # seconds
|
|
end
|
|
|
|
def teardown
|
|
return if !@browsers
|
|
|
|
@browsers.each_value do |local_browser|
|
|
screenshot(browser: local_browser, comment: 'teardown')
|
|
browser_instance_close(local_browser)
|
|
end
|
|
end
|
|
|
|
def screenshot(params = {})
|
|
instance = params[:browser] || @browser
|
|
comment = params[:comment] || ''
|
|
filename = "tmp/#{Time.zone.now.strftime('screenshot_%Y_%m_%d__%H_%M_%S_%L')}_#{comment}#{instance.hash}.png"
|
|
log('screenshot', { filename: filename })
|
|
instance.save_screenshot(filename)
|
|
end
|
|
|
|
=begin
|
|
|
|
username = login(
|
|
browser: browser1,
|
|
username: 'someuser',
|
|
password: 'somepassword',
|
|
url: 'some url', # optional, in case of aleady opened brower a reload is done because url is called again
|
|
remember_me: true, # optional
|
|
auto_wizard: false, # optional, in case of auto wizard, skip login
|
|
success: false, #optional
|
|
)
|
|
|
|
=end
|
|
|
|
def login(params)
|
|
switch_window_focus(params)
|
|
log('login', params)
|
|
instance = params[:browser] || @browser
|
|
|
|
if params[:url]
|
|
instance.get(params[:url])
|
|
end
|
|
|
|
# submit logs anyway
|
|
instance.execute_script('App.Track.force()')
|
|
|
|
element = instance.find_elements(css: '#login input[name="username"]')[0]
|
|
if !element
|
|
|
|
if params[:auto_wizard]
|
|
watch_for(
|
|
browser: instance,
|
|
css: 'body',
|
|
value: 'auto wizard is enabled',
|
|
timeout: 10,
|
|
)
|
|
location(url: "#{browser_url}/#getting_started/auto_wizard")
|
|
sleep 10
|
|
login = instance.find_elements(css: '.user-menu .user a')[0].attribute('title')
|
|
if login != params[:username]
|
|
screenshot(browser: instance, comment: 'auto wizard login failed')
|
|
raise 'auto wizard login failed'
|
|
end
|
|
assert(true, 'auto wizard login ok')
|
|
|
|
clues_close(
|
|
browser: instance,
|
|
optional: true,
|
|
)
|
|
|
|
return
|
|
end
|
|
screenshot(browser: instance, comment: 'login_failed')
|
|
raise 'No login box found'
|
|
end
|
|
|
|
element.clear
|
|
element.send_keys(params[:username])
|
|
|
|
element = instance.find_elements(css: '#login input[name="password"]')[0]
|
|
element.clear
|
|
element.send_keys(params[:password])
|
|
|
|
if params[:remember_me]
|
|
instance.find_elements(css: '#login .checkbox-replacement')[0].click
|
|
end
|
|
instance.find_elements(css: '#login button')[0].click
|
|
|
|
sleep 4
|
|
login_failed = false
|
|
if instance.find_elements(css: '.user-menu .user a')[0]
|
|
login = instance.find_elements(css: '.user-menu .user a')[0].attribute('title')
|
|
if login != params[:username]
|
|
login_failed = true
|
|
end
|
|
else
|
|
login_failed = true
|
|
end
|
|
if login_failed
|
|
if params[:success] == false
|
|
assert(true, 'login not successfull, like wanted')
|
|
return true
|
|
end
|
|
screenshot(browser: instance, comment: 'login_failed')
|
|
raise 'login failed'
|
|
end
|
|
|
|
if params[:success] == false
|
|
raise 'login successfull but should not'
|
|
end
|
|
|
|
clues_close(
|
|
browser: instance,
|
|
optional: true,
|
|
)
|
|
|
|
assert(true, 'login ok')
|
|
login
|
|
end
|
|
|
|
=begin
|
|
|
|
logout(
|
|
browser: browser1
|
|
)
|
|
|
|
=end
|
|
|
|
def logout(params = {})
|
|
switch_window_focus(params)
|
|
log('logout', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#current_user"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#logout"]',
|
|
mute_log: true,
|
|
)
|
|
|
|
5.times do
|
|
sleep 1
|
|
login = instance.find_elements(css: '#login')[0]
|
|
|
|
next if !login
|
|
|
|
assert(true, 'logout ok')
|
|
return
|
|
end
|
|
screenshot(browser: instance, comment: 'logout_failed')
|
|
raise 'no login box found, seems logout was not successfully!'
|
|
end
|
|
|
|
=begin
|
|
|
|
clues_close(
|
|
browser: browser1,
|
|
optional: false,
|
|
)
|
|
|
|
=end
|
|
|
|
def clues_close(params = {})
|
|
switch_window_focus(params)
|
|
log('clues_close', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
clues = instance.find_elements(css: '.js-modal--clue .js-close')[0]
|
|
if !params[:optional] && !clues
|
|
screenshot(browser: instance, comment: 'no_clues')
|
|
raise 'Unable to closes clues, no clues found!'
|
|
end
|
|
return if !clues
|
|
|
|
checks = 25
|
|
previous = clues.location
|
|
(checks + 1).times do |check|
|
|
raise "Element still moving after #{checks} checks" if check == checks
|
|
|
|
current = clues.location
|
|
sleep 0.2 if ENV['CI']
|
|
break if previous == current
|
|
|
|
previous = current
|
|
end
|
|
clues.click
|
|
|
|
watch_for_disappear(
|
|
browser: instance,
|
|
css: 'modal-backdrop js-backdrop',
|
|
)
|
|
|
|
assert(true, 'clues closed')
|
|
end
|
|
|
|
=begin
|
|
|
|
notify_close(
|
|
browser: browser1,
|
|
optional: false,
|
|
)
|
|
|
|
=end
|
|
|
|
def notify_close(params = {})
|
|
switch_window_focus(params)
|
|
log('notify_close', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
notify = instance.find_elements(css: '.noty_inline_layout_container.i-am-new')[0]
|
|
if !params[:optional] && !notify
|
|
screenshot(browser: instance, comment: 'no_notify')
|
|
raise 'Unable to closes notify, no notify found!'
|
|
end
|
|
return if !notify
|
|
|
|
notify.click
|
|
assert(true, 'notify closed')
|
|
sleep 1
|
|
end
|
|
|
|
=begin
|
|
|
|
location(
|
|
browser: browser1,
|
|
url: 'http://someurl',
|
|
)
|
|
|
|
=end
|
|
|
|
def location(params)
|
|
switch_window_focus(params)
|
|
log('location', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
instance.get(params[:url])
|
|
|
|
# check if reload was successfull
|
|
if !instance.find_elements(css: 'body')[0] || instance.find_elements(css: 'body')[0].text =~ %r{unavailable or too busy}i
|
|
instance.navigate.refresh
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
location_check(
|
|
browser: browser1,
|
|
url: 'http://someurl',
|
|
)
|
|
|
|
=end
|
|
|
|
def location_check(params)
|
|
switch_window_focus(params)
|
|
log('location_check', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
sleep 0.7
|
|
current_url = instance.current_url
|
|
if !current_url.match?(%r{#{Regexp.quote(params[:url])}})
|
|
screenshot(browser: instance, comment: 'location_check_failed')
|
|
raise "url #{current_url} is not matching #{params[:url]}"
|
|
end
|
|
assert(true, "url #{current_url} is matching #{params[:url]}")
|
|
end
|
|
|
|
=begin
|
|
|
|
reload(
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def reload(params = {})
|
|
switch_window_focus(params)
|
|
log('reload', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
instance.navigate.refresh
|
|
|
|
# check if reload was successfull
|
|
if !instance.find_elements(css: 'body')[0] || instance.find_elements(css: 'body')[0].text =~ %r{unavailable or too busy}i
|
|
instance.navigate.refresh
|
|
end
|
|
screenshot(browser: instance, comment: 'reload_after')
|
|
end
|
|
|
|
=begin
|
|
|
|
click(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
fast: false, # do not wait
|
|
wait: 1, # wait 1 sec.
|
|
)
|
|
|
|
click(
|
|
browser: browser1,
|
|
xpath: '//a[contains(@class,".text-1")]',
|
|
fast: false, # do not wait
|
|
wait: 1, # wait 1 sec.
|
|
)
|
|
|
|
click(
|
|
browser: browser1,
|
|
text: '.partial_link_text',
|
|
fast: false, # do not wait
|
|
wait: 1, # wait 1 sec.
|
|
)
|
|
|
|
=end
|
|
|
|
def click(params)
|
|
switch_window_focus(params)
|
|
log('click', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
if params.include?(:css)
|
|
param_key = :css
|
|
find_element_key = :css
|
|
elsif params.include?(:xpath)
|
|
param_key = :xpath
|
|
find_element_key = :xpath
|
|
else
|
|
param_key = :text
|
|
find_element_key = :partial_link_text
|
|
sleep 0.5
|
|
end
|
|
|
|
begin
|
|
elements = instance.find_elements(find_element_key => params[param_key])
|
|
.tap { |e| e.slice!(1..-1) if !params[:all] }
|
|
|
|
if elements.empty?
|
|
return if params[:only_if_exists] == true
|
|
|
|
raise "No such element '#{params[param_key]}'"
|
|
end
|
|
|
|
# a clumsy substitute for elements.each(&:click)
|
|
# (we need to refresh element references after each element.click
|
|
# because if clicks alter page content,
|
|
# subsequent element.clicks will raise a StaleElementReferenceError)
|
|
elements.length.times do |i|
|
|
instance.find_elements(find_element_key => params[param_key])[i].try(:click)
|
|
end
|
|
rescue => e
|
|
raise e if (fail_count ||= 0).positive?
|
|
|
|
fail_count += 1
|
|
log('click', { rescure: true })
|
|
sleep 0.5
|
|
retry
|
|
end
|
|
|
|
sleep 0.2 if !params[:fast]
|
|
sleep params[:wait] if params[:wait]
|
|
|
|
if params[:expect_alert]
|
|
check_alert(params)
|
|
else
|
|
await_empty_ajax_queue(params)
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
perform_macro('Close & Tag as Spam')
|
|
|
|
# or
|
|
|
|
perform_macro(
|
|
name: 'Close & Tag as Spam',
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def perform_macro(params)
|
|
switch_window_focus(params)
|
|
log('perform_macro', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
click(
|
|
browser: instance,
|
|
css: '.active.content .js-submitDropdown .js-openDropdownMacro'
|
|
)
|
|
|
|
click(
|
|
browser: instance,
|
|
xpath: "//div[contains(@class, 'content') and contains(@class, 'active')]//li[contains(@class, 'js-dropdownActionMacro') and contains(text(), '#{params[:name]}')]"
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
scroll_to(
|
|
browser: browser1,
|
|
position: 'top', # botton
|
|
css: '.some_class',
|
|
)
|
|
|
|
=end
|
|
|
|
def scroll_to(params)
|
|
switch_window_focus(params)
|
|
log('scroll_to', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
position = 'true'
|
|
if params[:position] == 'botton'
|
|
position = 'false'
|
|
end
|
|
execute(
|
|
browser: instance,
|
|
js: "\$('#{params[:css]}').get(0).scrollIntoView(#{position})",
|
|
mute_log: params[:mute_log]
|
|
)
|
|
sleep 0.3
|
|
screenshot(browser: instance, comment: 'scroll_to_after')
|
|
end
|
|
|
|
=begin
|
|
|
|
modal_close(
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def modal_close(params = {})
|
|
switch_window_focus(params)
|
|
log('modal_close', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
element = instance.find_elements(css: '.modal .js-close')[0]
|
|
raise "No such modal to close #{params.inspect}" if !element
|
|
|
|
element.click
|
|
end
|
|
|
|
=begin
|
|
|
|
modal_ready(
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def modal_ready(params = {})
|
|
switch_window_focus(params)
|
|
log('modal_ready', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.modal.in',
|
|
timeout: params[:timeout] || 4,
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
modal_disappear(
|
|
browser: browser1,
|
|
timeout: 12, # default 8
|
|
)
|
|
|
|
=end
|
|
|
|
def modal_disappear(params = {})
|
|
switch_window_focus(params)
|
|
log('modal_disappear', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
watch_for_disappear(
|
|
browser: instance,
|
|
css: '.modal',
|
|
timeout: params[:timeout] || 8,
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
execute(
|
|
browser: browser1,
|
|
js: '.some_class',
|
|
)
|
|
|
|
=end
|
|
|
|
def execute(params)
|
|
switch_window_focus(params)
|
|
log('js', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
if params[:js]
|
|
return instance.execute_script(params[:js])
|
|
end
|
|
|
|
raise "Invalid execute params #{params.inspect}"
|
|
end
|
|
|
|
=begin
|
|
|
|
exists(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
)
|
|
|
|
exists(
|
|
displayed: false, # true|false
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
displayed: true, # true|false
|
|
)
|
|
|
|
=end
|
|
|
|
def exists(params)
|
|
retries ||= 0
|
|
|
|
switch_window_focus(params)
|
|
log('exists', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
if !instance.find_elements(css: params[:css])[0]
|
|
screenshot(browser: instance, comment: 'exists_failed')
|
|
raise "#{params[:css]} dosn't exist, but should"
|
|
end
|
|
|
|
if params.key?(:displayed)
|
|
if params[:displayed] == true && !instance.find_elements(css: params[:css])[0].displayed?
|
|
raise "#{params[:css]} is not displayed, but should"
|
|
end
|
|
if params[:displayed] == false && instance.find_elements(css: params[:css])[0].displayed?
|
|
raise "#{params[:css]} is displayed, but should not"
|
|
end
|
|
end
|
|
|
|
true
|
|
rescue
|
|
sleep retries
|
|
retries += 1
|
|
retry if retries < 3
|
|
end
|
|
|
|
=begin
|
|
|
|
exists_not(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
)
|
|
|
|
=end
|
|
|
|
def exists_not(params)
|
|
switch_window_focus(params)
|
|
log('exists_not', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
if instance.find_elements(css: params[:css])[0]
|
|
screenshot(browser: instance, comment: 'exists_not_failed')
|
|
raise "#{params[:css]} exists but should not"
|
|
end
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
set(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
value: true,
|
|
slow: false,
|
|
blur: true, # default false
|
|
clear: true, # todo | default: true
|
|
no_click: true,
|
|
)
|
|
|
|
=end
|
|
|
|
def set(params)
|
|
switch_window_focus(params)
|
|
log('set', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
begin
|
|
retries ||= 0
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
if !params[:no_click]
|
|
element.click
|
|
end
|
|
element.clear
|
|
rescue Selenium::WebDriver::Error::StaleElementReferenceError
|
|
sleep retries
|
|
retries += 1
|
|
retry if retries < 3
|
|
end
|
|
|
|
begin
|
|
if params[:slow]
|
|
element.send_keys('')
|
|
keys = params[:value].to_s.chars
|
|
keys.each do |key|
|
|
instance.action.send_keys(key).perform
|
|
end
|
|
else
|
|
element.send_keys(params[:value])
|
|
end
|
|
rescue
|
|
sleep 0.5
|
|
|
|
# just try again
|
|
log('set', { rescure: true })
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
raise "No such element '#{params[:css]}'" if !element
|
|
|
|
if params[:slow]
|
|
element.send_keys('')
|
|
keys = params[:value].to_s.chars
|
|
keys.each do |key|
|
|
instance.action.send_keys(key).perform
|
|
end
|
|
else
|
|
element.send_keys(params[:value])
|
|
end
|
|
end
|
|
|
|
# it's not working stable with ff via selenium, use js
|
|
if browser =~ %r{firefox}i && params[:css].include?('[data-name=')
|
|
log('set_ff_trigger_workaround', params)
|
|
instance.execute_script("$('#{params[:css]}').trigger('focusout')")
|
|
end
|
|
|
|
if params[:blur]
|
|
instance.execute_script("$('#{params[:css]}').blur()")
|
|
end
|
|
|
|
sleep 0.2
|
|
await_empty_ajax_queue(params)
|
|
end
|
|
|
|
=begin
|
|
|
|
select(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
value: 'Some Value',
|
|
deselect_all: false, # default false
|
|
)
|
|
|
|
=end
|
|
|
|
def select(params)
|
|
switch_window_focus(params)
|
|
log('select', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
# searchable select
|
|
element = instance.find_elements(css: "#{params[:css]}.js-shadow")[0]
|
|
if element
|
|
element = instance.find_elements(css: "#{params[:css]}.js-shadow + .js-input")[0]
|
|
element.click
|
|
element.clear
|
|
sleep 0.2
|
|
element.send_keys(params[:value])
|
|
sleep 0.2
|
|
element.send_keys(:enter)
|
|
sleep 0.2
|
|
return
|
|
end
|
|
|
|
# native select
|
|
begin
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
if params[:deselect_all]
|
|
dropdown.deselect_all
|
|
end
|
|
dropdown.select_by(:text, params[:value])
|
|
# puts "select - #{params.inspect}"
|
|
rescue
|
|
sleep 0.4
|
|
|
|
# just try again
|
|
log('select', { rescure: true })
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
if params[:deselect_all]
|
|
dropdown.deselect_all
|
|
end
|
|
dropdown.select_by(:text, params[:value])
|
|
# puts "select2 - #{params.inspect}"
|
|
end
|
|
|
|
await_empty_ajax_queue(params)
|
|
end
|
|
|
|
=begin
|
|
|
|
switch(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
type: 'on', # 'off'
|
|
no_check: true, # do not check is switch has changed, in case if js alert
|
|
)
|
|
|
|
=end
|
|
|
|
def switch(params)
|
|
switch_window_focus(params)
|
|
log('switch', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
element = instance.find_elements(css: "#{params[:css]} input[type=checkbox]")[0]
|
|
checked = element.attribute('checked')
|
|
|
|
if !checked
|
|
if params[:type] == 'on'
|
|
instance.find_elements(css: "#{params[:css]} label")[0].click
|
|
sleep 2
|
|
|
|
if params[:no_check] != true
|
|
element = instance.find_elements(css: "#{params[:css]} input[type=checkbox]")[0]
|
|
checked = element.attribute('checked')
|
|
raise 'Switch not on!' if !checked
|
|
end
|
|
end
|
|
elsif params[:type] == 'off'
|
|
instance.find_elements(css: "#{params[:css]} label")[0].click
|
|
sleep 2
|
|
|
|
if params[:no_check] != true
|
|
element = instance.find_elements(css: "#{params[:css]} input[type=checkbox]")[0]
|
|
checked = element.attribute('checked')
|
|
raise 'Switch not off!' if checked
|
|
end
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
check(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
)
|
|
|
|
=end
|
|
|
|
def check(params)
|
|
switch_window_focus(params)
|
|
log('check', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
instance.execute_script("$('#{params[:css]}:not(:checked)').click()")
|
|
# element = instance.find_elements(css: params[:css])[0]
|
|
# checked = element.attribute('checked')
|
|
# element.click if !checked
|
|
end
|
|
|
|
=begin
|
|
|
|
uncheck(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
)
|
|
|
|
=end
|
|
|
|
def uncheck(params)
|
|
switch_window_focus(params)
|
|
log('uncheck', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
instance.execute_script("$('#{params[:css]}:checked').click()")
|
|
# element = instance.find_elements(css: params[:css])[0]
|
|
# checked = element.attribute('checked')
|
|
# element.click if checked
|
|
end
|
|
|
|
=begin
|
|
|
|
sendkey(
|
|
browser: browser1,
|
|
value: :enter,
|
|
slow: false, # default false
|
|
)
|
|
|
|
=end
|
|
|
|
def sendkey(params)
|
|
switch_window_focus(params)
|
|
log('sendkey', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
element = nil
|
|
if params[:css]
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
end
|
|
if params[:value].instance_of?(Array)
|
|
params[:value].each do |key|
|
|
if element
|
|
element.send_keys(key)
|
|
else
|
|
instance.action.send_keys(key).perform
|
|
end
|
|
end
|
|
return
|
|
end
|
|
|
|
if element
|
|
element.send_keys(params[:value])
|
|
else
|
|
instance.action.send_keys(params[:value]).perform
|
|
end
|
|
if params[:slow]
|
|
sleep 1.5
|
|
else
|
|
sleep 0.2
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
match(
|
|
browser: browser1,
|
|
css: '#content .text-1',
|
|
value: 'some test for browser and some other for browser',
|
|
attribute: 'some_attribute', # match on attribute
|
|
should_not_match: true,
|
|
no_quote: false, # use regex
|
|
)
|
|
|
|
=end
|
|
|
|
def match(params, fallback = false)
|
|
switch_window_focus(params)
|
|
log('match', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
|
|
if params[:css].include?('select')
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
success = false
|
|
dropdown.selected_options&.each do |option|
|
|
if option.text == params[:value]
|
|
success = true
|
|
end
|
|
end
|
|
if params[:should_not_match]
|
|
if success
|
|
screenshot(browser: instance, comment: 'match_failed')
|
|
raise "should not match '#{params[:value]}' in select list, but is matching"
|
|
end
|
|
elsif !success
|
|
screenshot(browser: instance, comment: 'match_failed')
|
|
raise "not matching '#{params[:value]}' in select list"
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
# match on attribute
|
|
begin
|
|
text = if params[:attribute]
|
|
element.attribute(params[:attribute])
|
|
elsif params[:css].match?(%r{(input|textarea)}i)
|
|
element.attribute('value')
|
|
else
|
|
element.text
|
|
end
|
|
rescue => e
|
|
# just try again
|
|
if !fallback
|
|
return match(params, true)
|
|
end
|
|
|
|
raise e.inspect
|
|
end
|
|
|
|
# do cleanups (needed for richtext tests)
|
|
if params[:cleanup]
|
|
text.gsub!(%r{\s+$}m, '')
|
|
params[:value].gsub!(%r{\s+$}m, '')
|
|
end
|
|
|
|
match = false
|
|
if params[:no_quote]
|
|
# puts "aaaa #{text}/#{params[:value]}"
|
|
if text =~ %r{#{params[:value]}}i
|
|
match = $1 || true
|
|
end
|
|
elsif text.match?(%r{#{Regexp.quote(params[:value])}}i)
|
|
match = true
|
|
end
|
|
|
|
if match
|
|
if params[:should_not_match]
|
|
screenshot(browser: instance, comment: 'match_failed')
|
|
raise "matching '#{params[:value]}' in content '#{text}' but should not!"
|
|
end
|
|
elsif !params[:should_not_match]
|
|
screenshot(browser: instance, comment: 'match_failed')
|
|
raise "not matching '#{params[:value]}' in content '#{text}' but should!"
|
|
end
|
|
sleep 0.2
|
|
match
|
|
end
|
|
|
|
=begin
|
|
|
|
match_not(
|
|
browser: browser1,
|
|
css: '#content .text-1',
|
|
value: 'some test for browser and some other for browser',
|
|
attribute: 'some_attribute', # match on attribute
|
|
should_not_match: true,
|
|
no_quote: false, # use regex
|
|
)
|
|
|
|
=end
|
|
|
|
def match_not(params)
|
|
switch_window_focus(params)
|
|
log('match_not', params)
|
|
|
|
params[:should_not_match] = true
|
|
match(params)
|
|
end
|
|
|
|
=begin
|
|
|
|
Get the on-screen pixel coordinates of a given DOM element. Can be used to compare
|
|
the relative location of table rows before and after sort, for example.
|
|
|
|
Returns a Selenium::WebDriver::Point object. Use result.x and result.y to access
|
|
its X and Y coordinates respectively.
|
|
|
|
get_location(
|
|
browser: browser1,
|
|
css: '.some_class',
|
|
)
|
|
|
|
=end
|
|
|
|
def get_location(params)
|
|
switch_window_focus(params)
|
|
log('exists', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
if params[:css]
|
|
query = { css: params[:css] }
|
|
end
|
|
if params[:xpath]
|
|
query = { xpath: params[:xpath] }
|
|
end
|
|
if !instance.find_elements(query)[0]
|
|
screenshot(browser: instance, comment: 'exists_failed')
|
|
raise "#{query} dosn't exist, but should"
|
|
end
|
|
|
|
instance.find_elements(query)[0].location
|
|
end
|
|
|
|
=begin
|
|
|
|
set type of task (closeTab, closeNextInOverview, stayOnTab)
|
|
|
|
task_type(
|
|
browser: browser1,
|
|
type: 'stayOnTab',
|
|
)
|
|
|
|
=end
|
|
|
|
def task_type(params)
|
|
switch_window_focus(params)
|
|
log('task_type', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
if params[:type]
|
|
instance.find_elements(css: '.content.active .js-secondaryActionButtonLabel')[0].click
|
|
instance.find_elements(css: ".content.active .js-secondaryActionLabel[data-type=#{params[:type]}]")[0].click
|
|
return
|
|
end
|
|
raise "Unknown params for task_type: #{params.inspect}"
|
|
end
|
|
|
|
=begin
|
|
|
|
cookie(
|
|
browser: browser1,
|
|
name: '^_zammad.+?',
|
|
value: '.+?',
|
|
expires: nil,
|
|
)
|
|
|
|
cookie(
|
|
browser: browser1,
|
|
name: '^_zammad.+?',
|
|
should_not_exist: true,
|
|
)
|
|
|
|
=end
|
|
|
|
def cookie(params)
|
|
switch_window_focus(params)
|
|
log('cookie', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
if !browser_support_cookies
|
|
assert(true, "'#{params[:value]}' ups browser is not supporting reading cookies, go ahead")
|
|
return true
|
|
end
|
|
|
|
cookies = instance.manage.all_cookies
|
|
cookies.each do |cookie|
|
|
# :name=>"_zammad_session_c25832f4de2", :value=>"adc31cd21615cb0a7ab269184ec8b76f", :path=>"/", :domain=>"localhost", :expires=>nil, :secure=>false}
|
|
next if !cookie[:name].match?(%r{#{params[:name]}}i)
|
|
|
|
if params.key?(:value) && cookie[:value].to_s =~ %r{#{params[:value]}}i
|
|
assert(true, "matching value '#{params[:value]}' in cookie '#{cookie}'")
|
|
else
|
|
raise "not matching value '#{params[:value]}' in cookie '#{cookie}'"
|
|
end
|
|
if params.key?(:expires) && cookie[:expires].to_s =~ %r{#{params[:expires]}}i
|
|
assert(true, "matching expires '#{params[:expires].inspect}' in cookie '#{cookie}'")
|
|
else
|
|
raise "not matching expires '#{params[:expires]}' in cookie '#{cookie}'"
|
|
end
|
|
|
|
return if !params[:should_not_exist]
|
|
|
|
raise "cookie with name '#{params[:name]}' should not exist, but exists '#{cookies}'"
|
|
end
|
|
if params[:should_not_exist]
|
|
assert(true, "cookie with name '#{params[:name]}' is not existing")
|
|
return
|
|
end
|
|
raise "not matching name '#{params[:name]}' in cookie '#{cookies}'"
|
|
end
|
|
|
|
=begin
|
|
|
|
verify_title(
|
|
browser: browser1,
|
|
value: 'some title',
|
|
)
|
|
|
|
=end
|
|
|
|
def verify_title(params = {})
|
|
switch_window_focus(params)
|
|
log('verify_title', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
title = instance.title
|
|
if title.match?(%r{#{params[:value]}}i)
|
|
assert(true, "matching '#{params[:value]}' in title '#{title}'")
|
|
else
|
|
raise "not matching '#{params[:value]}' in title '#{title}'"
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
verify_task(
|
|
browser: browser1,
|
|
data: {
|
|
title: 'some title',
|
|
modified: true, # optional
|
|
}
|
|
)
|
|
|
|
=end
|
|
|
|
def verify_task(params = {})
|
|
switch_window_focus(params)
|
|
log('verify_task', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
begin
|
|
retries ||= 0
|
|
sleep 1
|
|
|
|
# verify title
|
|
if data[:title]
|
|
title = instance.find_elements(css: '.tasks .is-active')[0].text.strip
|
|
if title.match?(%r{#{data[:title]}}i)
|
|
assert(true, "matching '#{data[:title]}' in title '#{title}'")
|
|
else
|
|
screenshot(browser: instance, comment: 'verify_task_failed')
|
|
raise "not matching '#{data[:title]}' in title '#{title}'"
|
|
end
|
|
end
|
|
|
|
# verify modified
|
|
if data.key?(:modified)
|
|
exists = instance.find_elements(css: '.tasks .is-active')[0]
|
|
is_modified = instance.find_elements(css: '.tasks .is-modified')[0]
|
|
puts "m #{data[:modified].inspect}"
|
|
if exists
|
|
puts ' exists'
|
|
end
|
|
if is_modified
|
|
puts ' is_modified'
|
|
end
|
|
if data[:modified] == true
|
|
if is_modified
|
|
assert(true, "task '#{data[:title]}' is modifed")
|
|
elsif !exists
|
|
screenshot(browser: instance, comment: 'verify_task_failed')
|
|
raise "task '#{data[:title]}' not exists, should not modified"
|
|
else
|
|
screenshot(browser: instance, comment: 'verify_task_failed')
|
|
raise "task '#{data[:title]}' is not modifed"
|
|
end
|
|
elsif !is_modified
|
|
assert(true, "task '#{data[:title]}' is modifed")
|
|
elsif !exists
|
|
screenshot(browser: instance, comment: 'verify_task_failed')
|
|
raise "task '#{data[:title]}' not exists, should be not modified"
|
|
else
|
|
screenshot(browser: instance, comment: 'verify_task_failed')
|
|
raise "task '#{data[:title]}' is modifed, but should not"
|
|
end
|
|
end
|
|
rescue => e
|
|
retries += 1
|
|
retry if retries < 5
|
|
raise "ERROR: #{e.inspect}"
|
|
end
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
open_task(
|
|
browser: browser1,
|
|
data: {
|
|
title: 'some title',
|
|
}
|
|
)
|
|
|
|
=end
|
|
|
|
def open_task(params = {})
|
|
switch_window_focus(params)
|
|
log('open_task', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
element = instance.find_element(css: '#navigation').find_element(partial_link_text: data[:title])
|
|
if !element
|
|
screenshot(browser: instance, comment: 'open_task_failed')
|
|
raise "no task with title '#{data[:title]}' found"
|
|
end
|
|
# firefix/marionette issue with Selenium::WebDriver::Error::ElementNotInteractableError: could not be scrolled into view
|
|
# use js workaround instead of native click
|
|
instance.execute_script("$('#navigation .tasks .task:contains(\"#{data[:title]}\") .nav-tab-name').click()")
|
|
# element.click
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
close_task(
|
|
browser: browser1,
|
|
data: {
|
|
title: 'some title',
|
|
},
|
|
discard_changes: true,
|
|
)
|
|
|
|
=end
|
|
|
|
def close_task(params = {})
|
|
switch_window_focus(params)
|
|
log('close_task', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
element = instance.find_element(css: '#navigation').find_element(partial_link_text: data[:title])
|
|
if !element
|
|
screenshot(browser: instance, comment: 'close_task_failed')
|
|
raise "no task with title '#{data[:title]}' found"
|
|
end
|
|
|
|
instance.action.move_to(element).release.perform
|
|
sleep 0.1
|
|
instance.execute_script("$('#navigation .tasks .task:contains(\"#{data[:title]}\") .js-close').click()")
|
|
|
|
# accept task close warning
|
|
if params[:discard_changes]
|
|
modal_ready(browser: instance)
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
end
|
|
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
file_upload(
|
|
browser: browser1,
|
|
css: '.content.active .attachmentPlaceholder-inputHolder input'
|
|
files: ['path/in/home/some_file.ext'], # 'test/data/pdf/test1.pdf'
|
|
)
|
|
|
|
=end
|
|
|
|
def file_upload(params = {})
|
|
switch_window_focus(params)
|
|
log('file_upload', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
params[:files].each do |file|
|
|
instance.find_elements(css: params[:css])[0].send_keys(Rails.root.join(file))
|
|
end
|
|
return if params[:no_sleep]
|
|
|
|
sleep 2 * params[:files].count
|
|
end
|
|
|
|
=begin
|
|
|
|
watch_for(
|
|
browser: browser1,
|
|
container: element # optional, defaults to browser, must exist at the time of dispatch
|
|
css: '#content .text-1', # xpath or css required
|
|
xpath: '/content[contains(@class,".text-1")]', # xpath or css required
|
|
value: 'some text',
|
|
attribute: 'some_attribute' # optional
|
|
timeout: 16, # in sec, default 16
|
|
)
|
|
|
|
=end
|
|
|
|
def watch_for(params = {})
|
|
switch_window_focus(params)
|
|
log('watch_for', params)
|
|
|
|
browser = params[:browser] || @browser
|
|
instance = params[:container] || browser
|
|
|
|
selector = params[:css] || params[:xpath]
|
|
selector_type = if params.key?(:css)
|
|
:css
|
|
elsif params.key?(:xpath)
|
|
:xpath
|
|
end
|
|
|
|
timeout = 16
|
|
if params[:timeout]
|
|
timeout = params[:timeout]
|
|
end
|
|
loops = timeout.to_i * 2
|
|
text = ''
|
|
(1..loops).each do
|
|
element = instance.find_elements(selector_type => selector)[0]
|
|
if element # && element.displayed?
|
|
begin
|
|
# watch for selector
|
|
if !params[:attribute] && !params[:value]
|
|
assert(true, "'#{selector}' found")
|
|
sleep 0.5
|
|
return true
|
|
|
|
# match an attribute
|
|
else
|
|
text = if params[:attribute]
|
|
element.attribute(params[:attribute])
|
|
elsif selector.match?(%r{(input|textarea)}i)
|
|
element.attribute('value')
|
|
else
|
|
element.text
|
|
end
|
|
if text.match?(%r{#{params[:value]}}i)
|
|
assert(true, "'#{params[:value]}' found in '#{text}'")
|
|
sleep 0.5
|
|
return true
|
|
end
|
|
end
|
|
rescue
|
|
# try again
|
|
end
|
|
end
|
|
sleep 0.5
|
|
end
|
|
screenshot(browser: browser, comment: 'watch_for_failed')
|
|
if !params[:attribute] && !params[:value]
|
|
raise "'#{selector}' not found"
|
|
end
|
|
|
|
raise "'#{params[:value]}' not found in '#{text}'"
|
|
end
|
|
|
|
=begin
|
|
|
|
wait untill selector disabppears
|
|
|
|
watch_for_disappear(
|
|
browser: browser1,
|
|
css: '#content .text-1',
|
|
timeout: 16, # in sec, default 16
|
|
)
|
|
|
|
wait untill text in selector disabppears
|
|
|
|
watch_for_disappear(
|
|
browser: browser1,
|
|
css: '#content .text-1',
|
|
value: 'some value as regexp',
|
|
timeout: 16, # in sec, default 16
|
|
)
|
|
|
|
=end
|
|
|
|
def watch_for_disappear(params = {})
|
|
switch_window_focus(params)
|
|
log('watch_for_disappear', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
timeout = 16
|
|
if params[:timeout]
|
|
timeout = params[:timeout]
|
|
end
|
|
loops = timeout.to_i
|
|
text = ''
|
|
(1..loops).each do
|
|
element = instance.find_elements(css: params[:css])[0]
|
|
if !element # || element.displayed?
|
|
assert(true, 'not found')
|
|
sleep 1
|
|
return true
|
|
end
|
|
if params[:value]
|
|
begin
|
|
text = instance.find_elements(css: params[:css])[0].text
|
|
if !text.match?(%r{#{params[:value]}}i)
|
|
assert(true, "not matching '#{params[:value]}' in text '#{text}'")
|
|
sleep 1
|
|
return true
|
|
end
|
|
rescue
|
|
# try again
|
|
end
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'disappear_failed')
|
|
raise "#{params[:css]}) still exsists"
|
|
end
|
|
|
|
=begin
|
|
|
|
shortcut(
|
|
browser: browser1,
|
|
key: 'x',
|
|
)
|
|
|
|
=end
|
|
|
|
def shortcut(params = {})
|
|
switch_window_focus(params)
|
|
log('shortcut', params)
|
|
instance = params[:browser] || @browser
|
|
screenshot(browser: instance, comment: 'shortcut_before')
|
|
instance.action.key_down(:control)
|
|
.key_down(:shift)
|
|
.send_keys(params[:key])
|
|
.key_up(:shift)
|
|
.key_up(:control)
|
|
.perform
|
|
screenshot(browser: instance, comment: 'shortcut_after')
|
|
await_empty_ajax_queue(params)
|
|
end
|
|
|
|
=begin
|
|
|
|
window_keys(
|
|
browser: browser1,
|
|
value: 'x',
|
|
)
|
|
|
|
=end
|
|
|
|
def window_keys(params = {})
|
|
switch_window_focus(params)
|
|
log('window_keys', params)
|
|
instance = params[:browser] || @browser
|
|
instance.action.send_keys(params[:value]).perform
|
|
await_empty_ajax_queue(params)
|
|
end
|
|
|
|
=begin
|
|
|
|
tasks_close_all(
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def tasks_close_all(params = {})
|
|
switch_window_focus(params)
|
|
log('tasks_close_all', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
99.times do
|
|
# sleep 0.5
|
|
|
|
if instance.find_elements(css: '#navigation .tasks .task:first-child')[0]
|
|
instance.action.move_to(instance.find_elements(css: '#navigation .tasks .task:first-child')[0]).release.perform
|
|
click_element = instance.find_elements(css: '#navigation .tasks .task:first-child .js-close')[0]
|
|
if click_element
|
|
click_element.click
|
|
|
|
# accept task close warning
|
|
if instance.find_elements(css: '.modal button.js-submit')[0]
|
|
sleep 0.4
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
end
|
|
end
|
|
else
|
|
break
|
|
end
|
|
rescue
|
|
# Firefox doesn't move the mouse if it's already at the position.
|
|
# Therefore the hover event is not triggered in all cases.
|
|
# That's why we move the mouse a bit as a workaround and try again.
|
|
# The last working selenium version was: https://github.com/elgalu/docker-selenium/releases/tag/3.14.0-p17
|
|
instance.action.move_by(100, 100).perform
|
|
|
|
# try again
|
|
end
|
|
assert(true, 'all tasks closed')
|
|
end
|
|
|
|
=begin
|
|
|
|
close_online_notitifcation(
|
|
browser: browser1,
|
|
data: {
|
|
#title: 'some title',
|
|
position: 3,
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def close_online_notitifcation(params = {})
|
|
switch_window_focus(params)
|
|
log('close_online_notitifcation', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
if data[:title]
|
|
element = instance.find_elements(partial_link_text: data[:title])[0]
|
|
if !element
|
|
screenshot(browser: instance, comment: 'close_online_notitifcation')
|
|
raise "no online notification with title '#{data[:title]}' found"
|
|
end
|
|
instance.action.move_to(element).release.perform
|
|
sleep 0.1
|
|
instance.execute_script("$('.js-notificationsContainer .js-items .js-item .activity-text:contains(\"#{data[:title]}\") .js-remove').first().click()")
|
|
|
|
else
|
|
css = ".js-notificationsContainer .js-items .js-item:nth-child(#{data[:position]})"
|
|
element = instance.find_elements(css: css)[0]
|
|
if !element
|
|
screenshot(browser: instance, comment: 'close_online_notitifcation')
|
|
raise "no online notification with postion '#{css}' found"
|
|
end
|
|
|
|
instance.action.move_to(element).release.perform
|
|
sleep 0.1
|
|
instance.find_elements(css: "#{css} .js-remove")[0].click
|
|
end
|
|
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
online_notitifcation_close_all(
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def online_notitifcation_close_all(params = {})
|
|
switch_window_focus(params)
|
|
log('online_notitifcation_close_all', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
99.times do
|
|
sleep 0.5
|
|
begin
|
|
if instance.find_elements(css: '.js-notificationsContainer .js-item:first-child')[0]
|
|
instance.action.move_to(instance.find_elements(css: '.js-notificationsContainer .js-item:first-child')[0]).perform
|
|
sleep 0.1
|
|
click_element = instance.find_elements(css: '.js-notificationsContainer .js-item:first-child .js-remove')[0]
|
|
click_element&.click
|
|
else
|
|
break
|
|
end
|
|
rescue
|
|
# try again
|
|
end
|
|
end
|
|
assert(true, 'all online notification closed')
|
|
end
|
|
|
|
=begin
|
|
|
|
empty_search(
|
|
browser: browser1,
|
|
)
|
|
|
|
=end
|
|
|
|
def empty_search(params = {})
|
|
switch_window_focus(params)
|
|
log('empty_search', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
# empty search box by x
|
|
begin
|
|
instance.find_elements(css: '.search .js-emptySearch')[0].click
|
|
rescue
|
|
# in issues with ff & selenium, sometimes exeption appears
|
|
# "Element is not currently visible and so may not be interacted with"
|
|
log('empty_search via js')
|
|
instance.execute_script('$(".search .js-emptySearch").click()')
|
|
end
|
|
sleep 0.5
|
|
text = instance.find_elements(css: '#global-search')[0].attribute('value')
|
|
if !text
|
|
raise '#global-search is not empty!'
|
|
end
|
|
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket_customer_select(
|
|
browser: browser1,
|
|
css: '#content .text-1',
|
|
customer: '',
|
|
)
|
|
|
|
=end
|
|
|
|
def ticket_customer_select(params = {})
|
|
switch_window_focus(params)
|
|
log('ticket_customer_select', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
element = instance.find_elements(css: %(#{params[:css]} input[name="customer_id_completion"]))[0]
|
|
element.click
|
|
element.clear
|
|
|
|
element.send_keys(params[:customer])
|
|
sleep 2.5
|
|
|
|
element.send_keys(:enter)
|
|
# instance.find_elements(css: params[:css] + ' .recipientList-entry.js-object.is-active')[0].click
|
|
sleep 0.4
|
|
assert(true, 'ticket_customer_select')
|
|
end
|
|
|
|
=begin
|
|
|
|
overview_create(
|
|
browser: browser1,
|
|
data: {
|
|
name: name,
|
|
roles: ['Agent'],
|
|
selector: {
|
|
'Priority': '1 low',
|
|
},
|
|
'order::direction' => 'down',
|
|
}
|
|
)
|
|
|
|
=end
|
|
|
|
def overview_create(params)
|
|
switch_window_focus(params)
|
|
log('overview_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/overviews"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
if data[:name]
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=name]',
|
|
value: data[:name],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:roles]
|
|
99.times do
|
|
|
|
element = instance.find_elements(css: '.modal .js-selected[data-name=role_ids] .js-option:not(.is-hidden)')[0]
|
|
break if !element
|
|
|
|
element.click
|
|
sleep 0.1
|
|
|
|
end
|
|
data[:roles].each do |role|
|
|
instance.execute_script("$(\".modal [data-name=role_ids] .js-pool .js-option:not(.is-hidden):contains('#{role}')\").first().click()")
|
|
end
|
|
end
|
|
|
|
data[:attributes]&.each do |key, value|
|
|
if value
|
|
check(
|
|
browser: instance,
|
|
css: ".modal .checkbox input[value=\"#{key}\"]",
|
|
)
|
|
else
|
|
uncheck(
|
|
browser: instance,
|
|
css: ".modal .checkbox input[value=\"#{key}\"]",
|
|
)
|
|
end
|
|
end
|
|
|
|
data[:selector]&.each do |key, value|
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_selector .js-attributeSelector select',
|
|
value: key,
|
|
mute_log: true,
|
|
)
|
|
sleep 0.5
|
|
if data.key?('text_input')
|
|
set(
|
|
browser: instance,
|
|
css: '.modal .ticket_selector .js-value input',
|
|
value: value,
|
|
mute_log: true,
|
|
)
|
|
elsif value.instance_of? Array
|
|
value.each do |item|
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_selector .js-value select',
|
|
value: item,
|
|
mute_log: true,
|
|
)
|
|
end
|
|
else
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_selector .js-value select',
|
|
value: value,
|
|
deselect_all: true,
|
|
mute_log: true,
|
|
)
|
|
end
|
|
end
|
|
|
|
if data['order::direction']
|
|
select(
|
|
browser: instance,
|
|
css: '.modal select[name="order::direction"]',
|
|
value: data['order::direction'],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:group_by]
|
|
select(
|
|
browser: instance,
|
|
css: '.modal select[name="group_by"]',
|
|
value: data[:group_by],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:group_direction]
|
|
select(
|
|
browser: instance,
|
|
css: '.modal select[name="group_direction"]',
|
|
value: data[:group_direction],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
11.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'overview created')
|
|
overview = {
|
|
name: name,
|
|
}
|
|
sleep 1
|
|
return overview
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'overview_create_failed')
|
|
raise 'overview creation failed'
|
|
end
|
|
|
|
=begin
|
|
|
|
overview_update(
|
|
browser: browser1,
|
|
data: {
|
|
name: name,
|
|
roles: ['Agent'],
|
|
selector: {
|
|
'Priority': '1 low',
|
|
},
|
|
'order::direction' => 'down',
|
|
}
|
|
)
|
|
|
|
=end
|
|
|
|
def overview_update(params)
|
|
switch_window_focus(params)
|
|
log('overview_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/overviews"]',
|
|
mute_log: true,
|
|
)
|
|
|
|
instance.execute_script("$(\".content.active td:contains('#{data[:name]}')\").first().click()")
|
|
sleep 2
|
|
|
|
if data[:name]
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=name]',
|
|
value: data[:name],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
if data[:roles]
|
|
99.times do
|
|
|
|
element = instance.find_elements(css: '.modal .js-selected[data-name=role_ids] .js-option:not(.is-hidden)')[0]
|
|
break if !element
|
|
|
|
element.click
|
|
sleep 0.1
|
|
|
|
end
|
|
data[:roles].each do |role|
|
|
instance.execute_script("$(\".modal [data-name=role_ids] .js-pool .js-option:not(.is-hidden):contains('#{role}')\").first().click()")
|
|
end
|
|
end
|
|
|
|
data[:selector]&.each do |key, value|
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_selector .js-attributeSelector select',
|
|
value: key,
|
|
mute_log: true,
|
|
)
|
|
sleep 0.5
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_selector .js-value select',
|
|
value: value,
|
|
deselect_all: true,
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data['order::direction']
|
|
select(
|
|
browser: instance,
|
|
css: '.modal select[name="order::direction"]',
|
|
value: data['order::direction'],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:group_direction]
|
|
select(
|
|
browser: instance,
|
|
css: '.modal select[name="group_direction"]',
|
|
value: data[:group_direction],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
11.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'overview updated')
|
|
overview = {
|
|
name: name,
|
|
}
|
|
sleep 1
|
|
return overview
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'overview_update_failed')
|
|
raise 'overview update failed'
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket = ticket_create(
|
|
browser: browser1,
|
|
data: {
|
|
customer: 'nico',
|
|
group: 'Users', # optional / '-NONE-' # if group selection should not be shown
|
|
priority: '2 normal',
|
|
state: 'open',
|
|
title: 'overview #1',
|
|
body: 'overview #1',
|
|
},
|
|
do_not_submit: true,
|
|
)
|
|
|
|
returns (in case of submitted)
|
|
{
|
|
id: 123,
|
|
number: '100001',
|
|
title: 'overview #1',
|
|
}
|
|
|
|
ticket = ticket_create(
|
|
browser: browser1,
|
|
data: {
|
|
customer: 'nico',
|
|
group: 'Users', # optional / '-NONE-' # if group selection should not be shown
|
|
priority: '2 normal',
|
|
state: 'open',
|
|
title: 'overview #1',
|
|
body: 'overview #1',
|
|
},
|
|
custom_data_select: {
|
|
key1: 'some value',
|
|
},
|
|
custom_data_input: {
|
|
key1: 'some value',
|
|
},
|
|
custom_data_date: {
|
|
key!: '02/28/2018',
|
|
}
|
|
disable_group_check: true,
|
|
)
|
|
|
|
ticket = ticket_create(
|
|
browser: browser1,
|
|
data: {
|
|
customer: 'nico',
|
|
priority: '2 normal',
|
|
state: 'pending close',
|
|
pending_date: '11/24/2018',
|
|
pending_time: '08:00',
|
|
title: 'overview #1',
|
|
body: 'overview #1',
|
|
},
|
|
do_not_submit: true,
|
|
)
|
|
=end
|
|
|
|
def ticket_create(params)
|
|
switch_window_focus(params)
|
|
log('ticket_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#new"]',
|
|
mute_log: true,
|
|
only_if_exists: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#ticket/create"]',
|
|
mute_log: true,
|
|
)
|
|
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active .newTicket',
|
|
timeout: 30,
|
|
)
|
|
|
|
# Rumors say there is a modal reaper which will kill your modals if you dont sleep before a new ticket create
|
|
sleep 3
|
|
|
|
if data[:group]
|
|
if data[:group] == '-NONE-'
|
|
|
|
# check if owner selection exists
|
|
count = instance.find_elements(css: '.content.active .newTicket select[name="group_id"] option').count
|
|
if count.nonzero?
|
|
instance.find_elements(css: '.content.active .newTicket select[name="group_id"] option').each do |element|
|
|
log('ticket_create invalid group count', text: element.text)
|
|
end
|
|
end
|
|
assert_equal(2, count, 'group_id selection should not be shown because of only one group exists (auto select + hide)')
|
|
|
|
# check count of agents, should be only 3 / - selection + master + agent on init screen
|
|
count = instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').count
|
|
if count != 3
|
|
instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').each do |element|
|
|
log('ticket_create invalid owner count', text: element.text)
|
|
end
|
|
end
|
|
assert_equal(3, count, 'check if owner selection is - selection + master + agent per default')
|
|
else
|
|
|
|
# check count of agents, should be only 1 selection, the "-" selection on init screen
|
|
if !params[:disable_group_check]
|
|
count = instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').count
|
|
if count != 1
|
|
instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').each do |element|
|
|
log('ticket_create invalid owner count', text: element.text)
|
|
end
|
|
end
|
|
assert_equal(1, count, 'check if owner selection is empty per default')
|
|
end
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .newTicket select[name="group_id"]',
|
|
value: data[:group],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
end
|
|
if data[:priority]
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .newTicket select[name="priority_id"]',
|
|
value: data[:priority],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
if data[:state]
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .newTicket select[name="state_id"]',
|
|
value: data[:state],
|
|
mute_log: true,
|
|
)
|
|
if ['pending close', 'pending reminder'].include?(data[:state]) &&
|
|
data[:pending_date] &&
|
|
data[:pending_time]
|
|
set(
|
|
browser: instance,
|
|
css: '.content.active .newTicket input.js-datepicker',
|
|
value: data[:pending_date],
|
|
clear: true,
|
|
mute_log: true,
|
|
)
|
|
set(
|
|
browser: instance,
|
|
css: '.content.active .newTicket input.js-timepicker',
|
|
value: data[:pending_time],
|
|
clear: true,
|
|
mute_log: true,
|
|
)
|
|
end
|
|
end
|
|
if data[:title]
|
|
set(
|
|
browser: instance,
|
|
css: '.content.active .newTicket input[name="title"]',
|
|
value: data[:title],
|
|
clear: true,
|
|
mute_log: true,
|
|
)
|
|
end
|
|
if data[:body]
|
|
set(
|
|
browser: instance,
|
|
css: '.content.active .newTicket div[data-name=body]',
|
|
value: data[:body],
|
|
clear: true,
|
|
mute_log: true,
|
|
)
|
|
end
|
|
if data[:customer]
|
|
element = instance.find_elements(css: '.content.active .newTicket input[name="customer_id_completion"]')[0]
|
|
element.click
|
|
element.clear
|
|
|
|
# ff issue, sometimes focus event gets dropped
|
|
# if drowdown is not open, try it again
|
|
if !instance.find_elements(css: '.content.active .newTicket .js-recipientDropdown.open')[0]
|
|
instance.execute_script('$(".active .newTicket .js-recipientDropdown").addClass("open")')
|
|
end
|
|
|
|
element.send_keys(data[:customer])
|
|
sleep 2.5
|
|
|
|
element.send_keys(:enter)
|
|
sleep 0.2
|
|
# ff issue, sometimes enter event gets dropped
|
|
# take user manually
|
|
if instance.find_elements(css: '.content.active .newTicket .js-recipientDropdown.open')[0]
|
|
instance.find_elements(css: '.content.active .newTicket .recipientList-entry.js-object.is-active')[0].click
|
|
sleep 0.4
|
|
end
|
|
end
|
|
|
|
params[:custom_data_select]&.each do |local_key, local_value|
|
|
select(
|
|
browser: instance,
|
|
css: ".content.active .newTicket select[name=\"#{local_key}\"]",
|
|
value: local_value,
|
|
)
|
|
end
|
|
params[:custom_data_input]&.each do |local_key, local_value|
|
|
set(
|
|
browser: instance,
|
|
css: ".content.active .newTicket input[name=\"#{local_key}\"]",
|
|
value: local_value,
|
|
clear: true,
|
|
)
|
|
end
|
|
params[:custom_data_date]&.each do |local_key, local_value|
|
|
set(
|
|
browser: instance,
|
|
css: ".content.active .newTicket div[data-name=\"#{local_key}\"] input[data-item=\"date\"]",
|
|
value: local_value,
|
|
clear: true,
|
|
)
|
|
end
|
|
|
|
if data[:attachment]
|
|
file_upload(
|
|
browser: instance,
|
|
css: '.content.active .text-1',
|
|
value: 'some text',
|
|
)
|
|
end
|
|
|
|
if params[:do_not_submit]
|
|
assert(true, 'ticket created without submit')
|
|
return
|
|
end
|
|
|
|
# instance.execute_script('$(".content.active .newTicket form").submit();')
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active .newTicket button.js-submit',
|
|
mute_log: true,
|
|
)
|
|
|
|
sleep 1
|
|
9.times do
|
|
if instance.current_url.match?(%r{#{Regexp.quote('#ticket/zoom/')}})
|
|
assert(true, 'ticket created')
|
|
sleep 2
|
|
id = instance.current_url
|
|
id.gsub!(%r{},)
|
|
id.gsub!(%r{^.+?/(\d+)$}, '\\1')
|
|
|
|
element = instance.find_elements(css: '.content.active .ticketZoom-header .ticket-number')[0]
|
|
if element
|
|
number = element.text
|
|
ticket = {
|
|
id: id,
|
|
number: number,
|
|
title: data[:title],
|
|
}
|
|
sleep 2 # wait until notify is gone
|
|
return ticket
|
|
end
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'ticket_create_failed')
|
|
raise "ticket creation failed, can't get zoom url (current url is '#{instance.current_url}')"
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket_update(
|
|
browser: browser1,
|
|
data: {
|
|
title: '',
|
|
customer: 'some_customer@example.com',
|
|
body: 'some body',
|
|
group: 'some group', # optional
|
|
priority: '1 low',
|
|
state: 'closed',
|
|
},
|
|
do_not_submit: true,
|
|
)
|
|
|
|
ticket_update(
|
|
browser: browser1,
|
|
data: {
|
|
title: '',
|
|
customer: 'some_customer@example.com',
|
|
body: 'some body',
|
|
group: 'some group', # optional
|
|
priority: '1 low',
|
|
state: 'closed',
|
|
},
|
|
custom_data_select: {
|
|
key1: 'some value',
|
|
},
|
|
custom_data_input: {
|
|
key1: 'some value',
|
|
},
|
|
custom_data_date: {
|
|
key1: '02/21/2018',
|
|
},
|
|
do_not_submit: true,
|
|
task_type: 'stayOnTab', # default: stayOnTab / possible: closeTab, closeNextInOverview, stayOnTab
|
|
)
|
|
|
|
=end
|
|
|
|
def ticket_update(params)
|
|
switch_window_focus(params)
|
|
log('ticket_update', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
if data[:title]
|
|
# element = instance.find_elements(:css => '.content.active .ticketZoom-header .js-objectTitle')[0]
|
|
# element.clear
|
|
# sleep 0.5
|
|
# element = instance.find_elements(:css => '.content.active .ticketZoom-header .js-objectTitle')[0]
|
|
# element.send_keys(data[:title])
|
|
# sleep 0.5
|
|
# element.send_keys(:tab)
|
|
|
|
instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").focus()')
|
|
instance.execute_script(%($(".content.active .ticketZoom-header .js-objectTitle").text("#{data[:title]}")))
|
|
instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").blur()')
|
|
instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").trigger("blur")')
|
|
# {
|
|
# :where => :instance2,
|
|
# :execute => 'sendkey',
|
|
# :css => '.content.active .ticketZoom-header .js-objectTitle',
|
|
# :value => 'TTT',
|
|
# },
|
|
# {
|
|
# :where => :instance2,
|
|
# :execute => 'sendkey',
|
|
# :css => '.content.active .ticketZoom-header .js-objectTitle',
|
|
# :value => :tab,
|
|
# },
|
|
end
|
|
if data[:customer]
|
|
|
|
# select tab
|
|
click(browser: instance, css: '.content.active .tabsSidebar-tab[data-tab="customer"]')
|
|
|
|
click(browser: instance, css: '.content.active div[data-tab="customer"] .js-actions .icon-arrow-down')
|
|
click(browser: instance, css: '.content.active div[data-tab="customer"] .js-actions [data-type="customer-change"]')
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.modal',
|
|
value: 'change',
|
|
)
|
|
|
|
element = instance.find_elements(css: '.modal input[name="customer_id_completion"]')[0]
|
|
element.click
|
|
element.clear
|
|
|
|
element.send_keys(data[:customer])
|
|
sleep 2.5
|
|
|
|
element.send_keys(:enter)
|
|
# instance.find_elements(css: '.modal .user_autocompletion .recipientList-entry.js-object.is-active')[0].click
|
|
sleep 0.2
|
|
|
|
click(browser: instance, css: '.modal .js-submit')
|
|
|
|
modal_disappear(browser: instance)
|
|
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active .tabsSidebar',
|
|
value: data[:customer],
|
|
)
|
|
|
|
# select tab
|
|
click(browser: instance, css: '.content.active .tabsSidebar-tab[data-tab="ticket"]')
|
|
|
|
end
|
|
if data[:body]
|
|
set(
|
|
browser: instance,
|
|
css: '.content.active div[data-name=body]',
|
|
value: data[:body],
|
|
no_click: true,
|
|
mute_log: true,
|
|
)
|
|
|
|
# it's not working stable via selenium, use js
|
|
value = instance.find_elements(css: '.content.active div[data-name=body]')[0].text
|
|
if value != data[:body]
|
|
body_quoted = quote(data[:body])
|
|
instance.execute_script("$('.content.active div[data-name=body]').html('#{body_quoted}').trigger('focusout')")
|
|
end
|
|
|
|
end
|
|
|
|
if data[:group]
|
|
if data[:group] == '-NONE-'
|
|
|
|
# check if owner selection exists
|
|
count = instance.find_elements(css: '.content.active .sidebar select[name="group_id"] option').count
|
|
assert_equal(2, count, 'group_id selection should not be shown because of only one group exists (auto select + hide)')
|
|
|
|
# check count of agents, should be only 3 / - selection + master + agent on init screen
|
|
count = instance.find_elements(css: '.content.active .sidebar select[name="owner_id"] option').count
|
|
assert_equal(3, count, 'check if owner selection is - selection + master + agent per default')
|
|
|
|
else
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .sidebar select[name="group_id"]',
|
|
value: data[:group],
|
|
mute_log: true,
|
|
)
|
|
sleep 0.2
|
|
end
|
|
end
|
|
|
|
if data[:priority]
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .sidebar select[name="priority_id"]',
|
|
value: data[:priority],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:state]
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .sidebar select[name="state_id"]',
|
|
value: data[:state],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:files]
|
|
file_upload(
|
|
css: '.content.active .attachmentPlaceholder-inputHolder input',
|
|
files: data[:files],
|
|
)
|
|
end
|
|
|
|
params[:custom_data_select]&.each do |local_key, local_value|
|
|
select(
|
|
browser: instance,
|
|
css: ".active .sidebar select[name=\"#{local_key}\"]",
|
|
value: local_value,
|
|
)
|
|
end
|
|
params[:custom_data_input]&.each do |local_key, local_value|
|
|
set(
|
|
browser: instance,
|
|
css: ".active .sidebar input[name=\"#{local_key}\"]",
|
|
value: local_value,
|
|
clear: true,
|
|
)
|
|
end
|
|
params[:custom_data_date]&.each do |local_key, local_value|
|
|
click(
|
|
browser: instance,
|
|
css: ".active .sidebar div[data-name=\"#{local_key}\"] input[data-item=\"date\"]",
|
|
mute_log: true,
|
|
)
|
|
# weird bug where you cannot "clear" for date/time input
|
|
# this is specific chrome problem, chrome bug report: https://bugs.chromium.org/p/chromedriver/issues/detail?id=1319#c2
|
|
# indirect issue: https://github.com/angular/protractor/issues/562#issuecomment-47745263
|
|
11.times do
|
|
sendkey(
|
|
value: :backspace,
|
|
)
|
|
end
|
|
set(
|
|
browser: instance,
|
|
css: ".active .sidebar div[data-name=\"#{local_key}\"] input[data-item=\"date\"]",
|
|
value: local_value,
|
|
)
|
|
end
|
|
|
|
if data[:state] || data[:group] || data[:body] || params[:custom_data_select].present? || params[:custom_data_input].present?
|
|
found = nil
|
|
9.times do
|
|
|
|
break if found
|
|
|
|
begin
|
|
text = instance.find_elements(css: '.content.active .js-reset')[0].text
|
|
if text.match?(%r{(Discard your unsaved changes.|Verwerfen der)})
|
|
found = true
|
|
end
|
|
rescue
|
|
# try again
|
|
end
|
|
sleep 1
|
|
end
|
|
if !found
|
|
screenshot(browser: instance, comment: 'ticket_update_discard_message_failed')
|
|
raise 'no discard message found'
|
|
end
|
|
end
|
|
|
|
# avoid accessing a stale element when accessing task type
|
|
sleep 1
|
|
task_type(
|
|
browser: instance,
|
|
type: params[:task_type] || 'stayOnTab',
|
|
)
|
|
|
|
if params[:do_not_submit]
|
|
assert(true, 'ticket updated without submit')
|
|
return true
|
|
end
|
|
|
|
instance.find_elements(css: '.content.active .js-submit')[0].click
|
|
await_empty_ajax_queue(params)
|
|
|
|
# do not stay on tab
|
|
if params[:task_type] == 'closeTab' || params[:task_type] == 'closeNextInOverview'
|
|
sleep 1
|
|
return
|
|
end
|
|
|
|
9.times do
|
|
begin
|
|
text = instance.find_elements(css: '.content.active .js-reset')[0].text
|
|
if text.blank?
|
|
sleep 1
|
|
return true
|
|
end
|
|
rescue
|
|
# try again
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'ticket_update_failed')
|
|
raise 'unable to update ticket'
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket_verify(
|
|
browser: browser1,
|
|
data: {
|
|
title: 'some title',
|
|
body: 'some body',
|
|
## group: 'some group',
|
|
## state: 'closed',
|
|
custom_data_select: {
|
|
key1: 'some value',
|
|
},
|
|
custom_data_input: {
|
|
key1: 'some value',
|
|
},
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def ticket_verify(params)
|
|
switch_window_focus(params)
|
|
log('ticket_verify', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
if data[:title]
|
|
title = instance.find_elements(css: '.content.active .ticketZoom-header .js-objectTitle').first.text.strip
|
|
if title.match?(%r{#{data[:title]}}i)
|
|
assert(true, "matching '#{data[:title]}' in title '#{title}'")
|
|
else
|
|
raise "not matching '#{data[:title]}' in title '#{title}'"
|
|
end
|
|
end
|
|
|
|
if data[:body]
|
|
body = instance.find_elements(css: '.content.active [data-name="body"]').first.text.strip
|
|
if body.match?(%r{#{data[:body]}}i)
|
|
assert(true, "matching '#{data[:body]}' in body '#{body}'")
|
|
else
|
|
raise "not matching '#{data[:body]}' in body '#{body}'"
|
|
end
|
|
end
|
|
|
|
params[:custom_data_select]&.each do |local_key, local_value|
|
|
element = instance.find_elements(css: ".active .sidebar select[name=\"#{local_key}\"] option[selected]").first
|
|
value = element.text.strip
|
|
if value.match?(%r{#{local_value}}i)
|
|
assert(true, "matching '#{value}' in #{local_key} '#{local_value}'")
|
|
else
|
|
raise "not matching '#{value}' in #{local_key} '#{local_value}'"
|
|
end
|
|
end
|
|
params[:custom_data_input]&.each do |local_key, local_value|
|
|
element = instance.find_elements(css: ".active .sidebar input[name=\"#{local_key}\"]").first
|
|
value = element.text.strip
|
|
if value.match?(%r{#{local_value}}i)
|
|
assert(true, "matching '#{value}' in #{local_key} '#{local_value}'")
|
|
else
|
|
raise "not matching '#{value}' in #{local_key} '#{local_value}'"
|
|
end
|
|
end
|
|
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
overview_open(
|
|
browser: browser2,
|
|
name: overview_name,
|
|
)
|
|
|
|
overview_open(
|
|
browser: browser2,
|
|
link: "#ticket/view/some_special_name",
|
|
)
|
|
|
|
=end
|
|
|
|
def overview_open(params)
|
|
switch_window_focus(params)
|
|
log('overview_open', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
# click on overview task in sidebar
|
|
instance.find_elements(css: '.js-overviewsMenuItem')[0].click
|
|
|
|
# show larger overview selection list
|
|
sleep 0.5
|
|
execute(
|
|
browser: instance,
|
|
js: '$(".content.active .sidebar").css("display", "block")',
|
|
)
|
|
|
|
link = if params[:link]
|
|
params[:link]
|
|
elsif params[:name]
|
|
"\#ticket/view/#{params[:name]}"
|
|
end
|
|
|
|
# switch to overview
|
|
element = nil
|
|
6.times do
|
|
element = instance.find_elements(css: ".content.active .sidebar a[href=\"#{link}\"]")[0]
|
|
break if element
|
|
|
|
sleep 1
|
|
end
|
|
element.click
|
|
|
|
# hide larger overview selection list again
|
|
sleep 0.5
|
|
execute(
|
|
browser: instance,
|
|
js: '$(".content.active .sidebar").css("display", "none")',
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket_open_by_overview(
|
|
browser: browser2,
|
|
number: ticket1[:number],
|
|
link: "#ticket/view/#{name}",
|
|
)
|
|
|
|
ticket_open_by_overview(
|
|
browser: browser2,
|
|
number: ticket1[:number],
|
|
text: title,
|
|
link: "#ticket/view/#{name}",
|
|
)
|
|
|
|
=end
|
|
|
|
def ticket_open_by_overview(params)
|
|
switch_window_focus(params)
|
|
log('ticket_open_by_overview', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
overview_open(params)
|
|
|
|
element = nil
|
|
if params[:title]
|
|
6.times do
|
|
element = instance.find_element(css: '.content.active').find_element(partial_link_text: params[:title])
|
|
break if element
|
|
|
|
sleep 1
|
|
end
|
|
if !element
|
|
screenshot(browser: instance, comment: 'ticket_open_by_overview_no_ticket_failed')
|
|
raise "unable to find ticket #{params[:title]} in overview #{params[:link]}!"
|
|
end
|
|
else
|
|
6.times do
|
|
|
|
# prefere find_elements ofer find_element because of exception handling
|
|
element = instance.find_elements(partial_link_text: params[:number])[0]
|
|
break if element
|
|
|
|
sleep 1
|
|
end
|
|
if !element
|
|
screenshot(browser: instance, comment: 'ticket_open_by_overview_no_ticket_failed')
|
|
raise "unable to find ticket #{params[:number]} in overview #{params[:link]}!"
|
|
end
|
|
end
|
|
element.click
|
|
sleep 1
|
|
number = instance.find_element(css: '.content.active .ticketZoom-header .ticket-number').text
|
|
if !number.match?(%r{#{params[:number]}})
|
|
screenshot(browser: instance, comment: 'ticket_open_by_overview_open_failed_failed')
|
|
raise "unable to open ticket #{params[:number]}!"
|
|
end
|
|
assert(true, "ticket #{params[:number]} found")
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket_open_by_search(
|
|
browser: browser2,
|
|
number: ticket1[:number],
|
|
)
|
|
|
|
=end
|
|
|
|
def ticket_open_by_search(params)
|
|
switch_window_focus(params)
|
|
log('ticket_open_by_search', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
# search by number
|
|
element = instance.find_elements(css: '#global-search')[0]
|
|
element.click
|
|
element.clear
|
|
element.send_keys(params[:number])
|
|
sleep 3
|
|
|
|
# open ticket
|
|
# instance.find_element(partial_link_text: params[:number] } ).click
|
|
instance.execute_script("$(\".js-global-search-result a:contains('#{params[:number]}') .nav-tab-name\").first().click()")
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active .ticketZoom-header .ticket-number'
|
|
)
|
|
number = instance.find_elements(css: '.content.active .ticketZoom-header .ticket-number')[0].text
|
|
if !number.match?(%r{#{params[:number]}})
|
|
screenshot(browser: instance, comment: 'ticket_open_by_search_failed')
|
|
raise "unable to search/find ticket #{params[:number]}!"
|
|
end
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
ticket_open_by_title(
|
|
browser: browser2,
|
|
title: ticket1[:title],
|
|
)
|
|
|
|
=end
|
|
|
|
def ticket_open_by_title(params)
|
|
switch_window_focus(params)
|
|
log('ticket_open_by_title', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
# search by number
|
|
element = instance.find_elements(css: '#global-search')[0]
|
|
element.click
|
|
element.clear
|
|
element.send_keys(params[:title])
|
|
sleep 3
|
|
|
|
# open ticket
|
|
# instance.find_element(partial_link_text: params[:title] } ).click
|
|
instance.execute_script("$(\".js-global-search-result a:contains('#{params[:title]}') .nav-tab-name\").first().click()")
|
|
sleep 1
|
|
title = instance.find_elements(css: '.content.active .ticketZoom-header .js-objectTitle')[0].text
|
|
if !title.match?(%r{#{params[:title]}})
|
|
screenshot(browser: instance, comment: 'ticket_open_by_title_failed')
|
|
raise "unable to search/find ticket #{params[:title]}!"
|
|
end
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
overview_count = overview_counter(
|
|
browser: browser2,
|
|
)
|
|
|
|
returns
|
|
{
|
|
'#ticket/view/all_unassigned' => 42,
|
|
}
|
|
|
|
=end
|
|
|
|
def overview_counter(params = {})
|
|
switch_window_focus(params)
|
|
log('overview_counter', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
instance.find_elements(css: '.js-overviewsMenuItem')[0].click
|
|
await_empty_ajax_queue(params)
|
|
|
|
execute(
|
|
browser: instance,
|
|
js: '$(".content.active .sidebar").css("display", "block")',
|
|
)
|
|
# execute(
|
|
# browser: instance,
|
|
# js: '$(".content.active .overview-header").css("display", "none")',
|
|
# )
|
|
|
|
begin
|
|
overviews = {}
|
|
instance.find_elements(css: '.content.active .sidebar a[href]').each do |element|
|
|
url = element.attribute('href')
|
|
url.gsub!(%r{(http|https)://.+?/(.+?)$}, '\\2')
|
|
overviews[url] = 0
|
|
# puts url.inspect
|
|
# puts element.inspect
|
|
end
|
|
|
|
overviews.each_key do |url|
|
|
count = instance.find_elements(css: ".content.active .sidebar a[href=\"#{url}\"] .badge")[0].text
|
|
overviews[url] = count.to_i
|
|
end
|
|
rescue => e
|
|
retries ||= 0
|
|
retries += 1
|
|
sleep 0.5
|
|
retry if retries < 5
|
|
raise e
|
|
end
|
|
|
|
log('overview_counter', overviews)
|
|
overviews
|
|
end
|
|
|
|
=begin
|
|
|
|
organization_open_by_search(
|
|
browser: browser2,
|
|
value: 'some value',
|
|
)
|
|
|
|
=end
|
|
|
|
def organization_open_by_search(params = {})
|
|
switch_window_focus(params)
|
|
log('organization_open_by_search', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
element = instance.find_elements(css: '#global-search')[0]
|
|
|
|
element.click
|
|
element.clear
|
|
element.send_keys(params[:value])
|
|
sleep 3
|
|
|
|
empty_search(browser: instance)
|
|
|
|
element = instance.find_elements(css: '#global-search')[0]
|
|
element.click
|
|
element.clear
|
|
element.send_keys(params[:value])
|
|
sleep 2
|
|
|
|
watch_for_disappear(
|
|
browser: instance,
|
|
css: '.navigation .search.loading'
|
|
)
|
|
|
|
# instance.find_element(partial_link_text: params[:value] } ).click
|
|
instance.execute_script("$(\".js-global-search-result a:contains('#{params[:value]}') .nav-tab-name\").first().click()")
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active h1'
|
|
)
|
|
name = instance.find_elements(css: '.content.active h1')[0].text
|
|
if !name.match?(%r{#{params[:value]}})
|
|
screenshot(browser: instance, comment: 'organization_open_by_search_failed')
|
|
raise "unable to search/find org #{params[:value]}!"
|
|
end
|
|
assert(true, "org #{params[:value]} found")
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
user_open_by_search(
|
|
browser: browser2,
|
|
value: 'some value',
|
|
)
|
|
|
|
=end
|
|
|
|
def user_open_by_search(params = {})
|
|
switch_window_focus(params)
|
|
log('user_open_by_search', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
element = instance.find_elements(css: '#global-search')[0]
|
|
element.click
|
|
element.clear
|
|
element.send_keys(params[:value])
|
|
sleep 3
|
|
|
|
# instance.find_element(partial_link_text: params[:value]).click
|
|
instance.execute_script("$(\".js-global-search-result a:contains('#{params[:value]}') .nav-tab-name\").first().click()")
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active h1'
|
|
)
|
|
name = instance.find_elements(css: '.content.active h1')[0].text
|
|
if !name.match?(%r{#{params[:value]}})
|
|
screenshot(browser: instance, comment: 'user_open_by_search_failed')
|
|
raise "unable to search/find user #{params[:value]}!"
|
|
end
|
|
assert(true, "user #{params[:term]} found")
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
user_create(
|
|
browser: browser2,
|
|
data: {
|
|
#login: 'some login' + random,
|
|
firstname: 'Manage Firstname' + random,
|
|
lastname: 'Manage Lastname' + random,
|
|
email: user_email,
|
|
password: 'some-pass',
|
|
role: 'Admin', # optional, choose among [Admin, Agent, Customer]
|
|
# defaults to Customer if not provided
|
|
},
|
|
)
|
|
|
|
user_create(
|
|
browser: browser2,
|
|
data: {
|
|
#login: 'some login' + random,
|
|
firstname: 'Manage Firstname' + random,
|
|
lastname: 'Manage Lastname' + random,
|
|
email: user_email,
|
|
password: 'some-pass',
|
|
role: 'Agent', # when the role is Agent an array of permissions for each group is optionally accepted
|
|
permissions: { 1 => %w[read create overview],
|
|
2 => ['full'], }
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def user_create(params = {})
|
|
switch_window_focus(params)
|
|
log('user_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
raise 'user_create() requires either email or phone' if data[:email].blank? && data[:phone].blank?
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/users"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.modal input[name=firstname]')[0]
|
|
element.clear
|
|
element.send_keys(data[:firstname])
|
|
element = instance.find_elements(css: '.modal input[name=lastname]')[0]
|
|
element.clear
|
|
element.send_keys(data[:lastname])
|
|
element = instance.find_elements(css: '.modal input[name=email]')[0]
|
|
element.clear
|
|
element.send_keys(data[:email])
|
|
element = instance.find_elements(css: '.modal input[name=password]')[0]
|
|
element.clear
|
|
element.send_keys(data[:password])
|
|
element = instance.find_elements(css: '.modal input[name=password_confirm]')[0]
|
|
element.clear
|
|
element.send_keys(data[:password])
|
|
element = instance.find_elements(css: '.modal input[name=phone]')[0]
|
|
element.clear
|
|
element.send_keys(data[:phone])
|
|
|
|
if data[:active] == false
|
|
select(css: 'select[name="active"]', value: 'inactive')
|
|
end
|
|
|
|
if data[:organization]
|
|
|
|
begin
|
|
target = nil
|
|
retries ||= 0
|
|
|
|
5.times do
|
|
element = instance.find_elements(css: '.modal input.searchableSelect-main')[0]
|
|
element.clear
|
|
element.send_keys(data[:organization])
|
|
|
|
10.times do
|
|
sleep 0.5
|
|
target = instance.find_elements(css: ".modal li[title='#{data[:organization]}']")[0]
|
|
break if target
|
|
end
|
|
break if target
|
|
end
|
|
raise "Can't find organization #{data[:organization]}" if target.blank?
|
|
|
|
target.click
|
|
rescue Selenium::WebDriver::Error::StaleElementReferenceError
|
|
sleep retries
|
|
retries += 1
|
|
retry if retries < 3
|
|
end
|
|
end
|
|
|
|
if data[:role]
|
|
case data[:role]
|
|
when 'Admin'
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=1]',
|
|
)
|
|
when 'Customer'
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=3]',
|
|
)
|
|
when 'Agent'
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=2]',
|
|
)
|
|
data[:permissions].each do |key, value|
|
|
value.each do |permission|
|
|
check(
|
|
browser: instance,
|
|
css: ".modal input[name=\"group_ids::#{key}\"][value=\"#{permission}\"]",
|
|
)
|
|
end
|
|
end
|
|
else
|
|
raise "Unknown :role \"#{data[:role]}\" in user_create()"
|
|
end
|
|
else
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=3]',
|
|
)
|
|
end
|
|
|
|
click(
|
|
browser: instance,
|
|
css: '.modal .js-submit',
|
|
)
|
|
|
|
modal_disappear(
|
|
browser: instance,
|
|
timeout: 10,
|
|
)
|
|
|
|
if data[:email]
|
|
search_query = data[:email]
|
|
search_target = data[:email]
|
|
search_css = '.content.active .user-list .js-tableBody td:first-child'
|
|
else
|
|
search_query = data[:phone]
|
|
search_target = data[:firstname]
|
|
search_css = '.content.active .user-list .js-tableBody td:nth-child(2)'
|
|
end
|
|
|
|
60.times do |i|
|
|
if (i % 10).zero?
|
|
set(
|
|
browser: instance,
|
|
css: '.content.active .js-search',
|
|
value: search_query,
|
|
)
|
|
end
|
|
sleep 1
|
|
search_result = instance.find_elements(css: search_css).map(&:text).map(&:strip)
|
|
break if search_result.include? search_target
|
|
raise 'user creation failed' if i >= 19
|
|
|
|
log "new user #{search_query} not found on the #{i.ordinalize} try, retrying"
|
|
end
|
|
|
|
assert(true, 'user created')
|
|
end
|
|
|
|
=begin
|
|
|
|
user_edit(
|
|
browser: browser2,
|
|
data: {
|
|
login: 'some login' + random,
|
|
firstname: 'Manage Firstname' + random,
|
|
lastname: 'Manage Lastname' + random,
|
|
email: user_email,
|
|
password: 'some-pass',
|
|
role: 'Agent', # when the role is Agent an array of permissions for each group is optionally accepted
|
|
permissions: { 1 => %w[read create overview],
|
|
2 => ['full'], }
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def user_edit(params = {})
|
|
switch_window_focus(params)
|
|
log('user_edit', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/users"]',
|
|
mute_log: true,
|
|
)
|
|
instance.find_elements(css: '.content.active .user-list td:first-child').each do |element|
|
|
next if element.text.strip != data[:login]
|
|
|
|
element.click
|
|
break
|
|
end
|
|
modal_ready(browser: instance)
|
|
if data[:firstname]
|
|
element = instance.find_elements(css: '.modal input[name=firstname]')[0]
|
|
element.clear
|
|
element.send_keys(data[:firstname])
|
|
end
|
|
if data[:lastname]
|
|
element = instance.find_elements(css: '.modal input[name=lastname]')[0]
|
|
element.clear
|
|
element.send_keys(data[:lastname])
|
|
end
|
|
if data[:email]
|
|
element = instance.find_elements(css: '.modal input[name=email]')[0]
|
|
element.clear
|
|
element.send_keys(data[:email])
|
|
end
|
|
if data[:password]
|
|
element = instance.find_elements(css: '.modal input[name=password]')[0]
|
|
element.clear
|
|
element.send_keys(data[:password])
|
|
element = instance.find_elements(css: '.modal input[name=password_confirm]')[0]
|
|
element.clear
|
|
element.send_keys(data[:password])
|
|
end
|
|
if data[:phone]
|
|
element = instance.find_elements(css: '.modal input[name=phone]')[0]
|
|
element.clear
|
|
element.send_keys(data[:phone])
|
|
end
|
|
if data[:active].present?
|
|
select(css: 'select[name="active"]', value: data[:active] ? 'active' : 'inactive')
|
|
end
|
|
|
|
if data[:organization]
|
|
element = instance.find_elements(css: '.modal input.searchableSelect-main')[0]
|
|
element.clear
|
|
element.send_keys(data[:organization])
|
|
|
|
begin
|
|
retries ||= 0
|
|
target = nil
|
|
until target
|
|
sleep 0.5
|
|
target = instance.find_elements(css: ".modal li[title='#{data[:organization]}']")[0]
|
|
end
|
|
target.click
|
|
rescue Selenium::WebDriver::Error::StaleElementReferenceError
|
|
sleep retries
|
|
retries += 1
|
|
retry if retries < 3
|
|
end
|
|
end
|
|
|
|
if data[:role]
|
|
case data[:role]
|
|
when 'Admin'
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=1]',
|
|
)
|
|
when 'Customer'
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=3]',
|
|
)
|
|
when 'Agent'
|
|
check(
|
|
browser: instance,
|
|
css: '.modal input[name=role_ids][value=2]',
|
|
)
|
|
else
|
|
raise "Unknown :role \"#{data[:role]}\" in user_create()"
|
|
end
|
|
end
|
|
|
|
if data[:permissions].present?
|
|
data[:permissions].each do |key, value|
|
|
value.each do |permission|
|
|
check(
|
|
browser: instance,
|
|
css: ".modal input[name=\"group_ids::#{key}\"][value=\"#{permission}\"]",
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
click(
|
|
browser: instance,
|
|
css: '.modal .js-submit',
|
|
)
|
|
|
|
modal_disappear(
|
|
browser: instance,
|
|
timeout: 10,
|
|
)
|
|
|
|
assert(true, 'user updated')
|
|
end
|
|
|
|
=begin
|
|
|
|
organization_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'Test Organization',
|
|
}
|
|
)
|
|
|
|
=end
|
|
|
|
def organization_create(params = {})
|
|
switch_window_focus(params)
|
|
log('organization_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/organizations"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.modal input[name=name]')[0]
|
|
element.clear
|
|
element.send_keys(data[:name])
|
|
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
await_empty_ajax_queue(params)
|
|
modal_disappear(
|
|
browser: instance,
|
|
timeout: 5,
|
|
)
|
|
watch_for(
|
|
browser: instance,
|
|
css: 'body',
|
|
value: data[:name],
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
calendar_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some calendar' + random,
|
|
first_response_time_in_text: 61
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def calendar_create(params = {})
|
|
switch_window_focus(params)
|
|
log('calendar_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/calendars"]',
|
|
mute_log: true,
|
|
)
|
|
sleep 4
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a.js-new',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.content.active .modal input[name=name]')[0]
|
|
element.clear
|
|
element.send_keys(data[:name])
|
|
element = instance.find_elements(css: '.content.active .modal .js-input')[0]
|
|
element.clear
|
|
element.send_keys(data[:timezone])
|
|
element.send_keys(:enter)
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
7.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'calendar created')
|
|
sleep 1
|
|
return true
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'calendar_create_failed')
|
|
raise 'calendar creation failed'
|
|
end
|
|
|
|
=begin
|
|
|
|
sla_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some sla' + random,
|
|
calendar: 'some calendar name',
|
|
first_response_time_in_text: 61
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def sla_create(params = {})
|
|
switch_window_focus(params)
|
|
log('sla_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/slas"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a.js-new',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.modal input[name=name]')[0]
|
|
element.clear
|
|
element.send_keys(data[:name])
|
|
if data[:calendar].present?
|
|
element = instance.find_elements(css: '.modal select[name="calendar_id"]')[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
dropdown.select_by(:text, data[:calendar])
|
|
end
|
|
element = instance.find_elements(css: '.modal input[name=first_response_time_in_text]')[0]
|
|
element.clear
|
|
element.send_keys(data[:first_response_time_in_text])
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
await_empty_ajax_queue(params)
|
|
modal_disappear(browser: instance)
|
|
7.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'sla created')
|
|
sleep 1
|
|
return true
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'sla_create_failed')
|
|
raise 'sla creation failed'
|
|
end
|
|
|
|
=begin
|
|
|
|
text_module_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some sla' + random,
|
|
keywords: 'some keywords',
|
|
content: 'some content',
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def text_module_create(params = {})
|
|
switch_window_focus(params)
|
|
log('text_module_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/text_modules"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=name]',
|
|
value: data[:name],
|
|
)
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=keywords]',
|
|
value: data[:keywords],
|
|
)
|
|
set(
|
|
browser: instance,
|
|
css: '.modal [data-name=content]',
|
|
value: data[:content],
|
|
)
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
7.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'text module created')
|
|
sleep 1
|
|
return true
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'text_module_create_failed')
|
|
raise 'text module creation failed'
|
|
end
|
|
|
|
=begin
|
|
|
|
signature_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some sla' + random,
|
|
body: 'some body',
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def signature_create(params = {})
|
|
switch_window_focus(params)
|
|
log('signature_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#channels/email"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#c-signature"]',
|
|
mute_log: true,
|
|
)
|
|
sleep 4
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active #c-signature a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=name]',
|
|
value: data[:name],
|
|
)
|
|
set(
|
|
browser: instance,
|
|
css: '.modal [data-name=body]',
|
|
value: data[:body],
|
|
)
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
11.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'signature created')
|
|
sleep 1
|
|
return true
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: 'signature_create_failed')
|
|
raise 'signature creation failed'
|
|
end
|
|
|
|
=begin
|
|
|
|
group_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some sla' + random,
|
|
signature: 'some signature bame',
|
|
member: [
|
|
{
|
|
login: 'some_user_login',
|
|
access: 'all',
|
|
},
|
|
],
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def group_create(params = {})
|
|
switch_window_focus(params)
|
|
log('group_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/groups"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.modal input[name=name]')[0]
|
|
element.clear
|
|
element.send_keys(data[:name])
|
|
element = instance.find_elements(css: '.modal select[name="email_address_id"]')[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
dropdown.select_by(:index, 1)
|
|
# dropdown.select_by(:text, action[:group])
|
|
if data[:signature]
|
|
element = instance.find_elements(css: '.modal select[name="signature_id"]')[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
dropdown.select_by(:text, data[:signature])
|
|
end
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
await_empty_ajax_queue(params)
|
|
modal_disappear(browser: instance)
|
|
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'group created')
|
|
modal_disappear(browser: instance) # wait until modal has gone
|
|
|
|
# add member
|
|
data[:member]&.each do |member|
|
|
instance.find_elements(css: 'a[href="#manage"]')[0].click
|
|
sleep 1
|
|
scroll_to(params.merge(css: '.content.active a[href="#manage/users"]'))
|
|
instance.find_elements(css: '.content.active a[href="#manage/users"]')[0].click
|
|
sleep 3
|
|
element = instance.find_elements(css: '.content.active [name="search"]')[0]
|
|
element.clear
|
|
element.send_keys(member[:login])
|
|
sleep 3
|
|
# instance.find_elements(:css => '.content.active table [data-id]')[0].click
|
|
instance.execute_script('$(".content.active table [data-id] td").first().click()')
|
|
modal_ready(browser: instance)
|
|
# instance.find_elements(:css => 'label:contains(" ' + action[:name] + '")')[0].click
|
|
instance.execute_script(%($(".js-groupList tr:contains(\\"#{data[:name]}\\") .js-groupListItem[value=#{member[:access]}]").prop("checked", true)))
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
await_empty_ajax_queue(params)
|
|
modal_disappear(browser: instance)
|
|
end
|
|
end
|
|
|
|
sleep 1
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
macro_create(
|
|
browser: browser1,
|
|
name: 'Emmanuel Macro',
|
|
ux_flow_next_up: 'Stay on tab', # possible: 'Stay on tab', 'Close tab', 'Advance to next ticket from overview'
|
|
actions: {
|
|
'Tags' => { # currently only 'Tags' is supported
|
|
operator: 'add',
|
|
value: 'spam',
|
|
}
|
|
}
|
|
)
|
|
|
|
=end
|
|
|
|
def macro_create(params)
|
|
switch_window_focus(params)
|
|
log('macro_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
|
|
click(
|
|
browser: instance,
|
|
css: '.sidebar a[href="#manage/macros"]',
|
|
mute_log: true,
|
|
)
|
|
|
|
click(
|
|
browser: instance,
|
|
css: '.page-header-meta > a[data-type="new"]'
|
|
)
|
|
|
|
sendkey(
|
|
browser: instance,
|
|
css: '.modal-body input[name="name"]',
|
|
value: params[:name]
|
|
)
|
|
|
|
params[:actions]&.each do |attribute, changes|
|
|
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_perform_action .js-filterElement .js-attributeSelector select',
|
|
value: attribute,
|
|
mute_log: true,
|
|
)
|
|
|
|
next if attribute != 'Tags'
|
|
|
|
select(
|
|
browser: instance,
|
|
css: '.modal .ticket_perform_action .js-filterElement .js-operator select',
|
|
value: changes[:operator],
|
|
mute_log: true,
|
|
)
|
|
|
|
sendkey(
|
|
browser: instance,
|
|
css: '.modal .ticket_perform_action .js-filterElement .js-value .token-input',
|
|
value: changes[:value],
|
|
mute_log: true,
|
|
)
|
|
sendkey(
|
|
browser: instance,
|
|
value: :enter,
|
|
)
|
|
end
|
|
|
|
select(
|
|
browser: instance,
|
|
css: '.modal-body select[name="ux_flow_next_up"]',
|
|
value: params[:ux_flow_next_up]
|
|
)
|
|
|
|
click(
|
|
browser: instance,
|
|
css: '.modal-footer button[type="submit"]'
|
|
)
|
|
|
|
watch_for(
|
|
browser: instance,
|
|
css: 'body',
|
|
value: params[:name],
|
|
)
|
|
|
|
assert(true, 'macro created')
|
|
end
|
|
|
|
=begin
|
|
|
|
role_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some role' + random,
|
|
default_at_signup: false,
|
|
permission: {
|
|
'admin.group' => true,
|
|
'preferences.password' => true,
|
|
},
|
|
member: [
|
|
'some_user_login',
|
|
],
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def role_create(params = {})
|
|
switch_window_focus(params)
|
|
log('role_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/roles"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.modal input[name=name]')[0]
|
|
element.clear
|
|
element.send_keys(data[:name])
|
|
|
|
if data.key?(:default_at_signup)
|
|
element = instance.find_elements(css: '.modal select[name="default_at_signup"]')[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
if data[:default_at_signup] == true
|
|
dropdown.select_by(:text, 'yes')
|
|
else
|
|
dropdown.select_by(:text, 'no')
|
|
end
|
|
end
|
|
|
|
if data.key?(:permission)
|
|
data[:permission].each do |permission_name, permission_value|
|
|
if permission_value == false
|
|
uncheck(
|
|
browser: instance,
|
|
css: ".modal [data-permission-name=\"#{permission_name}\"]",
|
|
)
|
|
else
|
|
check(
|
|
browser: instance,
|
|
css: ".modal [data-permission-name=\"#{permission_name}\"]",
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
if data[:active] == false
|
|
select(css: 'select[name="active"]', value: 'inactive')
|
|
end
|
|
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'role created')
|
|
modal_disappear(browser: instance) # wait until modal has gone
|
|
|
|
# add member
|
|
data[:member]&.each do |login|
|
|
instance.find_elements(css: 'a[href="#manage"]')[0].click
|
|
sleep 1
|
|
instance.find_elements(css: '.content.active a[href="#manage/users"]')[0].click
|
|
sleep 3
|
|
element = instance.find_elements(css: '.content.active [name="search"]')[0]
|
|
element.clear
|
|
element.send_keys(login)
|
|
sleep 3
|
|
# instance.find_elements(:css => '.content.active table [data-id]')[0].click
|
|
instance.execute_script('$(".content.active table [data-id] td").first().click()')
|
|
sleep 3
|
|
# instance.find_elements(:css => 'label:contains(" ' + action[:name] + '")')[0].click
|
|
instance.execute_script(%($('label:contains(" #{data[:name]}")').first().click()))
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
end
|
|
end
|
|
|
|
sleep 1
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
role_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some role' + random,
|
|
default_at_signup: false,
|
|
permission: {
|
|
'admin.group' => true,
|
|
'preferences.password' => true,
|
|
},
|
|
member: [
|
|
'some_user_login',
|
|
],
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def role_edit(params = {})
|
|
switch_window_focus(params)
|
|
log('role_edit', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/roles"]',
|
|
mute_log: true,
|
|
)
|
|
|
|
await_text(container: '.content.active table tr td', text: data[:name])
|
|
instance.execute_script(%($('.content.active table tr td:contains(" #{data[:name]}")').first().click()))
|
|
|
|
modal_ready(browser: instance)
|
|
element = instance.find_elements(css: '.modal input[name=name]')[0]
|
|
element.clear
|
|
element.send_keys(data[:name])
|
|
|
|
if data.key?(:default_at_signup)
|
|
element = instance.find_elements(css: '.modal select[name="default_at_signup"]')[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
if data[:default_at_signup] == true
|
|
dropdown.select_by(:text, 'yes')
|
|
else
|
|
dropdown.select_by(:text, 'no')
|
|
end
|
|
end
|
|
|
|
if data.key?(:permission)
|
|
data[:permission].each do |permission_name, permission_value|
|
|
if permission_value == false
|
|
uncheck(
|
|
browser: instance,
|
|
css: ".modal [data-permission-name=\"#{permission_name}\"]",
|
|
)
|
|
else
|
|
check(
|
|
browser: instance,
|
|
css: ".modal [data-permission-name=\"#{permission_name}\"]",
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
if data.key?(:group_permissions)
|
|
data[:group_permissions].each do |key, value|
|
|
value.each do |permission|
|
|
check(
|
|
browser: instance,
|
|
css: ".modal input[name=\"group_ids::#{key}\"][value=\"#{permission}\"]",
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
if data.key?(:active)
|
|
element = instance.find_elements(css: '.modal select[name="active"]')[0]
|
|
dropdown = Selenium::WebDriver::Support::Select.new(element)
|
|
if data[:active] == true
|
|
dropdown.select_by(:text, 'active')
|
|
else
|
|
dropdown.select_by(:text, 'inactive')
|
|
end
|
|
end
|
|
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'role created')
|
|
modal_disappear(browser: instance) # wait until modal has gone
|
|
|
|
# add member
|
|
data[:member]&.each do |login|
|
|
instance.find_elements(css: 'a[href="#manage"]')[0].click
|
|
sleep 1
|
|
instance.find_elements(css: '.content.active a[href="#manage/users"]')[0].click
|
|
sleep 3
|
|
element = instance.find_elements(css: '.content.active [name="search"]')[0]
|
|
element.clear
|
|
element.send_keys(login)
|
|
sleep 3
|
|
# instance.find_elements(:css => '.content.active table [data-id]')[0].click
|
|
instance.execute_script('$(".content.active table [data-id] td").first().click()')
|
|
sleep 3
|
|
# instance.find_elements(:css => 'label:contains(" ' + action[:name] + '")')[0].click
|
|
instance.execute_script(%($('label:contains(" #{data[:name]}")').first().click()))
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
modal_disappear(browser: instance)
|
|
end
|
|
end
|
|
|
|
sleep 1
|
|
true
|
|
end
|
|
|
|
=begin
|
|
|
|
report_profile_create(
|
|
browser: browser2,
|
|
data: {
|
|
name: 'some profile' + random,
|
|
active: true
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def report_profile_create(params = {})
|
|
switch_window_focus(params)
|
|
log('report_profile_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#manage/report_profiles"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a.btn.primary[data-type="new"]',
|
|
mute_log: true,
|
|
)
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=name]',
|
|
value: data[:name],
|
|
mute_log: true,
|
|
)
|
|
if data[:active] == false
|
|
select(css: '.content.active .modal select[name="active"]', value: 'inactive')
|
|
end
|
|
sleep 0.5
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active .modal .js-submit',
|
|
mute_log: true,
|
|
)
|
|
modal_disappear
|
|
end
|
|
|
|
=begin
|
|
|
|
object_manager_attribute_create(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Select',
|
|
data_option: {
|
|
options: {
|
|
'aa' => 'AA',
|
|
'bb' => 'BB',
|
|
},
|
|
|
|
default: 'abc',
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
object_manager_attribute_create(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Text',
|
|
data_option: {
|
|
default: 'abc',
|
|
maxlength: 20,
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
object_manager_attribute_create(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Integer',
|
|
data_option: {
|
|
default: '15',
|
|
min: 1,
|
|
max: 999999,
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
object_manager_attribute_create(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Datetime',
|
|
data_option: {
|
|
future: true,
|
|
past: true,
|
|
diff: 24,
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
object_manager_attribute_create(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Date',
|
|
data_option: {
|
|
future: true,
|
|
past: true,
|
|
diff: 24,
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
object_manager_attribute_create(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Boolean',
|
|
data_option: {
|
|
options: {
|
|
true: 'YES',
|
|
false: 'NO',
|
|
}
|
|
default: undefined,
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
=end
|
|
|
|
def object_manager_attribute_create(params = {})
|
|
switch_window_focus(params)
|
|
log('object_manager_attribute_create', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
data[:object] = data[:object] || 'Ticket'
|
|
raise 'invalid object parameter in object_manager_attribute_create' if %w[Ticket User Organization Group].exclude? data[:object]
|
|
|
|
# make sure that required params are supplied
|
|
%i[name display data_type].each do |s|
|
|
next if data.key? s
|
|
|
|
raise "missing required param #{s} in object_manager_attribute_create()"
|
|
end
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#system/object_manager"]',
|
|
mute_log: true,
|
|
)
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active .js-new',
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: ".content.active a[href='#c-#{data[:object]}']",
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: ".content.active #c-#{data[:object]} .js-new",
|
|
mute_log: true,
|
|
)
|
|
|
|
object_manager_attribute_perform('create', params)
|
|
end
|
|
|
|
=begin
|
|
|
|
object_manager_attribute_update(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
display: 'Display Name of Field',
|
|
data_type: 'Select',
|
|
data_option: {
|
|
options: {
|
|
'aa' => 'AA',
|
|
'bb' => 'BB',
|
|
},
|
|
|
|
default: 'abc',
|
|
},
|
|
},
|
|
error: 'already exists'
|
|
)
|
|
|
|
=end
|
|
|
|
def object_manager_attribute_update(params = {})
|
|
switch_window_focus(params)
|
|
log('object_manager_attribute_update', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
data[:object] = data[:object] || 'Ticket'
|
|
raise 'invalid object parameter in object_manager_attribute_update' if %w[Ticket User Organization Group].exclude? data[:object]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#system/object_manager"]',
|
|
mute_log: true,
|
|
)
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active .js-new',
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: ".content.active a[href='#c-#{data[:object]}']",
|
|
mute_log: true,
|
|
)
|
|
instance.execute_script("$(\".content.active #c-#{data[:object]} td:contains('#{data[:name]}')\").first().click()")
|
|
|
|
object_manager_attribute_perform('update', params)
|
|
end
|
|
|
|
=begin
|
|
|
|
object_manager_attribute_delete(
|
|
browser: browser2,
|
|
data: {
|
|
object: 'Ticket', # optional, defaults to Ticket
|
|
name: 'field_name' + random,
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def object_manager_attribute_delete(params = {})
|
|
switch_window_focus(params)
|
|
log('object_manager_attribute_delete', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
data[:object] = data[:object] || 'Ticket'
|
|
raise 'invalid object parameter in object_manager_attribute_delete' if %w[Ticket User Organization Group].exclude? data[:object]
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#system/object_manager"]',
|
|
mute_log: true,
|
|
)
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active .js-new',
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: ".content.active a[href='#c-#{data[:object]}']",
|
|
mute_log: true,
|
|
)
|
|
sleep 4
|
|
|
|
instance.execute_script("$(\".content.active #c-#{data[:object]} td:contains('#{data[:name]}')\").first().closest('tr').find('.js-delete').click()")
|
|
end
|
|
|
|
=begin
|
|
|
|
object_manager_attribute_discard_changes(
|
|
browser: browser2,
|
|
)
|
|
|
|
=end
|
|
|
|
def object_manager_attribute_discard_changes(params = {})
|
|
switch_window_focus(params)
|
|
log('object_manager_attribute_discard_changes', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
click(
|
|
browser: instance,
|
|
css: 'a[href="#manage"]',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active a[href="#system/object_manager"]',
|
|
mute_log: true,
|
|
)
|
|
sleep 4
|
|
|
|
element = instance.find_elements(css: '.content.active .js-discard').first
|
|
element.click
|
|
|
|
watch_for_disappear(
|
|
browser: instance,
|
|
css: '.content.active .js-discard',
|
|
)
|
|
|
|
end
|
|
|
|
=begin
|
|
|
|
Execute any pending migrations in the object attribute manager
|
|
|
|
object_manager_attribute_migrate(
|
|
browser: browser2,
|
|
)
|
|
|
|
=end
|
|
|
|
def object_manager_attribute_migrate(params = {})
|
|
switch_window_focus(params)
|
|
log('object_manager_attribute_migrate', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active',
|
|
value: 'Database Update required',
|
|
mute_log: true,
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.content.active .tab-pane.active div.js-execute',
|
|
mute_log: true,
|
|
)
|
|
modal_ready(
|
|
browser: instance,
|
|
)
|
|
title_text = instance.find_elements(css: '.modal .modal-title').first.text
|
|
if ['Zammad is restarting...', 'Zammad need a restart!'].include?(title_text)
|
|
# in the complex case, wait for server to restart
|
|
modal_disappear(
|
|
browser: instance,
|
|
timeout: 7.minutes,
|
|
)
|
|
elsif title_text == 'Config has changed'
|
|
# in the simple case, just click the submit button
|
|
click(
|
|
browser: instance,
|
|
css: '.modal .js-submit',
|
|
mute_log: true,
|
|
)
|
|
else
|
|
raise "Unknown title text \"#{title_text}\" found when trying to update database"
|
|
end
|
|
sleep 5
|
|
watch_for(
|
|
browser: instance,
|
|
css: '.content.active',
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
tags_verify(
|
|
browser: browser2,
|
|
tags: {
|
|
'tag 1' => true,
|
|
'tag 2' => true,
|
|
'tag 3' => false,
|
|
},
|
|
)
|
|
|
|
=end
|
|
|
|
def tags_verify(params = {})
|
|
switch_window_focus(params)
|
|
log('tags_verify', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
tags = instance.find_elements({ css: '.content.active .js-tag' })
|
|
assert(tags)
|
|
assert(tags[0])
|
|
|
|
tags_found = {}
|
|
params[:tags].each_key do |key|
|
|
tags_found[key] = false
|
|
end
|
|
|
|
tags.each do |element|
|
|
text = element.text
|
|
if tags_found.key?(text)
|
|
tags_found[text] = true
|
|
else
|
|
assert(false, "tag exists but is not in check to verify '#{text}'")
|
|
end
|
|
end
|
|
params[:tags].each do |key, value|
|
|
assert_equal(value, tags_found[key], "tag '#{key}'")
|
|
end
|
|
end
|
|
|
|
def quote(string)
|
|
string_quoted = string
|
|
string_quoted.gsub!(%r{&}, '&')
|
|
string_quoted.gsub!(%r{<}, '<')
|
|
string_quoted.gsub!(%r{>}, '>')
|
|
string_quoted
|
|
end
|
|
|
|
def switch_window_focus(params)
|
|
instance = params[:browser] || @browser
|
|
if instance != @last_used_browser
|
|
log('switch browser window focus', {})
|
|
instance.switch_to.window(instance.window_handles.first)
|
|
end
|
|
@last_used_browser = instance
|
|
end
|
|
|
|
def log(method, params = {})
|
|
begin
|
|
instance = params[:browser] || @browser
|
|
if instance
|
|
logs = instance.manage.logs.get(:browser)
|
|
logs.each do |log|
|
|
next if log.level == 'WARNING' && log.message =~ %r{Declaration\sdropped.} # ignore ff css warnings
|
|
|
|
time = Time.zone.parse(Time.zone.at(log.timestamp / 1000).to_datetime.to_s)
|
|
puts "#{time}/#{log.level}: #{log.message}"
|
|
end
|
|
end
|
|
rescue
|
|
# failed to get logs
|
|
end
|
|
return if !DEBUG
|
|
return if params[:mute_log]
|
|
|
|
puts "#{Time.zone.now}/#{method}: #{params.inspect}"
|
|
end
|
|
|
|
private
|
|
|
|
def add_tree_options(instance:, options:)
|
|
|
|
# first level entries have to get added in regular order
|
|
options.each_key.with_index do |option, index|
|
|
|
|
if index != 0
|
|
element = instance.find_elements(css: '.modal .js-treeTable .js-addRow')[index - 1]
|
|
element.click
|
|
end
|
|
|
|
element = instance.find_elements(css: '.modal .js-treeTable .js-key')[index]
|
|
element.clear
|
|
element.send_keys(option)
|
|
end
|
|
|
|
add_sub_tree_recursion(
|
|
instance: instance,
|
|
options: options,
|
|
)
|
|
end
|
|
|
|
def add_sub_tree_recursion(instance:, options:, offset: 0)
|
|
options.each_value.inject(offset) do |child_offset, children|
|
|
|
|
child_offset += 1
|
|
|
|
# put your recursion glasses on 8-)
|
|
add_sub_tree_options(
|
|
instance: instance,
|
|
options: children,
|
|
offset: child_offset,
|
|
)
|
|
end
|
|
end
|
|
|
|
def add_sub_tree_options(instance:, options:, offset:)
|
|
|
|
# sub level entries have to get added in reversed order
|
|
level_options = options.to_a.reverse.to_h.keys
|
|
|
|
level_options.each do |option|
|
|
|
|
# sub level entries have to get added via 'add child row' link
|
|
click_index = offset - 1
|
|
|
|
element = instance.find_elements(css: '.modal .js-treeTable .js-addChild')[click_index]
|
|
element.click
|
|
|
|
element = instance.find_elements(css: '.modal .js-treeTable .js-key')[offset]
|
|
element.clear
|
|
element.send_keys(option)
|
|
sleep 0.25
|
|
end
|
|
|
|
add_sub_tree_recursion(
|
|
instance: instance,
|
|
options: options,
|
|
offset: offset,
|
|
)
|
|
end
|
|
|
|
def token_verify(css, value)
|
|
original_element = @browser.find_element(:css, css)
|
|
elem = original_element.find_element(xpath: '../input[contains(@class, "token-input")]')
|
|
elem.send_keys value
|
|
elem.send_keys :enter
|
|
|
|
watch_for(
|
|
xpath: '../*/span[contains(@class,"token-label")]',
|
|
value: value,
|
|
container: original_element
|
|
)
|
|
end
|
|
|
|
def toggle_checkbox(scope, value)
|
|
checkbox = scope.find_element(css: "input[value=#{value}]")
|
|
|
|
@browser
|
|
.action
|
|
.move_to(checkbox)
|
|
.click
|
|
.perform
|
|
end
|
|
|
|
def checkbox_is_selected(scope, value)
|
|
scope.find_element(css: "input[value=#{value}]").property('checked')
|
|
end
|
|
|
|
=begin
|
|
|
|
Switch the current logged in user's profile language to a new language
|
|
|
|
switch_language(
|
|
browser: browser2,
|
|
data: {
|
|
language: 'Deutsch'
|
|
},
|
|
)
|
|
|
|
IMPORTANT REMINDER! At the end of tests, the caller must manually set the language back to English again:
|
|
|
|
switch_language(
|
|
browser: browser2,
|
|
data: {
|
|
language: 'English (United States)'
|
|
},
|
|
)
|
|
|
|
Failure to switch back to English will cause large amounts of subsequent tests to fail due to the UI language differences.
|
|
|
|
=end
|
|
|
|
def switch_language(params = {})
|
|
switch_window_focus(params)
|
|
log('switch_language', params)
|
|
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
click(browser: instance, css: '#navigation .user-menu .js-avatar')
|
|
|
|
click(browser: instance, css: '#navigation .user-menu a[href="#profile"]')
|
|
|
|
select(
|
|
browser: instance,
|
|
css: '.content.active .searchableSelect-shadow',
|
|
value: data[:language],
|
|
)
|
|
|
|
click(browser: instance, css: '.content.active .btn--primary')
|
|
|
|
watch_for(
|
|
browser: instance,
|
|
css: '#notify',
|
|
)
|
|
end
|
|
|
|
=begin
|
|
|
|
Retrieve a hash of all the avaiable Zammad settings and their current values.
|
|
|
|
settings = fetch_settings()
|
|
|
|
=end
|
|
|
|
def fetch_settings
|
|
url = URI.parse(browser_url)
|
|
req = Net::HTTP::Get.new("#{browser_url}/api/v1/settings/")
|
|
req.basic_auth('admin@example.com', 'test')
|
|
|
|
res = Net::HTTP.start(url.host, url.port) do |http|
|
|
http.request(req)
|
|
end
|
|
raise "HTTP error #{res.code} while fetching #{browser_url}/api/v1/settings/" if res.code != '200'
|
|
|
|
JSON.parse(res.body)
|
|
end
|
|
|
|
=begin
|
|
|
|
Enable or disable Zammad experiemental features remotely.
|
|
|
|
set_setting('ui_ticket_zoom_attachments_preview', true)
|
|
|
|
=end
|
|
|
|
def set_setting(name, value)
|
|
name_to_id = fetch_settings.map { |s| [s['name'], s['id']] }.to_h
|
|
id = name_to_id[name]
|
|
|
|
url = URI.parse(browser_url)
|
|
req = Net::HTTP::Put.new("#{browser_url}/api/v1/settings/#{id}")
|
|
req['Content-Type'] = 'application/json'
|
|
req.basic_auth('admin@example.com', 'test')
|
|
req.body = { 'state_current' => { 'value' => value } }.to_json
|
|
res = Net::HTTP.start(url.host, url.port) do |http|
|
|
http.request(req)
|
|
end
|
|
raise "HTTP error #{res.code} while POSTing to #{browser_url}/api/v1/settings/" if res.code != '200'
|
|
end
|
|
|
|
=begin
|
|
|
|
Helper method for both object_manager_attribute_create and object_manager_attribute_update
|
|
|
|
=end
|
|
|
|
def object_manager_attribute_perform(action = 'create', params = {})
|
|
instance = params[:browser] || @browser
|
|
data = params[:data]
|
|
|
|
modal_ready(browser: instance)
|
|
|
|
if action == 'create'
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=name]',
|
|
value: data[:name],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:display]
|
|
set(
|
|
browser: instance,
|
|
css: '.modal input[name=display]',
|
|
value: data[:display],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:data_type]
|
|
select(
|
|
browser: instance,
|
|
css: '.modal select[name="data_type"]',
|
|
value: data[:data_type],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
if data[:data_option]
|
|
if data[:data_option][:options]
|
|
case data[:data_type]
|
|
when 'Boolean'
|
|
# rubocop:disable Lint/BooleanSymbol
|
|
element = instance.find_elements(css: '.modal .js-valueTrue').first
|
|
element.clear
|
|
element.send_keys(data[:data_option][:options][:true])
|
|
element = instance.find_elements(css: '.modal .js-valueFalse').first
|
|
element.clear
|
|
element.send_keys(data[:data_option][:options][:false])
|
|
# rubocop:enable Lint/BooleanSymbol
|
|
when 'Tree Select'
|
|
add_tree_options(
|
|
instance: instance,
|
|
options: data[:data_option][:options],
|
|
)
|
|
else
|
|
if action == 'update'
|
|
# first clear all existing entries
|
|
loop do
|
|
target = {
|
|
browser: instance,
|
|
css: '.modal .js-Table .js-remove',
|
|
mute_log: true,
|
|
}
|
|
break if !instance.find_elements(css: target[:css])[0]
|
|
|
|
click(target)
|
|
end
|
|
sleep 1
|
|
end
|
|
|
|
# then populate the table with the new values
|
|
data[:data_option][:options].each do |key, value|
|
|
element = instance.find_elements(css: '.modal .js-Table .js-key').last
|
|
element.clear
|
|
element.send_keys(key)
|
|
element = instance.find_elements(css: '.modal .js-Table .js-value').last
|
|
element.clear
|
|
element.send_keys(value)
|
|
element = instance.find_elements(css: '.modal .js-Table .js-add')[0]
|
|
element.click
|
|
end
|
|
end
|
|
end
|
|
|
|
%i[default min max diff].each do |key|
|
|
next if !data[:data_option].key?(key)
|
|
|
|
element = instance.find_elements(css: ".modal [name=\"data_option::#{key}\"]").first
|
|
element.clear
|
|
element.send_keys(data[:data_option][key])
|
|
end
|
|
|
|
%i[future past].each do |key|
|
|
next if !data[:data_option].key?(key)
|
|
|
|
select(
|
|
browser: instance,
|
|
css: ".modal select[name=\"data_option::#{key}\"]",
|
|
value: data[:data_option][key],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
|
|
%i[maxlength].each do |key|
|
|
next if !data[:data_option].key?(key)
|
|
|
|
set(
|
|
browser: instance,
|
|
css: ".modal input[name=\"data_option::#{key}\"]",
|
|
value: data[:data_option][key],
|
|
mute_log: true,
|
|
)
|
|
end
|
|
end
|
|
|
|
if params[:do_not_submit]
|
|
assert(true, "attribute #{action}d without submit")
|
|
return true
|
|
end
|
|
|
|
instance.find_elements(css: '.modal button.js-submit')[0].click
|
|
|
|
if params[:error]
|
|
sleep 4
|
|
watch_for(
|
|
css: '.modal',
|
|
value: params[:error],
|
|
)
|
|
click(
|
|
browser: instance,
|
|
css: '.modal .js-close',
|
|
)
|
|
modal_disappear(browser: instance)
|
|
return
|
|
end
|
|
|
|
modal_disappear(browser: instance)
|
|
|
|
11.times do
|
|
element = instance.find_elements(css: 'body')[0]
|
|
text = element.text
|
|
if text.match?(%r{#{Regexp.quote(data[:name])}})
|
|
assert(true, 'object manager attribute updated')
|
|
sleep 1
|
|
return true
|
|
end
|
|
sleep 1
|
|
end
|
|
screenshot(browser: instance, comment: "object_manager_attribute_#{action}_failed")
|
|
raise "object_manager_attribute_#{action}_failed"
|
|
end
|
|
|
|
def check_alert(params = {})
|
|
instance = params[:browser] || @browser
|
|
|
|
tries = 5
|
|
begin
|
|
alert = instance.switch_to.alert
|
|
alert.dismiss
|
|
rescue e
|
|
tries -= 1
|
|
sleep 0.5
|
|
retry if tries.positive?
|
|
raise e
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
This function waits for ajax requests and core workflow to be done
|
|
|
|
await_empty_ajax_queue
|
|
|
|
=end
|
|
|
|
def await_empty_ajax_queue(params = {})
|
|
return if params[:ajax] == false
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
10.times do
|
|
sleep 0.5
|
|
|
|
break if instance.execute_script('return typeof(App) === "undefined"')
|
|
break if instance.execute_script('return App.Ajax.queue().length').zero? && instance.execute_script('return Object.keys(App.FormHandlerCoreWorkflow.getRequests()).length').zero?
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
This function waits for a text to be ready in the dom. By default it searches in the active content.
|
|
|
|
await_text(text: 'New Ticket')
|
|
|
|
await_text(text: 'New Ticket', container: 'body')
|
|
|
|
=end
|
|
|
|
def await_text(params)
|
|
return if params[:ajax] == false
|
|
|
|
instance = params[:browser] || @browser
|
|
|
|
container = '.content.active'
|
|
if params[:container]
|
|
container = params[:container]
|
|
end
|
|
|
|
20.times do
|
|
log('await_text', params)
|
|
|
|
break if instance.execute_script("return $(\"#{container}:contains('#{params[:text]}')\").length").positive?
|
|
|
|
sleep 0.5
|
|
end
|
|
end
|
|
|
|
=begin
|
|
|
|
This function waits for the overview_counter to return a specific result.
|
|
|
|
await_overview_counter(view: '#ticket/view/all_unassigned', count: overview_counter_before['#ticket/view/all_unassigned'] - 2)
|
|
|
|
=end
|
|
|
|
def await_overview_counter(params)
|
|
result = nil
|
|
40.times do
|
|
result = overview_counter
|
|
|
|
if result[ params[:view] ] != params[:count]
|
|
sleep 0.5
|
|
next
|
|
end
|
|
|
|
break
|
|
end
|
|
|
|
assert_equal(params[:count], result[ params[:view] ])
|
|
end
|
|
|
|
=begin
|
|
|
|
This function waits for a search result to be available in the global search.
|
|
It can help to verify if a user is indexed in elastic search.
|
|
|
|
await_global_search(query: 'customer 1 firstname')
|
|
|
|
=end
|
|
|
|
def await_global_search(params)
|
|
instance = params[:browser] || @browser
|
|
|
|
30.times do
|
|
log('await_global_search', params)
|
|
|
|
set(
|
|
css: 'input#global-search',
|
|
value: params[:query],
|
|
)
|
|
|
|
break if instance.execute_script("return $(\"ul.global-search-result:visible:contains('#{params[:query]}')\").length") == 1
|
|
|
|
sleep 0.5
|
|
end
|
|
|
|
set(
|
|
css: 'input#global-search',
|
|
value: '',
|
|
)
|
|
end
|
|
end
|