browser_test_helper.rb 100 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864286528662867286828692870287128722873287428752876287728782879288028812882288328842885288628872888288928902891289228932894289528962897289828992900290129022903290429052906290729082909291029112912291329142915291629172918291929202921292229232924292529262927292829292930293129322933293429352936293729382939294029412942294329442945294629472948294929502951295229532954295529562957295829592960296129622963296429652966296729682969297029712972297329742975297629772978297929802981298229832984298529862987298829892990299129922993299429952996299729982999300030013002300330043005300630073008300930103011301230133014301530163017301830193020302130223023302430253026302730283029303030313032303330343035303630373038303930403041304230433044304530463047304830493050305130523053305430553056305730583059306030613062306330643065306630673068306930703071307230733074307530763077307830793080308130823083308430853086308730883089309030913092309330943095309630973098309931003101310231033104310531063107310831093110311131123113311431153116311731183119312031213122312331243125312631273128312931303131313231333134313531363137313831393140314131423143314431453146314731483149315031513152315331543155315631573158315931603161316231633164316531663167316831693170317131723173317431753176317731783179318031813182318331843185318631873188318931903191319231933194319531963197319831993200320132023203320432053206320732083209321032113212321332143215321632173218321932203221322232233224322532263227322832293230323132323233323432353236323732383239324032413242324332443245324632473248324932503251325232533254325532563257325832593260326132623263326432653266326732683269327032713272327332743275327632773278327932803281328232833284328532863287328832893290329132923293329432953296329732983299330033013302330333043305330633073308330933103311331233133314331533163317331833193320332133223323332433253326332733283329333033313332333333343335333633373338333933403341334233433344334533463347334833493350335133523353335433553356335733583359336033613362336333643365336633673368336933703371337233733374337533763377337833793380338133823383338433853386338733883389339033913392339333943395339633973398339934003401340234033404340534063407340834093410341134123413341434153416341734183419342034213422342334243425342634273428342934303431343234333434343534363437343834393440344134423443344434453446344734483449345034513452345334543455345634573458345934603461346234633464346534663467346834693470347134723473347434753476347734783479348034813482348334843485348634873488348934903491349234933494349534963497349834993500350135023503350435053506350735083509351035113512351335143515351635173518351935203521352235233524352535263527352835293530353135323533353435353536353735383539354035413542354335443545354635473548354935503551355235533554355535563557355835593560356135623563356435653566356735683569357035713572357335743575357635773578357935803581358235833584358535863587358835893590359135923593359435953596359735983599360036013602360336043605360636073608360936103611361236133614361536163617361836193620362136223623362436253626362736283629363036313632363336343635363636373638363936403641364236433644364536463647364836493650365136523653365436553656365736583659366036613662366336643665366636673668366936703671367236733674367536763677367836793680368136823683368436853686368736883689369036913692369336943695369636973698369937003701370237033704370537063707370837093710371137123713371437153716371737183719372037213722372337243725372637273728372937303731373237333734373537363737373837393740374137423743374437453746374737483749375037513752375337543755375637573758375937603761376237633764376537663767376837693770377137723773377437753776377737783779378037813782378337843785378637873788378937903791379237933794379537963797379837993800380138023803380438053806380738083809381038113812381338143815381638173818381938203821382238233824382538263827382838293830383138323833383438353836383738383839384038413842384338443845384638473848384938503851
  1. ENV['RAILS_ENV'] = 'test'
  2. # rubocop:disable HandleExceptions, NonLocalExitFromIterator, Style/GuardClause, Lint/MissingCopEnableDirective
  3. require File.expand_path('../config/environment', __dir__)
  4. require 'selenium-webdriver'
  5. class TestCase < Test::Unit::TestCase
  6. DEBUG = true
  7. def browser
  8. ENV['BROWSER'] || 'firefox'
  9. end
  10. def profile
  11. browser_profile = nil
  12. if browser == 'firefox'
  13. browser_profile = Selenium::WebDriver::Firefox::Profile.new
  14. browser_profile['intl.locale.matchOS'] = false
  15. browser_profile['intl.accept_languages'] = 'en-US'
  16. browser_profile['general.useragent.locale'] = 'en-US'
  17. # currently console log not working for firefox
  18. # https://github.com/SeleniumHQ/selenium/issues/1161
  19. #browser_profile['loggingPref'] = { browser: :all }
  20. elsif browser == 'chrome'
  21. # profile are only working on remote selenium
  22. if ENV['REMOTE_URL']
  23. browser_profile = Selenium::WebDriver::Chrome::Profile.new
  24. browser_profile['intl.accept_languages'] = 'en'
  25. browser_profile['loggingPref'] = { browser: :all }
  26. end
  27. end
  28. browser_profile
  29. end
  30. def browser_support_cookies
  31. if browser.match?(/(internet_explorer|ie)/i)
  32. return false
  33. end
  34. true
  35. end
  36. def browser_url
  37. ENV['BROWSER_URL'] || 'http://localhost:3000'
  38. end
  39. def browser_instance
  40. @browsers ||= {}
  41. if ENV['REMOTE_URL'].blank?
  42. local_browser = Selenium::WebDriver.for(browser.to_sym, profile: profile)
  43. @browsers[local_browser.hash] = local_browser
  44. browser_instance_preferences(local_browser)
  45. return local_browser
  46. end
  47. # avoid "Cannot read property 'get_Current' of undefined" issues
  48. (1..5).each do |count|
  49. begin
  50. local_browser = browser_instance_remote
  51. break
  52. rescue
  53. wait_until_ready = rand(5..13)
  54. sleep wait_until_ready
  55. log('browser_instance', { rescure: true, count: count, sleep: wait_until_ready })
  56. end
  57. end
  58. local_browser
  59. end
  60. def browser_instance_remote
  61. caps = Selenium::WebDriver::Remote::Capabilities.send(browser)
  62. if ENV['BROWSER_OS']
  63. caps.platform = ENV['BROWSER_OS']
  64. end
  65. if ENV['BROWSER_VERSION']
  66. caps.version = ENV['BROWSER_VERSION']
  67. end
  68. local_browser = Selenium::WebDriver.for(
  69. :remote,
  70. url: ENV['REMOTE_URL'],
  71. desired_capabilities: caps,
  72. )
  73. @browsers[local_browser.hash] = local_browser
  74. browser_instance_preferences(local_browser)
  75. # upload files from remote dir
  76. local_browser.file_detector = lambda do |args|
  77. str = args.first.to_s
  78. str if File.file?(str)
  79. end
  80. local_browser
  81. end
  82. def browser_instance_close(local_browser)
  83. return if !@browsers[local_browser.hash]
  84. @browsers.delete(local_browser.hash)
  85. local_browser.quit
  86. end
  87. def browser_instance_preferences(local_browser)
  88. browser_width = ENV['BROWSER_WIDTH'] || 1024
  89. browser_height = ENV['BROWSER_HEIGHT'] || 800
  90. local_browser.manage.window.resize_to(browser_width, browser_height)
  91. if !ENV['REMOTE_URL']&.match?(/saucelabs|(grid|ci)\.(zammad\.org|znuny\.com)/i)
  92. if @browsers.count == 1
  93. local_browser.manage.window.move_to(0, 0)
  94. else
  95. local_browser.manage.window.move_to(browser_width, 0)
  96. end
  97. end
  98. local_browser.manage.timeouts.implicit_wait = 3 # seconds
  99. end
  100. def teardown
  101. return if !@browsers
  102. @browsers.each_value do |local_browser|
  103. screenshot(browser: local_browser, comment: 'teardown')
  104. browser_instance_close(local_browser)
  105. end
  106. end
  107. def screenshot(params)
  108. instance = params[:browser] || @browser
  109. comment = params[:comment] || ''
  110. filename = "tmp/#{Time.zone.now.strftime('screenshot_%Y_%m_%d__%H_%M_%S_%L')}_#{comment}#{instance.hash}.png"
  111. log('screenshot', { filename: filename })
  112. instance.save_screenshot(filename)
  113. end
  114. =begin
  115. username = login(
  116. browser: browser1,
  117. username: 'someuser',
  118. password: 'somepassword',
  119. url: 'some url', # optional, in case of aleady opened brower a reload is done because url is called again
  120. remember_me: true, # optional
  121. auto_wizard: false, # optional, in case of auto wizard, skip login
  122. success: false, #optional
  123. )
  124. =end
  125. def login(params)
  126. switch_window_focus(params)
  127. log('login', params)
  128. instance = params[:browser] || @browser
  129. if params[:url]
  130. instance.get(params[:url])
  131. end
  132. # submit logs anyway
  133. instance.execute_script('App.Track.force()')
  134. element = instance.find_elements(css: '#login input[name="username"]')[0]
  135. if !element
  136. if params[:auto_wizard]
  137. watch_for(
  138. browser: instance,
  139. css: 'body',
  140. value: 'auto wizard is enabled',
  141. timeout: 10,
  142. )
  143. location(url: "#{browser_url}/#getting_started/auto_wizard")
  144. sleep 10
  145. login = instance.find_elements(css: '.user-menu .user a')[0].attribute('title')
  146. if login != params[:username]
  147. screenshot(browser: instance, comment: 'auto wizard login failed')
  148. raise 'auto wizard login failed'
  149. end
  150. assert(true, 'auto wizard login ok')
  151. clues_close(
  152. browser: instance,
  153. optional: true,
  154. )
  155. return
  156. end
  157. screenshot(browser: instance, comment: 'login_failed')
  158. raise 'No login box found'
  159. end
  160. screenshot(browser: instance, comment: 'login')
  161. element.clear
  162. element.send_keys(params[:username])
  163. element = instance.find_elements(css: '#login input[name="password"]')[0]
  164. element.clear
  165. element.send_keys(params[:password])
  166. if params[:remember_me]
  167. instance.find_elements(css: '#login .checkbox-replacement')[0].click
  168. end
  169. instance.find_elements(css: '#login button')[0].click
  170. sleep 4
  171. login_failed = false
  172. if !instance.find_elements(css: '.user-menu .user a')[0]
  173. login_failed = true
  174. else
  175. login = instance.find_elements(css: '.user-menu .user a')[0].attribute('title')
  176. if login != params[:username]
  177. login_failed = true
  178. end
  179. end
  180. if login_failed
  181. if params[:success] == false
  182. assert(true, 'login not successfull, like wanted')
  183. return true
  184. end
  185. screenshot(browser: instance, comment: 'login_failed')
  186. raise 'login failed'
  187. end
  188. if params[:success] == false
  189. raise 'login successfull but should not'
  190. end
  191. clues_close(
  192. browser: instance,
  193. optional: true,
  194. )
  195. screenshot(browser: instance, comment: 'login_ok')
  196. assert(true, 'login ok')
  197. login
  198. end
  199. =begin
  200. logout(
  201. browser: browser1
  202. )
  203. =end
  204. def logout(params = {})
  205. switch_window_focus(params)
  206. log('logout', params)
  207. instance = params[:browser] || @browser
  208. click(
  209. browser: instance,
  210. css: 'a[href="#current_user"]',
  211. mute_log: true,
  212. )
  213. click(
  214. browser: instance,
  215. css: 'a[href="#logout"]',
  216. mute_log: true,
  217. )
  218. 5.times do
  219. sleep 1
  220. login = instance.find_elements(css: '#login')[0]
  221. next if !login
  222. screenshot(browser: instance, comment: 'logout_ok')
  223. assert(true, 'logout ok')
  224. return
  225. end
  226. screenshot(browser: instance, comment: 'logout_failed')
  227. raise 'no login box found, seems logout was not successfully!'
  228. end
  229. =begin
  230. clues_close(
  231. browser: browser1,
  232. optional: false,
  233. )
  234. =end
  235. def clues_close(params = {})
  236. switch_window_focus(params)
  237. log('clues_close', params)
  238. instance = params[:browser] || @browser
  239. clues = instance.find_elements(css: '.js-modal--clue .js-close')[0]
  240. if !params[:optional] && !clues
  241. screenshot(browser: instance, comment: 'no_clues')
  242. raise 'Unable to closes clues, no clues found!'
  243. end
  244. return if !clues
  245. instance.execute_script("$('.js-modal--clue .js-close').click()")
  246. assert(true, 'clues closed')
  247. sleep 1
  248. end
  249. =begin
  250. notify_close(
  251. browser: browser1,
  252. optional: false,
  253. )
  254. =end
  255. def notify_close(params = {})
  256. switch_window_focus(params)
  257. log('notify_close', params)
  258. instance = params[:browser] || @browser
  259. notify = instance.find_elements(css: '.noty_inline_layout_container.i-am-new')[0]
  260. if !params[:optional] && !notify
  261. screenshot(browser: instance, comment: 'no_notify')
  262. raise 'Unable to closes notify, no notify found!'
  263. end
  264. return if !notify
  265. notify.click
  266. assert(true, 'notify closed')
  267. sleep 1
  268. end
  269. =begin
  270. location(
  271. browser: browser1,
  272. url: 'http://someurl',
  273. )
  274. =end
  275. def location(params)
  276. switch_window_focus(params)
  277. log('location', params)
  278. instance = params[:browser] || @browser
  279. instance.get(params[:url])
  280. # check if reload was successfull
  281. if !instance.find_elements(css: 'body')[0] || instance.find_elements(css: 'body')[0].text =~ /unavailable or too busy/i
  282. instance.navigate.refresh
  283. end
  284. screenshot(browser: instance, comment: 'location')
  285. end
  286. =begin
  287. location_check(
  288. browser: browser1,
  289. url: 'http://someurl',
  290. )
  291. =end
  292. def location_check(params)
  293. switch_window_focus(params)
  294. log('location_check', params)
  295. instance = params[:browser] || @browser
  296. sleep 0.7
  297. current_url = instance.current_url
  298. if !current_url.match?(/#{Regexp.quote(params[:url])}/)
  299. screenshot(browser: instance, comment: 'location_check_failed')
  300. raise "url #{current_url} is not matching #{params[:url]}"
  301. end
  302. assert(true, "url #{current_url} is matching #{params[:url]}")
  303. end
  304. =begin
  305. reload(
  306. browser: browser1,
  307. )
  308. =end
  309. def reload(params = {})
  310. switch_window_focus(params)
  311. log('reload', params)
  312. instance = params[:browser] || @browser
  313. screenshot(browser: instance, comment: 'reload_before')
  314. instance.navigate.refresh
  315. # check if reload was successfull
  316. if !instance.find_elements(css: 'body')[0] || instance.find_elements(css: 'body')[0].text =~ /unavailable or too busy/i
  317. instance.navigate.refresh
  318. end
  319. screenshot(browser: instance, comment: 'reload_after')
  320. end
  321. =begin
  322. click(
  323. browser: browser1,
  324. css: '.some_class',
  325. fast: false, # do not wait
  326. wait: 1, # wait 1 sec.
  327. )
  328. click(
  329. browser: browser1,
  330. text: '.partial_link_text',
  331. fast: false, # do not wait
  332. wait: 1, # wait 1 sec.
  333. )
  334. =end
  335. def click(params)
  336. switch_window_focus(params)
  337. log('click', params)
  338. instance = params[:browser] || @browser
  339. screenshot(browser: instance, comment: 'click_before')
  340. if params[:css]
  341. begin
  342. element = instance.find_elements(css: params[:css])[0]
  343. return if !element && params[:only_if_exists] == true
  344. #if element
  345. # instance.action.move_to(element).release.perform
  346. #end
  347. element.click
  348. rescue => e
  349. sleep 0.5
  350. # just try again
  351. log('click', { rescure: true })
  352. element = instance.find_elements(css: params[:css])[0]
  353. return if !element && params[:only_if_exists] == true
  354. #if element
  355. # instance.action.move_to(element).release.perform
  356. #end
  357. raise "No such element '#{params[:css]}'" if !element
  358. element.click
  359. end
  360. else
  361. sleep 0.5
  362. begin
  363. instance.find_elements(partial_link_text: params[:text])[0].click
  364. rescue => e
  365. sleep 0.5
  366. # just try again
  367. log('click', { rescure: true })
  368. element = instance.find_elements(partial_link_text: params[:text])[0]
  369. raise "No such element '#{params[:text]}'" if !element
  370. element.click
  371. end
  372. end
  373. sleep 0.2 if !params[:fast]
  374. sleep params[:wait] if params[:wait]
  375. end
  376. =begin
  377. scroll_to(
  378. browser: browser1,
  379. position: 'top', # botton
  380. css: '.some_class',
  381. )
  382. =end
  383. def scroll_to(params)
  384. switch_window_focus(params)
  385. log('scroll_to', params)
  386. instance = params[:browser] || @browser
  387. position = 'true'
  388. if params[:position] == 'botton'
  389. position = 'false'
  390. end
  391. screenshot(browser: instance, comment: 'scroll_to_before')
  392. execute(
  393. browser: instance,
  394. js: "\$('#{params[:css]}').get(0).scrollIntoView(#{position})",
  395. mute_log: params[:mute_log]
  396. )
  397. sleep 0.3
  398. screenshot(browser: instance, comment: 'scroll_to_after')
  399. end
  400. =begin
  401. modal_close(
  402. browser: browser1,
  403. )
  404. =end
  405. def modal_close(params = {})
  406. switch_window_focus(params)
  407. log('modal_close', params)
  408. instance = params[:browser] || @browser
  409. element = instance.find_elements(css: '.modal .js-close')[0]
  410. raise "No such modal to close #{params.inspect}" if !element
  411. element.click
  412. end
  413. =begin
  414. modal_ready(
  415. browser: browser1,
  416. )
  417. =end
  418. def modal_ready(params = {})
  419. switch_window_focus(params)
  420. log('modal_ready', params)
  421. instance = params[:browser] || @browser
  422. screenshot(browser: instance, comment: 'modal_ready_before')
  423. sleep 3
  424. screenshot(browser: instance, comment: 'modal_ready_after')
  425. end
  426. =begin
  427. modal_disappear(
  428. browser: browser1,
  429. timeout: 12, # default 8
  430. )
  431. =end
  432. def modal_disappear(params = {})
  433. switch_window_focus(params)
  434. log('modal_disappear', params)
  435. instance = params[:browser] || @browser
  436. screenshot(browser: instance, comment: 'modal_disappear_before')
  437. watch_for_disappear(
  438. browser: instance,
  439. css: '.modal',
  440. timeout: params[:timeout] || 8,
  441. )
  442. screenshot(browser: instance, comment: 'modal_disappear_after')
  443. end
  444. =begin
  445. execute(
  446. browser: browser1,
  447. js: '.some_class',
  448. )
  449. =end
  450. def execute(params)
  451. switch_window_focus(params)
  452. log('js', params)
  453. instance = params[:browser] || @browser
  454. if params[:js]
  455. return instance.execute_script(params[:js])
  456. end
  457. raise "Invalid execute params #{params.inspect}"
  458. end
  459. =begin
  460. exists(
  461. browser: browser1,
  462. css: '.some_class',
  463. )
  464. exists(
  465. displayed: false, # true|false
  466. browser: browser1,
  467. css: '.some_class',
  468. )
  469. =end
  470. def exists(params)
  471. switch_window_focus(params)
  472. log('exists', params)
  473. instance = params[:browser] || @browser
  474. if !instance.find_elements(css: params[:css])[0]
  475. screenshot(browser: instance, comment: 'exists_failed')
  476. raise "#{params[:css]} dosn't exist, but should"
  477. end
  478. if params.key?(:displayed)
  479. if params[:displayed] == true && !instance.find_elements(css: params[:css])[0].displayed?
  480. raise "#{params[:css]} is not displayed, but should"
  481. end
  482. if params[:displayed] == false && instance.find_elements(css: params[:css])[0].displayed?
  483. raise "#{params[:css]} is displayed, but should not"
  484. end
  485. end
  486. true
  487. end
  488. =begin
  489. exists_not(
  490. browser: browser1,
  491. css: '.some_class',
  492. )
  493. =end
  494. def exists_not(params)
  495. switch_window_focus(params)
  496. log('exists_not', params)
  497. instance = params[:browser] || @browser
  498. if instance.find_elements(css: params[:css])[0]
  499. screenshot(browser: instance, comment: 'exists_not_failed')
  500. raise "#{params[:css]} exists but should not"
  501. end
  502. true
  503. end
  504. =begin
  505. set(
  506. browser: browser1,
  507. css: '.some_class',
  508. value: true,
  509. slow: false,
  510. blur: true, # default false
  511. clear: true, # todo | default: true
  512. no_click: true,
  513. )
  514. =end
  515. def set(params)
  516. switch_window_focus(params)
  517. log('set', params)
  518. instance = params[:browser] || @browser
  519. screenshot(browser: instance, comment: 'set_before')
  520. element = instance.find_elements(css: params[:css])[0]
  521. if !params[:no_click]
  522. element.click
  523. end
  524. element.clear
  525. begin
  526. if !params[:slow]
  527. element.send_keys(params[:value])
  528. else
  529. element.send_keys('')
  530. keys = params[:value].to_s.split('')
  531. keys.each do |key|
  532. instance.action.send_keys(key).perform
  533. end
  534. end
  535. rescue => e
  536. sleep 0.5
  537. # just try again
  538. log('set', { rescure: true })
  539. element = instance.find_elements(css: params[:css])[0]
  540. raise "No such element '#{params[:css]}'" if !element
  541. if !params[:slow]
  542. element.send_keys(params[:value])
  543. else
  544. element.send_keys('')
  545. keys = params[:value].to_s.split('')
  546. keys.each do |key|
  547. instance.action.send_keys(key).perform
  548. end
  549. end
  550. end
  551. # it's not working stable with ff via selenium, use js
  552. if browser =~ /firefox/i && params[:css] =~ /\[data-name=/
  553. log('set_ff_trigger_workaround', params)
  554. instance.execute_script("$('#{params[:css]}').trigger('focusout')")
  555. end
  556. if params[:blur]
  557. instance.execute_script("$('#{params[:css]}').blur()")
  558. end
  559. sleep 0.2
  560. screenshot(browser: instance, comment: 'set_after')
  561. end
  562. =begin
  563. select(
  564. browser: browser1,
  565. css: '.some_class',
  566. value: 'Some Value',
  567. deselect_all: false, # default false
  568. )
  569. =end
  570. def select(params)
  571. switch_window_focus(params)
  572. log('select', params)
  573. instance = params[:browser] || @browser
  574. screenshot(browser: instance, comment: 'select_before')
  575. # searchable select
  576. element = instance.find_elements(css: "#{params[:css]}.js-shadow")[0]
  577. if element
  578. element = instance.find_elements(css: "#{params[:css]}.js-shadow + .js-input")[0]
  579. element.click
  580. element.clear
  581. sleep 0.4
  582. element.send_keys(params[:value])
  583. sleep 0.2
  584. element.send_keys(:enter)
  585. sleep 0.2
  586. return
  587. end
  588. # native select
  589. begin
  590. element = instance.find_elements(css: params[:css])[0]
  591. dropdown = Selenium::WebDriver::Support::Select.new(element)
  592. if params[:deselect_all]
  593. dropdown.deselect_all
  594. end
  595. dropdown.select_by(:text, params[:value])
  596. #puts "select - #{params.inspect}"
  597. rescue
  598. sleep 0.4
  599. # just try again
  600. log('select', { rescure: true })
  601. element = instance.find_elements(css: params[:css])[0]
  602. dropdown = Selenium::WebDriver::Support::Select.new(element)
  603. if params[:deselect_all]
  604. dropdown.deselect_all
  605. end
  606. dropdown.select_by(:text, params[:value])
  607. #puts "select2 - #{params.inspect}"
  608. end
  609. sleep 0.4
  610. screenshot(browser: instance, comment: 'select_after')
  611. end
  612. =begin
  613. switch(
  614. browser: browser1,
  615. css: '.some_class',
  616. type: 'on', # 'off'
  617. no_check: true, # do not check is switch has changed, in case if js alert
  618. )
  619. =end
  620. def switch(params)
  621. switch_window_focus(params)
  622. log('switch', params)
  623. instance = params[:browser] || @browser
  624. screenshot(browser: instance, comment: 'switch_before')
  625. element = instance.find_elements(css: "#{params[:css]} input[type=checkbox]")[0]
  626. checked = element.attribute('checked')
  627. if !checked
  628. if params[:type] == 'on'
  629. instance.find_elements(css: "#{params[:css]} label")[0].click
  630. sleep 2
  631. if params[:no_check] != true
  632. element = instance.find_elements(css: "#{params[:css]} input[type=checkbox]")[0]
  633. checked = element.attribute('checked')
  634. raise 'Switch not on!' if !checked
  635. end
  636. end
  637. elsif params[:type] == 'off'
  638. instance.find_elements(css: "#{params[:css]} label")[0].click
  639. sleep 2
  640. if params[:no_check] != true
  641. element = instance.find_elements(css: "#{params[:css]} input[type=checkbox]")[0]
  642. checked = element.attribute('checked')
  643. raise 'Switch not off!' if checked
  644. end
  645. end
  646. screenshot(browser: instance, comment: 'switch_after')
  647. end
  648. =begin
  649. check(
  650. browser: browser1,
  651. css: '.some_class',
  652. )
  653. =end
  654. def check(params)
  655. switch_window_focus(params)
  656. log('check', params)
  657. instance = params[:browser] || @browser
  658. screenshot(browser: instance, comment: 'check_before')
  659. instance.execute_script("if (!$('#{params[:css]}').prop('checked')) { $('#{params[:css]}').click() }")
  660. #element = instance.find_elements(css: params[:css])[0]
  661. #checked = element.attribute('checked')
  662. #element.click if !checked
  663. screenshot(browser: instance, comment: 'check_after')
  664. end
  665. =begin
  666. uncheck(
  667. browser: browser1,
  668. css: '.some_class',
  669. )
  670. =end
  671. def uncheck(params)
  672. switch_window_focus(params)
  673. log('uncheck', params)
  674. instance = params[:browser] || @browser
  675. screenshot(browser: instance, comment: 'uncheck_before')
  676. instance.execute_script("if ($('#{params[:css]}').prop('checked')) { $('#{params[:css]}').click() }")
  677. #element = instance.find_elements(css: params[:css])[0]
  678. #checked = element.attribute('checked')
  679. #element.click if checked
  680. screenshot(browser: instance, comment: 'uncheck_after')
  681. end
  682. =begin
  683. sendkey(
  684. browser: browser1,
  685. value: :enter,
  686. slow: false, # default false
  687. )
  688. =end
  689. def sendkey(params)
  690. switch_window_focus(params)
  691. log('sendkey', params)
  692. instance = params[:browser] || @browser
  693. element = nil
  694. if params[:css]
  695. element = instance.find_elements(css: params[:css])[0]
  696. end
  697. screenshot(browser: instance, comment: 'sendkey_before')
  698. if params[:value].class == Array
  699. params[:value].each do |key|
  700. if element
  701. element.send_keys(key)
  702. else
  703. instance.action.send_keys(key).perform
  704. end
  705. end
  706. screenshot(browser: instance, comment: 'sendkey_after')
  707. return
  708. end
  709. if element
  710. element.send_keys(params[:value])
  711. else
  712. instance.action.send_keys(params[:value]).perform
  713. end
  714. if params[:slow]
  715. sleep 1.5
  716. else
  717. sleep 0.2
  718. end
  719. screenshot(browser: instance, comment: 'sendkey_after')
  720. end
  721. =begin
  722. match(
  723. browser: browser1,
  724. css: '#content .text-1',
  725. value: 'some test for browser and some other for browser',
  726. attribute: 'some_attribute', # match on attribute
  727. should_not_match: true,
  728. no_quote: false, # use regex
  729. )
  730. =end
  731. def match(params, fallback = false)
  732. switch_window_focus(params)
  733. log('match', params)
  734. instance = params[:browser] || @browser
  735. element = instance.find_elements(css: params[:css])[0]
  736. if params[:css].match?(/select/)
  737. dropdown = Selenium::WebDriver::Support::Select.new(element)
  738. success = false
  739. dropdown.selected_options&.each do |option|
  740. if option.text == params[:value]
  741. success = true
  742. end
  743. end
  744. if params[:should_not_match]
  745. if success
  746. screenshot(browser: instance, comment: 'match_failed')
  747. raise "should not match '#{params[:value]}' in select list, but is matching"
  748. end
  749. elsif !success
  750. screenshot(browser: instance, comment: 'match_failed')
  751. raise "not matching '#{params[:value]}' in select list"
  752. end
  753. return true
  754. end
  755. # match on attribute
  756. begin
  757. text = if params[:attribute]
  758. element.attribute(params[:attribute])
  759. elsif params[:css].match?(/(input|textarea)/i)
  760. element.attribute('value')
  761. else
  762. element.text
  763. end
  764. rescue => e
  765. # just try again
  766. if !fallback
  767. return match(params, true)
  768. end
  769. raise e.inspect
  770. end
  771. # do cleanups (needed for richtext tests)
  772. if params[:cleanup]
  773. text.gsub!(/\s+$/m, '')
  774. params[:value].gsub!(/\s+$/m, '')
  775. end
  776. match = false
  777. if params[:no_quote]
  778. #puts "aaaa #{text}/#{params[:value]}"
  779. if text =~ /#{params[:value]}/i
  780. match = $1 || true
  781. end
  782. elsif text.match?(/#{Regexp.quote(params[:value])}/i)
  783. match = true
  784. end
  785. if match
  786. if params[:should_not_match]
  787. screenshot(browser: instance, comment: 'match_failed')
  788. raise "matching '#{params[:value]}' in content '#{text}' but should not!"
  789. end
  790. elsif !params[:should_not_match]
  791. screenshot(browser: instance, comment: 'match_failed')
  792. raise "not matching '#{params[:value]}' in content '#{text}' but should!"
  793. end
  794. sleep 0.2
  795. match
  796. end
  797. =begin
  798. match_not(
  799. browser: browser1,
  800. css: '#content .text-1',
  801. value: 'some test for browser and some other for browser',
  802. attribute: 'some_attribute', # match on attribute
  803. should_not_match: true,
  804. no_quote: false, # use regex
  805. )
  806. =end
  807. def match_not(params)
  808. switch_window_focus(params)
  809. log('match_not', params)
  810. params[:should_not_match] = true
  811. match(params)
  812. end
  813. =begin
  814. set type of task (closeTab, closeNextInOverview, stayOnTab)
  815. task_type(
  816. browser: browser1,
  817. type: 'stayOnTab',
  818. )
  819. =end
  820. def task_type(params)
  821. switch_window_focus(params)
  822. log('task_type', params)
  823. instance = params[:browser] || @browser
  824. if params[:type]
  825. instance.find_elements(css: '.content.active .js-secondaryActionButtonLabel')[0].click
  826. instance.find_elements(css: ".content.active .js-secondaryActionLabel[data-type=#{params[:type]}]")[0].click
  827. return
  828. end
  829. raise "Unknown params for task_type: #{params.inspect}"
  830. end
  831. =begin
  832. cookie(
  833. browser: browser1,
  834. name: '^_zammad.+?',
  835. value: '.+?',
  836. expires: nil,
  837. )
  838. cookie(
  839. browser: browser1,
  840. name: '^_zammad.+?',
  841. should_not_exist: true,
  842. )
  843. =end
  844. def cookie(params)
  845. switch_window_focus(params)
  846. log('cookie', params)
  847. instance = params[:browser] || @browser
  848. if !browser_support_cookies
  849. assert(true, "'#{params[:value]}' ups browser is not supporting reading cookies, go ahead")
  850. return true
  851. end
  852. cookies = instance.manage.all_cookies
  853. cookies.each do |cookie|
  854. # :name=>"_zammad_session_c25832f4de2", :value=>"adc31cd21615cb0a7ab269184ec8b76f", :path=>"/", :domain=>"localhost", :expires=>nil, :secure=>false}
  855. next if cookie[:name] !~ /#{params[:name]}/i
  856. if params.key?(:value) && cookie[:value].to_s =~ /#{params[:value]}/i
  857. assert(true, "matching value '#{params[:value]}' in cookie '#{cookie}'")
  858. else
  859. raise "not matching value '#{params[:value]}' in cookie '#{cookie}'"
  860. end
  861. if params.key?(:expires) && cookie[:expires].to_s =~ /#{params[:expires]}/i
  862. assert(true, "matching expires '#{params[:expires].inspect}' in cookie '#{cookie}'")
  863. else
  864. raise "not matching expires '#{params[:expires]}' in cookie '#{cookie}'"
  865. end
  866. return if !params[:should_not_exist]
  867. raise "cookie with name '#{params[:name]}' should not exist, but exists '#{cookies}'"
  868. end
  869. if params[:should_not_exist]
  870. assert(true, "cookie with name '#{params[:name]}' is not existing")
  871. return
  872. end
  873. raise "not matching name '#{params[:name]}' in cookie '#{cookies}'"
  874. end
  875. =begin
  876. verify_title(
  877. browser: browser1,
  878. value: 'some title',
  879. )
  880. =end
  881. def verify_title(params = {})
  882. switch_window_focus(params)
  883. log('verify_title', params)
  884. instance = params[:browser] || @browser
  885. title = instance.title
  886. if title.match?(/#{params[:value]}/i)
  887. assert(true, "matching '#{params[:value]}' in title '#{title}'")
  888. else
  889. raise "not matching '#{params[:value]}' in title '#{title}'"
  890. end
  891. end
  892. =begin
  893. verify_task(
  894. browser: browser1,
  895. data: {
  896. title: 'some title',
  897. modified: true, # optional
  898. }
  899. )
  900. =end
  901. def verify_task(params = {}, fallback = false)
  902. switch_window_focus(params)
  903. log('verify_task', params)
  904. instance = params[:browser] || @browser
  905. data = params[:data]
  906. sleep 1
  907. begin
  908. # verify title
  909. if data[:title]
  910. title = instance.find_elements(css: '.tasks .is-active')[0].text.strip
  911. if title.match?(/#{data[:title]}/i)
  912. assert(true, "matching '#{data[:title]}' in title '#{title}'")
  913. else
  914. screenshot(browser: instance, comment: 'verify_task_failed')
  915. raise "not matching '#{data[:title]}' in title '#{title}'"
  916. end
  917. end
  918. # verify modified
  919. if data.key?(:modified)
  920. exists = instance.find_elements(css: '.tasks .is-active')[0]
  921. is_modified = instance.find_elements(css: '.tasks .is-modified')[0]
  922. puts "m #{data[:modified].inspect}"
  923. if exists
  924. puts ' exists'
  925. end
  926. if is_modified
  927. puts ' is_modified'
  928. end
  929. if data[:modified] == true
  930. if is_modified
  931. assert(true, "task '#{data[:title]}' is modifed")
  932. elsif !exists
  933. screenshot(browser: instance, comment: 'verify_task_failed')
  934. raise "task '#{data[:title]}' not exists, should not modified"
  935. else
  936. screenshot(browser: instance, comment: 'verify_task_failed')
  937. raise "task '#{data[:title]}' is not modifed"
  938. end
  939. elsif !is_modified
  940. assert(true, "task '#{data[:title]}' is modifed")
  941. elsif !exists
  942. screenshot(browser: instance, comment: 'verify_task_failed')
  943. raise "task '#{data[:title]}' not exists, should be not modified"
  944. else
  945. screenshot(browser: instance, comment: 'verify_task_failed')
  946. raise "task '#{data[:title]}' is modifed, but should not"
  947. end
  948. end
  949. rescue => e
  950. # just try again
  951. if !fallback
  952. verify_task(params, true)
  953. end
  954. raise 'ERROR: ' + e.inspect
  955. end
  956. true
  957. end
  958. =begin
  959. open_task(
  960. browser: browser1,
  961. data: {
  962. title: 'some title',
  963. }
  964. )
  965. =end
  966. def open_task(params = {})
  967. switch_window_focus(params)
  968. log('open_task', params)
  969. instance = params[:browser] || @browser
  970. data = params[:data]
  971. element = instance.find_element(css: '#navigation').find_element(partial_link_text: data[:title])
  972. if !element
  973. screenshot(browser: instance, comment: 'open_task_failed')
  974. raise "no task with title '#{data[:title]}' found"
  975. end
  976. # firefix/marionette issue with Selenium::WebDriver::Error::ElementNotInteractableError: could not be scrolled into view
  977. # use js workaround instead of native click
  978. instance.execute_script("$('#navigation .tasks .task:contains(\"#{data[:title]}\") .nav-tab-name').click()")
  979. #element.click
  980. true
  981. end
  982. =begin
  983. close_task(
  984. browser: browser1,
  985. data: {
  986. title: 'some title',
  987. },
  988. discard_changes: true,
  989. )
  990. =end
  991. def close_task(params = {})
  992. switch_window_focus(params)
  993. log('close_task', params)
  994. instance = params[:browser] || @browser
  995. data = params[:data]
  996. element = instance.find_element(css: '#navigation').find_element(partial_link_text: data[:title])
  997. if !element
  998. screenshot(browser: instance, comment: 'close_task_failed')
  999. raise "no task with title '#{data[:title]}' found"
  1000. end
  1001. instance.action.move_to(element).release.perform
  1002. sleep 0.1
  1003. instance.execute_script("$('#navigation .tasks .task:contains(\"#{data[:title]}\") .js-close').click()")
  1004. # accept task close warning
  1005. if params[:discard_changes]
  1006. modal_ready(browser: instance)
  1007. instance.find_elements(css: '.modal button.js-submit')[0].click
  1008. modal_disappear(browser: instance)
  1009. end
  1010. true
  1011. end
  1012. =begin
  1013. file_upload(
  1014. browser: browser1,
  1015. css: '.content.active .attachmentPlaceholder-inputHolder input'
  1016. files: ['path/in/home/some_file.ext'], # 'test/fixtures/test1.pdf'
  1017. )
  1018. =end
  1019. def file_upload(params = {})
  1020. switch_window_focus(params)
  1021. log('file_upload', params)
  1022. instance = params[:browser] || @browser
  1023. params[:files].each do |file|
  1024. instance.find_elements(css: params[:css])[0].send_keys(Rails.root.join(file))
  1025. end
  1026. sleep 2 * params[:files].count
  1027. end
  1028. =begin
  1029. watch_for(
  1030. browser: browser1,
  1031. css: '#content .text-1',
  1032. value: 'some text',
  1033. attribute: 'some_attribute' # optional
  1034. timeout: 16, # in sec, default 16
  1035. )
  1036. =end
  1037. def watch_for(params = {})
  1038. switch_window_focus(params)
  1039. log('watch_for', params)
  1040. instance = params[:browser] || @browser
  1041. timeout = 16
  1042. if params[:timeout]
  1043. timeout = params[:timeout]
  1044. end
  1045. loops = timeout.to_i * 2
  1046. text = ''
  1047. (1..loops).each do
  1048. element = instance.find_elements(css: params[:css])[0]
  1049. if element #&& element.displayed?
  1050. begin
  1051. # watch for selector
  1052. if !params[:attribute] && !params[:value]
  1053. assert(true, "'#{params[:css]}' found")
  1054. sleep 0.5
  1055. return true
  1056. # match an attribute
  1057. else
  1058. text = if params[:attribute]
  1059. element.attribute(params[:attribute])
  1060. elsif params[:css].match?(/(input|textarea)/i)
  1061. element.attribute('value')
  1062. else
  1063. element.text
  1064. end
  1065. if text.match?(/#{params[:value]}/i)
  1066. assert(true, "'#{params[:value]}' found in '#{text}'")
  1067. sleep 0.5
  1068. return true
  1069. end
  1070. end
  1071. rescue
  1072. # try again
  1073. end
  1074. end
  1075. sleep 0.5
  1076. end
  1077. screenshot(browser: instance, comment: 'watch_for_failed')
  1078. if !params[:attribute] && !params[:value]
  1079. raise "'#{params[:css]}' not found"
  1080. end
  1081. raise "'#{params[:value]}' not found in '#{text}'"
  1082. end
  1083. =begin
  1084. wait untill selector disabppears
  1085. watch_for_disappear(
  1086. browser: browser1,
  1087. css: '#content .text-1',
  1088. timeout: 16, # in sec, default 16
  1089. )
  1090. wait untill text in selector disabppears
  1091. watch_for_disappear(
  1092. browser: browser1,
  1093. css: '#content .text-1',
  1094. value: 'some value as regexp',
  1095. timeout: 16, # in sec, default 16
  1096. )
  1097. =end
  1098. def watch_for_disappear(params = {})
  1099. switch_window_focus(params)
  1100. log('watch_for_disappear', params)
  1101. instance = params[:browser] || @browser
  1102. timeout = 16
  1103. if params[:timeout]
  1104. timeout = params[:timeout]
  1105. end
  1106. loops = timeout.to_i
  1107. text = ''
  1108. (1..loops).each do
  1109. element = instance.find_elements(css: params[:css])[0]
  1110. if !element #|| element.displayed?
  1111. assert(true, 'not found')
  1112. sleep 1
  1113. return true
  1114. end
  1115. if params[:value]
  1116. begin
  1117. text = instance.find_elements(css: params[:css])[0].text
  1118. if !text.match?(/#{params[:value]}/i)
  1119. assert(true, "not matching '#{params[:value]}' in text '#{text}'")
  1120. sleep 1
  1121. return true
  1122. end
  1123. rescue
  1124. # try again
  1125. end
  1126. end
  1127. sleep 1
  1128. end
  1129. screenshot(browser: instance, comment: 'disappear_failed')
  1130. raise "#{params[:css]}) still exsists"
  1131. end
  1132. =begin
  1133. shortcut(
  1134. browser: browser1,
  1135. key: 'x',
  1136. )
  1137. =end
  1138. def shortcut(params = {})
  1139. switch_window_focus(params)
  1140. log('shortcut', params)
  1141. instance = params[:browser] || @browser
  1142. screenshot(browser: instance, comment: 'shortcut_before')
  1143. instance.action.key_down(:control)
  1144. .key_down(:shift)
  1145. .send_keys(params[:key])
  1146. .key_up(:shift)
  1147. .key_up(:control)
  1148. .perform
  1149. screenshot(browser: instance, comment: 'shortcut_after')
  1150. end
  1151. =begin
  1152. window_keys(
  1153. browser: browser1,
  1154. value: 'x',
  1155. )
  1156. =end
  1157. def window_keys(params = {})
  1158. switch_window_focus(params)
  1159. log('window_keys', params)
  1160. instance = params[:browser] || @browser
  1161. instance.action.send_keys(params[:value]).perform
  1162. end
  1163. =begin
  1164. tasks_close_all(
  1165. browser: browser1,
  1166. )
  1167. =end
  1168. def tasks_close_all(params = {})
  1169. switch_window_focus(params)
  1170. log('tasks_close_all', params)
  1171. instance = params[:browser] || @browser
  1172. 99.times do
  1173. #sleep 0.5
  1174. begin
  1175. if instance.find_elements(css: '#navigation .tasks .task:first-child')[0]
  1176. instance.action.move_to(instance.find_elements(css: '#navigation .tasks .task:first-child')[0]).release.perform
  1177. click_element = instance.find_elements(css: '#navigation .tasks .task:first-child .js-close')[0]
  1178. if click_element
  1179. click_element.click
  1180. # accept task close warning
  1181. if instance.find_elements(css: '.modal button.js-submit')[0]
  1182. sleep 0.4
  1183. instance.find_elements(css: '.modal button.js-submit')[0].click
  1184. end
  1185. end
  1186. else
  1187. break
  1188. end
  1189. rescue
  1190. # try again
  1191. end
  1192. end
  1193. assert(true, 'all tasks closed')
  1194. end
  1195. =begin
  1196. close_online_notitifcation(
  1197. browser: browser1,
  1198. data: {
  1199. #title: 'some title',
  1200. position: 3,
  1201. },
  1202. )
  1203. =end
  1204. def close_online_notitifcation(params = {})
  1205. switch_window_focus(params)
  1206. log('close_online_notitifcation', params)
  1207. instance = params[:browser] || @browser
  1208. data = params[:data]
  1209. if data[:title]
  1210. element = instance.find_elements(partial_link_text: data[:title])[0]
  1211. if !element
  1212. screenshot(browser: instance, comment: 'close_online_notitifcation')
  1213. raise "no online notification with title '#{data[:title]}' found"
  1214. end
  1215. instance.action.move_to(element).release.perform
  1216. sleep 0.1
  1217. instance.execute_script("$('.js-notificationsContainer .js-items .js-item .activity-text:contains(\"#{data[:title]}\") .js-remove').first().click()")
  1218. else
  1219. css = ".js-notificationsContainer .js-items .js-item:nth-child(#{data[:position]})"
  1220. element = instance.find_elements(css: css)[0]
  1221. if !element
  1222. screenshot(browser: instance, comment: 'close_online_notitifcation')
  1223. raise "no online notification with postion '#{css}' found"
  1224. end
  1225. instance.action.move_to(element).release.perform
  1226. sleep 0.1
  1227. instance.find_elements(css: "#{css} .js-remove")[0].click
  1228. end
  1229. true
  1230. end
  1231. =begin
  1232. online_notitifcation_close_all(
  1233. browser: browser1,
  1234. )
  1235. =end
  1236. def online_notitifcation_close_all(params = {})
  1237. switch_window_focus(params)
  1238. log('online_notitifcation_close_all', params)
  1239. instance = params[:browser] || @browser
  1240. 99.times do
  1241. sleep 0.5
  1242. begin
  1243. if instance.find_elements(css: '.js-notificationsContainer .js-item:first-child')[0]
  1244. instance.action.move_to(instance.find_elements(css: '.js-notificationsContainer .js-item:first-child')[0]).perform
  1245. sleep 0.1
  1246. click_element = instance.find_elements(css: '.js-notificationsContainer .js-item:first-child .js-remove')[0]
  1247. click_element&.click
  1248. else
  1249. break
  1250. end
  1251. rescue
  1252. # try again
  1253. end
  1254. end
  1255. assert(true, 'all online notification closed')
  1256. end
  1257. =begin
  1258. empty_search(
  1259. browser: browser1,
  1260. )
  1261. =end
  1262. def empty_search(params = {})
  1263. switch_window_focus(params)
  1264. log('empty_search', params)
  1265. instance = params[:browser] || @browser
  1266. # empty search box by x
  1267. begin
  1268. instance.find_elements(css: '.search .js-emptySearch')[0].click
  1269. rescue
  1270. # in issues with ff & selenium, sometimes exeption appears
  1271. # "Element is not currently visible and so may not be interacted with"
  1272. log('empty_search via js')
  1273. instance.execute_script('$(".search .js-emptySearch").click()')
  1274. end
  1275. sleep 0.5
  1276. text = instance.find_elements(css: '#global-search')[0].attribute('value')
  1277. if !text
  1278. raise '#global-search is not empty!'
  1279. end
  1280. true
  1281. end
  1282. =begin
  1283. ticket_customer_select(
  1284. browser: browser1,
  1285. css: '#content .text-1',
  1286. customer: '',
  1287. )
  1288. =end
  1289. def ticket_customer_select(params = {})
  1290. switch_window_focus(params)
  1291. log('ticket_customer_select', params)
  1292. instance = params[:browser] || @browser
  1293. element = instance.find_elements(css: params[:css] + ' input[name="customer_id_completion"]')[0]
  1294. element.click
  1295. element.clear
  1296. element.send_keys(params[:customer])
  1297. sleep 2.5
  1298. element.send_keys(:enter)
  1299. #instance.find_elements(css: params[:css] + ' .recipientList-entry.js-object.is-active')[0].click
  1300. sleep 0.4
  1301. assert(true, 'ticket_customer_select')
  1302. end
  1303. =begin
  1304. overview_create(
  1305. browser: browser1,
  1306. data: {
  1307. name: name,
  1308. roles: ['Agent'],
  1309. selector: {
  1310. 'Priority': '1 low',
  1311. },
  1312. 'order::direction' => 'down',
  1313. }
  1314. )
  1315. =end
  1316. def overview_create(params)
  1317. switch_window_focus(params)
  1318. log('overview_create', params)
  1319. instance = params[:browser] || @browser
  1320. data = params[:data]
  1321. click(
  1322. browser: instance,
  1323. css: 'a[href="#manage"]',
  1324. mute_log: true,
  1325. )
  1326. click(
  1327. browser: instance,
  1328. css: '.content.active a[href="#manage/overviews"]',
  1329. mute_log: true,
  1330. )
  1331. click(
  1332. browser: instance,
  1333. css: '.content.active a[data-type="new"]',
  1334. mute_log: true,
  1335. )
  1336. modal_ready(browser: instance)
  1337. if data[:name]
  1338. set(
  1339. browser: instance,
  1340. css: '.modal input[name=name]',
  1341. value: data[:name],
  1342. mute_log: true,
  1343. )
  1344. end
  1345. if data[:roles]
  1346. 99.times do
  1347. begin
  1348. element = instance.find_elements(css: '.modal .js-selected[data-name=role_ids] .js-option:not(.is-hidden)')[0]
  1349. break if !element
  1350. element.click
  1351. sleep 0.1
  1352. end
  1353. end
  1354. data[:roles].each do |role|
  1355. instance.execute_script("$(\".modal [data-name=role_ids] .js-pool .js-option:not(.is-hidden):contains('#{role}')\").first().click()")
  1356. end
  1357. end
  1358. data[:selector]&.each do |key, value|
  1359. select(
  1360. browser: instance,
  1361. css: '.modal .ticket_selector .js-attributeSelector select',
  1362. value: key,
  1363. mute_log: true,
  1364. )
  1365. sleep 0.5
  1366. select(
  1367. browser: instance,
  1368. css: '.modal .ticket_selector .js-value select',
  1369. value: value,
  1370. deselect_all: true,
  1371. mute_log: true,
  1372. )
  1373. end
  1374. if data['order::direction']
  1375. select(
  1376. browser: instance,
  1377. css: '.modal select[name="order::direction"]',
  1378. value: data['order::direction'],
  1379. mute_log: true,
  1380. )
  1381. end
  1382. instance.find_elements(css: '.modal button.js-submit')[0].click
  1383. modal_disappear(browser: instance)
  1384. 11.times do
  1385. element = instance.find_elements(css: 'body')[0]
  1386. text = element.text
  1387. if text.match?(/#{Regexp.quote(data[:name])}/)
  1388. assert(true, 'overview created')
  1389. overview = {
  1390. name: name,
  1391. }
  1392. sleep 1
  1393. return overview
  1394. end
  1395. sleep 1
  1396. end
  1397. screenshot(browser: instance, comment: 'overview_create_failed')
  1398. raise 'overview creation failed'
  1399. end
  1400. =begin
  1401. overview_update(
  1402. browser: browser1,
  1403. data: {
  1404. name: name,
  1405. roles: ['Agent'],
  1406. selector: {
  1407. 'Priority': '1 low',
  1408. },
  1409. 'order::direction' => 'down',
  1410. }
  1411. )
  1412. =end
  1413. def overview_update(params)
  1414. switch_window_focus(params)
  1415. log('overview_create', params)
  1416. instance = params[:browser] || @browser
  1417. data = params[:data]
  1418. click(
  1419. browser: instance,
  1420. css: 'a[href="#manage"]',
  1421. mute_log: true,
  1422. )
  1423. click(
  1424. browser: instance,
  1425. css: '.content.active a[href="#manage/overviews"]',
  1426. mute_log: true,
  1427. )
  1428. instance.execute_script("$(\".content.active td:contains('#{data[:name]}')\").first().click()")
  1429. sleep 2
  1430. if data[:name]
  1431. set(
  1432. browser: instance,
  1433. css: '.modal input[name=name]',
  1434. value: data[:name],
  1435. mute_log: true,
  1436. )
  1437. end
  1438. if data[:roles]
  1439. 99.times do
  1440. begin
  1441. element = instance.find_elements(css: '.modal .js-selected[data-name=role_ids] .js-option:not(.is-hidden)')[0]
  1442. break if !element
  1443. element.click
  1444. sleep 0.1
  1445. end
  1446. end
  1447. data[:roles].each do |role|
  1448. instance.execute_script("$(\".modal [data-name=role_ids] .js-pool .js-option:not(.is-hidden):contains('#{role}')\").first().click()")
  1449. end
  1450. end
  1451. data[:selector]&.each do |key, value|
  1452. select(
  1453. browser: instance,
  1454. css: '.modal .ticket_selector .js-attributeSelector select',
  1455. value: key,
  1456. mute_log: true,
  1457. )
  1458. sleep 0.5
  1459. select(
  1460. browser: instance,
  1461. css: '.modal .ticket_selector .js-value select',
  1462. value: value,
  1463. deselect_all: true,
  1464. mute_log: true,
  1465. )
  1466. end
  1467. if data['order::direction']
  1468. select(
  1469. browser: instance,
  1470. css: '.modal select[name="order::direction"]',
  1471. value: data['order::direction'],
  1472. mute_log: true,
  1473. )
  1474. end
  1475. instance.find_elements(css: '.modal button.js-submit')[0].click
  1476. modal_disappear(browser: instance)
  1477. 11.times do
  1478. element = instance.find_elements(css: 'body')[0]
  1479. text = element.text
  1480. if text.match?(/#{Regexp.quote(data[:name])}/)
  1481. assert(true, 'overview updated')
  1482. overview = {
  1483. name: name,
  1484. }
  1485. sleep 1
  1486. return overview
  1487. end
  1488. sleep 1
  1489. end
  1490. screenshot(browser: instance, comment: 'overview_update_failed')
  1491. raise 'overview update failed'
  1492. end
  1493. =begin
  1494. ticket = ticket_create(
  1495. browser: browser1,
  1496. data: {
  1497. customer: 'nico',
  1498. group: 'Users', # optional / '-NONE-' # if group selection should not be shown
  1499. priority: '2 normal',
  1500. state: 'open',
  1501. title: 'overview #1',
  1502. body: 'overview #1',
  1503. },
  1504. do_not_submit: true,
  1505. )
  1506. returns (in case of submitted)
  1507. {
  1508. id: 123,
  1509. number: '100001',
  1510. title: 'overview #1',
  1511. }
  1512. ticket = ticket_create(
  1513. browser: browser1,
  1514. data: {
  1515. customer: 'nico',
  1516. group: 'Users', # optional / '-NONE-' # if group selection should not be shown
  1517. priority: '2 normal',
  1518. state: 'open',
  1519. title: 'overview #1',
  1520. body: 'overview #1',
  1521. },
  1522. custom_data_select: {
  1523. key1: 'some value',
  1524. },
  1525. custom_data_input: {
  1526. key1: 'some value',
  1527. },
  1528. custom_data_date: {
  1529. key!: '02/28/2018',
  1530. }
  1531. disable_group_check: true,
  1532. )
  1533. =end
  1534. def ticket_create(params)
  1535. switch_window_focus(params)
  1536. log('ticket_create', params)
  1537. instance = params[:browser] || @browser
  1538. data = params[:data]
  1539. click(
  1540. browser: instance,
  1541. css: 'a[href="#new"]',
  1542. mute_log: true,
  1543. only_if_exists: true,
  1544. )
  1545. click(
  1546. browser: instance,
  1547. css: 'a[href="#ticket/create"]',
  1548. mute_log: true,
  1549. )
  1550. found = false
  1551. 7.times do
  1552. element = instance.find_elements(css: '.content.active .newTicket')[0]
  1553. if element
  1554. found = true
  1555. break
  1556. end
  1557. sleep 1
  1558. end
  1559. if !found
  1560. screenshot(browser: instance, comment: 'ticket_create_failed')
  1561. raise 'no ticket create screen found!'
  1562. end
  1563. if data[:group]
  1564. if data[:group] == '-NONE-'
  1565. # check if owner selection exists
  1566. count = instance.find_elements(css: '.content.active .newTicket select[name="group_id"] option').count
  1567. if count.nonzero?
  1568. instance.find_elements(css: '.content.active .newTicket select[name="group_id"] option').each do |element|
  1569. log('ticket_create invalid group count', text: element.text)
  1570. end
  1571. end
  1572. assert_equal(0, count, 'owner selection should not be showm')
  1573. # check count of agents, should be only 3 / - selection + master + agent on init screen
  1574. count = instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').count
  1575. if count != 3
  1576. instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').each do |element|
  1577. log('ticket_create invalid owner count', text: element.text)
  1578. end
  1579. end
  1580. assert_equal(3, count, 'check if owner selection is - selection + master + agent per default')
  1581. else
  1582. # check count of agents, should be only 1 / - selection on init screen
  1583. if !params[:disable_group_check]
  1584. count = instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').count
  1585. if count != 1
  1586. instance.find_elements(css: '.content.active .newTicket select[name="owner_id"] option').each do |element|
  1587. log('ticket_create invalid owner count', text: element.text)
  1588. end
  1589. end
  1590. assert_equal(1, count, 'check if owner selection is empty per default')
  1591. end
  1592. select(
  1593. browser: instance,
  1594. css: '.content.active .newTicket select[name="group_id"]',
  1595. value: data[:group],
  1596. mute_log: true,
  1597. )
  1598. sleep 0.2
  1599. end
  1600. end
  1601. if data[:priority]
  1602. select(
  1603. browser: instance,
  1604. css: '.content.active .newTicket select[name="priority_id"]',
  1605. value: data[:priority],
  1606. mute_log: true,
  1607. )
  1608. end
  1609. if data[:state]
  1610. select(
  1611. browser: instance,
  1612. css: '.content.active .newTicket select[name="state_id"]',
  1613. value: data[:state],
  1614. mute_log: true,
  1615. )
  1616. end
  1617. if data[:title]
  1618. set(
  1619. browser: instance,
  1620. css: '.content.active .newTicket input[name="title"]',
  1621. value: data[:title],
  1622. clear: true,
  1623. mute_log: true,
  1624. )
  1625. end
  1626. if data[:body]
  1627. set(
  1628. browser: instance,
  1629. css: '.content.active .newTicket div[data-name=body]',
  1630. value: data[:body],
  1631. clear: true,
  1632. mute_log: true,
  1633. )
  1634. end
  1635. if data[:customer]
  1636. element = instance.find_elements(css: '.content.active .newTicket input[name="customer_id_completion"]')[0]
  1637. element.click
  1638. element.clear
  1639. # ff issue, sometimes focus event gets dropped
  1640. # if drowdown is not open, try it again
  1641. if !instance.find_elements(css: '.content.active .newTicket .js-recipientDropdown.open')[0]
  1642. instance.execute_script('$(".active .newTicket .js-recipientDropdown").addClass("open")')
  1643. end
  1644. element.send_keys(data[:customer])
  1645. sleep 2.5
  1646. element.send_keys(:enter)
  1647. sleep 0.4
  1648. # ff issue, sometimes enter event gets dropped
  1649. # take user manually
  1650. if instance.find_elements(css: '.content.active .newTicket .js-recipientDropdown.open')[0]
  1651. instance.find_elements(css: '.content.active .newTicket .recipientList-entry.js-object.is-active')[0].click
  1652. sleep 0.4
  1653. end
  1654. end
  1655. params[:custom_data_select]&.each do |local_key, local_value|
  1656. select(
  1657. browser: instance,
  1658. css: ".content.active .newTicket select[name=\"#{local_key}\"]",
  1659. value: local_value,
  1660. )
  1661. end
  1662. params[:custom_data_input]&.each do |local_key, local_value|
  1663. set(
  1664. browser: instance,
  1665. css: ".content.active .newTicket input[name=\"#{local_key}\"]",
  1666. value: local_value,
  1667. clear: true,
  1668. )
  1669. end
  1670. params[:custom_data_date]&.each do |local_key, local_value|
  1671. set(
  1672. browser: instance,
  1673. css: ".content.active .newTicket div[data-name=\"#{local_key}\"] input[data-item=\"date\"]",
  1674. value: local_value,
  1675. clear: true,
  1676. )
  1677. end
  1678. if data[:attachment]
  1679. file_upload(
  1680. browser: instance,
  1681. css: '.content.active .text-1',
  1682. value: 'some text',
  1683. )
  1684. end
  1685. if params[:do_not_submit]
  1686. assert(true, 'ticket created without submit')
  1687. return
  1688. end
  1689. sleep 0.5
  1690. #instance.execute_script('$(".content.active .newTicket form").submit();')
  1691. click(
  1692. browser: instance,
  1693. css: '.content.active .newTicket button.js-submit',
  1694. mute_log: true,
  1695. )
  1696. sleep 1
  1697. 9.times do
  1698. if instance.current_url.match?(/#{Regexp.quote('#ticket/zoom/')}/)
  1699. assert(true, 'ticket created')
  1700. sleep 2.5
  1701. id = instance.current_url
  1702. id.gsub!(//,)
  1703. id.gsub!(%r{^.+?/(\d+)$}, '\\1')
  1704. element = instance.find_elements(css: '.content.active .ticketZoom-header .ticket-number')[0]
  1705. if element
  1706. number = element.text
  1707. ticket = {
  1708. id: id,
  1709. number: number,
  1710. title: data[:title],
  1711. }
  1712. sleep 3 # wait until notify is gone
  1713. screenshot(browser: instance, comment: 'ticket_create_ok')
  1714. return ticket
  1715. end
  1716. end
  1717. sleep 1
  1718. end
  1719. screenshot(browser: instance, comment: 'ticket_create_failed')
  1720. raise "ticket creation failed, can't get zoom url (current url is '#{instance.current_url}')"
  1721. end
  1722. =begin
  1723. ticket_update(
  1724. browser: browser1,
  1725. data: {
  1726. title: '',
  1727. customer: 'some_customer@example.com',
  1728. body: 'some body',
  1729. group: 'some group', # optional
  1730. priority: '1 low',
  1731. state: 'closed',
  1732. },
  1733. do_not_submit: true,
  1734. )
  1735. ticket_update(
  1736. browser: browser1,
  1737. data: {
  1738. title: '',
  1739. customer: 'some_customer@example.com',
  1740. body: 'some body',
  1741. group: 'some group', # optional
  1742. priority: '1 low',
  1743. state: 'closed',
  1744. },
  1745. custom_data_select: {
  1746. key1: 'some value',
  1747. },
  1748. custom_data_input: {
  1749. key1: 'some value',
  1750. },
  1751. custom_data_date: {
  1752. key1: '02/21/2018',
  1753. },
  1754. do_not_submit: true,
  1755. task_type: 'stayOnTab', # default: stayOnTab / possible: closeTab, closeNextInOverview, stayOnTab
  1756. )
  1757. =end
  1758. def ticket_update(params)
  1759. switch_window_focus(params)
  1760. log('ticket_update', params)
  1761. instance = params[:browser] || @browser
  1762. data = params[:data]
  1763. if data[:title]
  1764. #element = instance.find_elements(:css => '.content.active .ticketZoom-header .js-objectTitle')[0]
  1765. #element.clear
  1766. #sleep 0.5
  1767. #element = instance.find_elements(:css => '.content.active .ticketZoom-header .js-objectTitle')[0]
  1768. #element.send_keys(data[:title])
  1769. #sleep 0.5
  1770. #element.send_keys(:tab)
  1771. instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").focus()')
  1772. instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").text("' + data[:title] + '")')
  1773. instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").blur()')
  1774. instance.execute_script('$(".content.active .ticketZoom-header .js-objectTitle").trigger("blur")')
  1775. # {
  1776. # :where => :instance2,
  1777. # :execute => 'sendkey',
  1778. # :css => '.content.active .ticketZoom-header .js-objectTitle',
  1779. # :value => 'TTT',
  1780. # },
  1781. # {
  1782. # :where => :instance2,
  1783. # :execute => 'sendkey',
  1784. # :css => '.content.active .ticketZoom-header .js-objectTitle',
  1785. # :value => :tab,
  1786. # },
  1787. end
  1788. if data[:customer]
  1789. # select tab
  1790. click(browser: instance, css: '.content.active .tabsSidebar-tab[data-tab="customer"]')
  1791. click(browser: instance, css: '.content.active div[data-tab="customer"] .js-actions .icon-arrow-down')
  1792. click(browser: instance, css: '.content.active div[data-tab="customer"] .js-actions [data-type="customer-change"]')
  1793. watch_for(
  1794. browser: instance,
  1795. css: '.modal',
  1796. value: 'change',
  1797. )
  1798. element = instance.find_elements(css: '.modal input[name="customer_id_completion"]')[0]
  1799. element.click
  1800. element.clear
  1801. element.send_keys(data[:customer])
  1802. sleep 2.5
  1803. element.send_keys(:enter)
  1804. #instance.find_elements(css: '.modal .user_autocompletion .recipientList-entry.js-object.is-active')[0].click
  1805. sleep 0.2
  1806. click(browser: instance, css: '.modal .js-submit')
  1807. modal_disappear(browser: instance)
  1808. watch_for(
  1809. browser: instance,
  1810. css: '.content.active .tabsSidebar',
  1811. value: data[:customer],
  1812. )
  1813. # select tab
  1814. click(browser: instance, css: '.content.active .tabsSidebar-tab[data-tab="ticket"]')
  1815. end
  1816. if data[:body]
  1817. set(
  1818. browser: instance,
  1819. css: '.content.active div[data-name=body]',
  1820. value: data[:body],
  1821. no_click: true,
  1822. mute_log: true,
  1823. )
  1824. # it's not working stable via selenium, use js
  1825. value = instance.find_elements(css: '.content.active div[data-name=body]')[0].text
  1826. if value != data[:body]
  1827. body_quoted = quote(data[:body])
  1828. instance.execute_script("$('.content.active div[data-name=body]').html('#{body_quoted}').trigger('focusout')")
  1829. end
  1830. end
  1831. if data[:group]
  1832. if data[:group] == '-NONE-'
  1833. # check if owner selection exists
  1834. count = instance.find_elements(css: '.content.active .sidebar select[name="group_id"] option').count
  1835. assert_equal(0, count, 'owner selection should not be showm')
  1836. # check count of agents, should be only 3 / - selection + master + agent on init screen
  1837. count = instance.find_elements(css: '.content.active .sidebar select[name="owner_id"] option').count
  1838. assert_equal(3, count, 'check if owner selection is - selection + master + agent per default')
  1839. else
  1840. select(
  1841. browser: instance,
  1842. css: '.content.active .sidebar select[name="group_id"]',
  1843. value: data[:group],
  1844. mute_log: true,
  1845. )
  1846. sleep 0.2
  1847. end
  1848. end
  1849. if data[:priority]
  1850. select(
  1851. browser: instance,
  1852. css: '.content.active .sidebar select[name="priority_id"]',
  1853. value: data[:priority],
  1854. mute_log: true,
  1855. )
  1856. end
  1857. if data[:state]
  1858. select(
  1859. browser: instance,
  1860. css: '.content.active .sidebar select[name="state_id"]',
  1861. value: data[:state],
  1862. mute_log: true,
  1863. )
  1864. end
  1865. params[:custom_data_select]&.each do |local_key, local_value|
  1866. select(
  1867. browser: instance,
  1868. css: ".active .sidebar select[name=\"#{local_key}\"]",
  1869. value: local_value,
  1870. )
  1871. end
  1872. params[:custom_data_input]&.each do |local_key, local_value|
  1873. set(
  1874. browser: instance,
  1875. css: ".active .sidebar input[name=\"#{local_key}\"]",
  1876. value: local_value,
  1877. clear: true,
  1878. )
  1879. end
  1880. params[:custom_data_date]&.each do |local_key, local_value|
  1881. click(
  1882. browser: instance,
  1883. css: ".active .sidebar div[data-name=\"#{local_key}\"] input[data-item=\"date\"]",
  1884. mute_log: true,
  1885. )
  1886. # weird bug where you cannot "clear" for date/time input
  1887. # this is specific chrome problem, chrome bug report: https://bugs.chromium.org/p/chromedriver/issues/detail?id=1319#c2
  1888. # indirect issue: https://github.com/angular/protractor/issues/562#issuecomment-47745263
  1889. 11.times do
  1890. sendkey(
  1891. value: :backspace,
  1892. )
  1893. end
  1894. set(
  1895. browser: instance,
  1896. css: ".active .sidebar div[data-name=\"#{local_key}\"] input[data-item=\"date\"]",
  1897. value: local_value,
  1898. )
  1899. end
  1900. if data[:state] || data[:group] || data[:body] || params[:custom_data_select].present? || params[:custom_data_input].present?
  1901. found = nil
  1902. 9.times do
  1903. break if found
  1904. begin
  1905. text = instance.find_elements(css: '.content.active .js-reset')[0].text
  1906. if text.match?(/(Discard your unsaved changes.|Verwerfen der)/)
  1907. found = true
  1908. end
  1909. rescue
  1910. # try again
  1911. end
  1912. sleep 1
  1913. end
  1914. if !found
  1915. screenshot(browser: instance, comment: 'ticket_update_discard_message_failed')
  1916. raise 'no discard message found'
  1917. end
  1918. end
  1919. task_type(
  1920. browser: instance,
  1921. type: params[:task_type] || 'stayOnTab',
  1922. )
  1923. if params[:do_not_submit]
  1924. assert(true, 'ticket updated without submit')
  1925. return true
  1926. end
  1927. instance.find_elements(css: '.content.active .js-submit')[0].click
  1928. # do not stay on tab
  1929. if params[:task_type] == 'closeTab' || params[:task_type] == 'closeNextInOverview'
  1930. sleep 1
  1931. screenshot(browser: instance, comment: 'ticket_update')
  1932. return
  1933. end
  1934. 9.times do
  1935. begin
  1936. text = instance.find_elements(css: '.content.active .js-reset')[0].text
  1937. if text.blank?
  1938. screenshot(browser: instance, comment: 'ticket_update_ok')
  1939. sleep 1
  1940. return true
  1941. end
  1942. rescue
  1943. # try again
  1944. end
  1945. sleep 1
  1946. end
  1947. screenshot(browser: instance, comment: 'ticket_update_failed')
  1948. raise 'unable to update ticket'
  1949. end
  1950. =begin
  1951. ticket_verify(
  1952. browser: browser1,
  1953. data: {
  1954. title: 'some title',
  1955. body: 'some body',
  1956. ## group: 'some group',
  1957. ## state: 'closed',
  1958. custom_data_select: {
  1959. key1: 'some value',
  1960. },
  1961. custom_data_input: {
  1962. key1: 'some value',
  1963. },
  1964. },
  1965. )
  1966. =end
  1967. def ticket_verify(params)
  1968. switch_window_focus(params)
  1969. log('ticket_verify', params)
  1970. instance = params[:browser] || @browser
  1971. data = params[:data]
  1972. if data[:title]
  1973. title = instance.find_elements(css: '.content.active .ticketZoom-header .js-objectTitle').first.text.strip
  1974. if title.match?(/#{data[:title]}/i)
  1975. assert(true, "matching '#{data[:title]}' in title '#{title}'")
  1976. else
  1977. raise "not matching '#{data[:title]}' in title '#{title}'"
  1978. end
  1979. end
  1980. if data[:body]
  1981. body = instance.find_elements(css: '.content.active [data-name="body"]').first.text.strip
  1982. if body.match?(/#{data[:body]}/i)
  1983. assert(true, "matching '#{data[:body]}' in body '#{body}'")
  1984. else
  1985. raise "not matching '#{data[:body]}' in body '#{body}'"
  1986. end
  1987. end
  1988. params[:custom_data_select]&.each do |local_key, local_value|
  1989. element = instance.find_elements(css: ".active .sidebar select[name=\"#{local_key}\"] option[selected]").first
  1990. value = element.text.strip
  1991. if value.match?(/#{local_value}/i)
  1992. assert(true, "matching '#{value}' in #{local_key} '#{local_value}'")
  1993. else
  1994. raise "not matching '#{value}' in #{local_key} '#{local_value}'"
  1995. end
  1996. end
  1997. params[:custom_data_input]&.each do |local_key, local_value|
  1998. element = instance.find_elements(css: ".active .sidebar input[name=\"#{local_key}\"]").first
  1999. value = element.text.strip
  2000. if value.match?(/#{local_value}/i)
  2001. assert(true, "matching '#{value}' in #{local_key} '#{local_value}'")
  2002. else
  2003. raise "not matching '#{value}' in #{local_key} '#{local_value}'"
  2004. end
  2005. end
  2006. true
  2007. end
  2008. =begin
  2009. ticket_open_by_overview(
  2010. browser: browser2,
  2011. number: ticket1[:number],
  2012. link: "#ticket/view/#{name}",
  2013. )
  2014. ticket_open_by_overview(
  2015. browser: browser2,
  2016. number: ticket1[:number],
  2017. text: title,
  2018. link: "#ticket/view/#{name}",
  2019. )
  2020. =end
  2021. def ticket_open_by_overview(params)
  2022. switch_window_focus(params)
  2023. log('ticket_open_by_overview', params)
  2024. instance = params[:browser] || @browser
  2025. instance.find_elements(css: '.js-overviewsMenuItem')[0].click
  2026. sleep 1
  2027. execute(
  2028. browser: instance,
  2029. js: '$(".content.active .sidebar").css("display", "block")',
  2030. )
  2031. screenshot(browser: instance, comment: 'ticket_open_by_overview')
  2032. instance.find_elements(css: ".content.active .sidebar a[href=\"#{params[:link]}\"]")[0].click
  2033. sleep 1
  2034. execute(
  2035. browser: instance,
  2036. js: '$(".content.active .sidebar").css("display", "none")',
  2037. )
  2038. screenshot(browser: instance, comment: 'ticket_open_by_overview_search')
  2039. if params[:title]
  2040. element = instance.find_element(css: '.content.active').find_element(partial_link_text: params[:title])
  2041. if !element
  2042. screenshot(browser: instance, comment: 'ticket_open_by_overview_no_ticket_failed')
  2043. raise "unable to find ticket #{params[:title]} in overview #{params[:link]}!"
  2044. end
  2045. else
  2046. element = instance.find_elements(partial_link_text: params[:number])[0]
  2047. if !element
  2048. screenshot(browser: instance, comment: 'ticket_open_by_overview_no_ticket_failed')
  2049. raise "unable to find ticket #{params[:number]} in overview #{params[:link]}!"
  2050. end
  2051. end
  2052. element.click
  2053. sleep 1
  2054. number = instance.find_elements(css: '.content.active .ticketZoom-header .ticket-number')[0].text
  2055. if !number.match?(/#{params[:number]}/)
  2056. screenshot(browser: instance, comment: 'ticket_open_by_overview_open_failed_failed')
  2057. raise "unable to open ticket #{params[:number]}!"
  2058. end
  2059. sleep 1
  2060. assert(true, "ticket #{params[:number]} found")
  2061. true
  2062. end
  2063. =begin
  2064. ticket_open_by_search(
  2065. browser: browser2,
  2066. number: ticket1[:number],
  2067. )
  2068. =end
  2069. def ticket_open_by_search(params)
  2070. switch_window_focus(params)
  2071. log('ticket_open_by_search', params)
  2072. instance = params[:browser] || @browser
  2073. # search by number
  2074. element = instance.find_elements(css: '#global-search')[0]
  2075. element.click
  2076. element.clear
  2077. element.send_keys(params[:number])
  2078. sleep 3
  2079. empty_search(browser: instance)
  2080. # search by number again
  2081. element = instance.find_elements(css: '#global-search')[0]
  2082. element.click
  2083. element.clear
  2084. element.send_keys(params[:number])
  2085. sleep 1
  2086. # open ticket
  2087. screenshot(browser: instance, comment: 'ticket_open_by_search')
  2088. #instance.find_element(partial_link_text: params[:number] } ).click
  2089. instance.execute_script("$(\".js-global-search-result a:contains('#{params[:number]}') .nav-tab-icon\").first().click()")
  2090. sleep 1
  2091. number = instance.find_elements(css: '.content.active .ticketZoom-header .ticket-number')[0].text
  2092. if !number.match?(/#{params[:number]}/)
  2093. screenshot(browser: instance, comment: 'ticket_open_by_search_failed')
  2094. raise "unable to search/find ticket #{params[:number]}!"
  2095. end
  2096. sleep 1
  2097. true
  2098. end
  2099. =begin
  2100. ticket_open_by_title(
  2101. browser: browser2,
  2102. title: ticket1[:title],
  2103. )
  2104. =end
  2105. def ticket_open_by_title(params)
  2106. switch_window_focus(params)
  2107. log('ticket_open_by_title', params)
  2108. instance = params[:browser] || @browser
  2109. # search by number
  2110. element = instance.find_elements(css: '#global-search')[0]
  2111. element.click
  2112. element.clear
  2113. element.send_keys(params[:title])
  2114. sleep 3
  2115. # open ticket
  2116. screenshot(browser: instance, comment: 'ticket_open_by_title_search')
  2117. #instance.find_element(partial_link_text: params[:title] } ).click
  2118. instance.execute_script("$(\".js-global-search-result a:contains('#{params[:title]}') .nav-tab-icon\").click()")
  2119. sleep 1
  2120. title = instance.find_elements(css: '.content.active .ticketZoom-header .js-objectTitle')[0].text
  2121. if !title.match?(/#{params[:title]}/)
  2122. screenshot(browser: instance, comment: 'ticket_open_by_title_failed')
  2123. raise "unable to search/find ticket #{params[:title]}!"
  2124. end
  2125. sleep 1
  2126. true
  2127. end
  2128. =begin
  2129. overview_count = overview_counter(
  2130. browser: browser2,
  2131. )
  2132. returns
  2133. {
  2134. '#ticket/view/all_unassigned' => 42,
  2135. }
  2136. =end
  2137. def overview_counter(params = {})
  2138. switch_window_focus(params)
  2139. log('overview_counter', params)
  2140. instance = params[:browser] || @browser
  2141. instance.find_elements(css: '.js-overviewsMenuItem')[0].click
  2142. sleep 2
  2143. execute(
  2144. browser: instance,
  2145. js: '$(".content.active .sidebar").css("display", "block")',
  2146. )
  2147. #execute(
  2148. # browser: instance,
  2149. # js: '$(".content.active .overview-header").css("display", "none")',
  2150. #)
  2151. overviews = {}
  2152. instance.find_elements(css: '.content.active .sidebar a[href]').each do |element|
  2153. url = element.attribute('href')
  2154. url.gsub!(%r{(http|https)://.+?/(.+?)$}, '\\2')
  2155. overviews[url] = 0
  2156. #puts url.inspect
  2157. #puts element.inspect
  2158. end
  2159. overviews.each_key do |url|
  2160. count = instance.find_elements(css: ".content.active .sidebar a[href=\"#{url}\"] .badge")[0].text
  2161. overviews[url] = count.to_i
  2162. end
  2163. log('overview_counter', overviews)
  2164. overviews
  2165. end
  2166. =begin
  2167. organization_open_by_search(
  2168. browser: browser2,
  2169. value: 'some value',
  2170. )
  2171. =end
  2172. def organization_open_by_search(params = {})
  2173. switch_window_focus(params)
  2174. log('organization_open_by_search', params)
  2175. instance = params[:browser] || @browser
  2176. element = instance.find_elements(css: '#global-search')[0]
  2177. element.click
  2178. element.clear
  2179. element.send_keys(params[:value])
  2180. sleep 3
  2181. empty_search(browser: instance)
  2182. element = instance.find_elements(css: '#global-search')[0]
  2183. element.click
  2184. element.clear
  2185. element.send_keys(params[:value])
  2186. sleep 2
  2187. #instance.find_element(partial_link_text: params[:value] } ).click
  2188. instance.execute_script("$(\".js-global-search-result a:contains('#{params[:value]}') .nav-tab-icon\").click()")
  2189. sleep 1
  2190. name = instance.find_elements(css: '.content.active h1')[0].text
  2191. if !name.match?(/#{params[:value]}/)
  2192. screenshot(browser: instance, comment: 'organization_open_by_search_failed')
  2193. raise "unable to search/find org #{params[:value]}!"
  2194. end
  2195. assert(true, "org #{params[:value]} found")
  2196. sleep 2
  2197. true
  2198. end
  2199. =begin
  2200. user_open_by_search(
  2201. browser: browser2,
  2202. value: 'some value',
  2203. )
  2204. =end
  2205. def user_open_by_search(params = {})
  2206. switch_window_focus(params)
  2207. log('user_open_by_search', params)
  2208. instance = params[:browser] || @browser
  2209. element = instance.find_elements(css: '#global-search')[0]
  2210. element.click
  2211. element.clear
  2212. element.send_keys(params[:value])
  2213. sleep 3
  2214. screenshot(browser: instance, comment: 'user_open_by_search')
  2215. #instance.find_element(partial_link_text: params[:value]).click
  2216. instance.execute_script("$(\".js-global-search-result a:contains('#{params[:value]}') .nav-tab-icon\").click()")
  2217. sleep 1
  2218. name = instance.find_elements(css: '.content.active h1')[0].text
  2219. if !name.match?(/#{params[:value]}/)
  2220. screenshot(browser: instance, comment: 'user_open_by_search_failed')
  2221. raise "unable to search/find user #{params[:value]}!"
  2222. end
  2223. assert(true, "user #{params[:term]} found")
  2224. sleep 2
  2225. true
  2226. end
  2227. =begin
  2228. user_create(
  2229. browser: browser2,
  2230. data: {
  2231. #login: 'some login' + random,
  2232. firstname: 'Manage Firstname' + random,
  2233. lastname: 'Manage Lastname' + random,
  2234. email: user_email,
  2235. password: 'some-pass',
  2236. },
  2237. )
  2238. =end
  2239. def user_create(params = {})
  2240. switch_window_focus(params)
  2241. log('user_create', params)
  2242. instance = params[:browser] || @browser
  2243. data = params[:data]
  2244. click(
  2245. browser: instance,
  2246. css: 'a[href="#manage"]',
  2247. mute_log: true,
  2248. )
  2249. click(
  2250. browser: instance,
  2251. css: '.content.active a[href="#manage/users"]',
  2252. mute_log: true,
  2253. )
  2254. click(
  2255. browser: instance,
  2256. css: '.content.active a[data-type="new"]',
  2257. mute_log: true,
  2258. )
  2259. modal_ready(browser: instance)
  2260. element = instance.find_elements(css: '.modal input[name=firstname]')[0]
  2261. element.clear
  2262. element.send_keys(data[:firstname])
  2263. element = instance.find_elements(css: '.modal input[name=lastname]')[0]
  2264. element.clear
  2265. element.send_keys(data[:lastname])
  2266. element = instance.find_elements(css: '.modal input[name=email]')[0]
  2267. element.clear
  2268. element.send_keys(data[:email])
  2269. element = instance.find_elements(css: '.modal input[name=password]')[0]
  2270. element.clear
  2271. element.send_keys(data[:password])
  2272. element = instance.find_elements(css: '.modal input[name=password_confirm]')[0]
  2273. element.clear
  2274. element.send_keys(data[:password])
  2275. check(
  2276. browser: instance,
  2277. css: '.modal input[name=role_ids][value=3]',
  2278. )
  2279. instance.find_elements(css: '.modal button.js-submit')[0].click
  2280. modal_disappear(
  2281. browser: instance,
  2282. timeout: 10,
  2283. )
  2284. set(
  2285. browser: instance,
  2286. css: '.content .js-search',
  2287. value: data[:email],
  2288. )
  2289. watch_for(
  2290. browser: instance,
  2291. css: 'body',
  2292. value: data[:lastname],
  2293. )
  2294. assert(true, 'user created')
  2295. end
  2296. =begin
  2297. calendar_create(
  2298. browser: browser2,
  2299. data: {
  2300. name: 'some calendar' + random,
  2301. first_response_time_in_text: 61
  2302. },
  2303. )
  2304. =end
  2305. def calendar_create(params = {})
  2306. switch_window_focus(params)
  2307. log('calendar_create', params)
  2308. instance = params[:browser] || @browser
  2309. data = params[:data]
  2310. click(
  2311. browser: instance,
  2312. css: 'a[href="#manage"]',
  2313. mute_log: true,
  2314. )
  2315. click(
  2316. browser: instance,
  2317. css: '.content.active a[href="#manage/calendars"]',
  2318. mute_log: true,
  2319. )
  2320. sleep 4
  2321. click(
  2322. browser: instance,
  2323. css: '.content.active a.js-new',
  2324. mute_log: true,
  2325. )
  2326. modal_ready(browser: instance)
  2327. element = instance.find_elements(css: '.content.active .modal input[name=name]')[0]
  2328. element.clear
  2329. element.send_keys(data[:name])
  2330. element = instance.find_elements(css: '.content.active .modal .js-input')[0]
  2331. element.clear
  2332. element.send_keys(data[:timezone])
  2333. element.send_keys(:enter)
  2334. instance.find_elements(css: '.modal button.js-submit')[0].click
  2335. modal_disappear(browser: instance)
  2336. 7.times do
  2337. element = instance.find_elements(css: 'body')[0]
  2338. text = element.text
  2339. if text.match?(/#{Regexp.quote(data[:name])}/)
  2340. assert(true, 'calendar created')
  2341. sleep 1
  2342. return true
  2343. end
  2344. sleep 1
  2345. end
  2346. screenshot(browser: instance, comment: 'calendar_create_failed')
  2347. raise 'calendar creation failed'
  2348. end
  2349. =begin
  2350. sla_create(
  2351. browser: browser2,
  2352. data: {
  2353. name: 'some sla' + random,
  2354. calendar: 'some calendar name',
  2355. first_response_time_in_text: 61
  2356. },
  2357. )
  2358. =end
  2359. def sla_create(params = {})
  2360. switch_window_focus(params)
  2361. log('sla_create', params)
  2362. instance = params[:browser] || @browser
  2363. data = params[:data]
  2364. click(
  2365. browser: instance,
  2366. css: 'a[href="#manage"]',
  2367. mute_log: true,
  2368. )
  2369. click(
  2370. browser: instance,
  2371. css: '.content.active a[href="#manage/slas"]',
  2372. mute_log: true,
  2373. )
  2374. click(
  2375. browser: instance,
  2376. css: '.content.active a.js-new',
  2377. mute_log: true,
  2378. )
  2379. modal_ready(browser: instance)
  2380. element = instance.find_elements(css: '.modal input[name=name]')[0]
  2381. element.clear
  2382. element.send_keys(data[:name])
  2383. if data[:calendar].present?
  2384. element = instance.find_elements(css: '.modal select[name="calendar_id"]')[0]
  2385. dropdown = Selenium::WebDriver::Support::Select.new(element)
  2386. dropdown.select_by(:text, data[:calendar])
  2387. end
  2388. element = instance.find_elements(css: '.modal input[name=first_response_time_in_text]')[0]
  2389. element.clear
  2390. element.send_keys(data[:first_response_time_in_text])
  2391. instance.find_elements(css: '.modal button.js-submit')[0].click
  2392. modal_disappear(browser: instance)
  2393. 7.times do
  2394. element = instance.find_elements(css: 'body')[0]
  2395. text = element.text
  2396. if text.match?(/#{Regexp.quote(data[:name])}/)
  2397. assert(true, 'sla created')
  2398. sleep 1
  2399. return true
  2400. end
  2401. sleep 1
  2402. end
  2403. screenshot(browser: instance, comment: 'sla_create_failed')
  2404. raise 'sla creation failed'
  2405. end
  2406. =begin
  2407. text_module_create(
  2408. browser: browser2,
  2409. data: {
  2410. name: 'some sla' + random,
  2411. keywords: 'some keywords',
  2412. content: 'some content',
  2413. },
  2414. )
  2415. =end
  2416. def text_module_create(params = {})
  2417. switch_window_focus(params)
  2418. log('text_module_create', params)
  2419. instance = params[:browser] || @browser
  2420. data = params[:data]
  2421. click(
  2422. browser: instance,
  2423. css: 'a[href="#manage"]',
  2424. mute_log: true,
  2425. )
  2426. click(
  2427. browser: instance,
  2428. css: '.content.active a[href="#manage/text_modules"]',
  2429. mute_log: true,
  2430. )
  2431. click(
  2432. browser: instance,
  2433. css: '.content.active a[data-type="new"]',
  2434. mute_log: true,
  2435. )
  2436. modal_ready(browser: instance)
  2437. set(
  2438. browser: instance,
  2439. css: '.modal input[name=name]',
  2440. value: data[:name],
  2441. )
  2442. set(
  2443. browser: instance,
  2444. css: '.modal input[name=keywords]',
  2445. value: data[:keywords],
  2446. )
  2447. set(
  2448. browser: instance,
  2449. css: '.modal [data-name=content]',
  2450. value: data[:content],
  2451. )
  2452. instance.find_elements(css: '.modal button.js-submit')[0].click
  2453. modal_disappear(browser: instance)
  2454. 7.times do
  2455. element = instance.find_elements(css: 'body')[0]
  2456. text = element.text
  2457. if text.match?(/#{Regexp.quote(data[:name])}/)
  2458. assert(true, 'text module created')
  2459. sleep 1
  2460. return true
  2461. end
  2462. sleep 1
  2463. end
  2464. screenshot(browser: instance, comment: 'text_module_create_failed')
  2465. raise 'text module creation failed'
  2466. end
  2467. =begin
  2468. signature_create(
  2469. browser: browser2,
  2470. data: {
  2471. name: 'some sla' + random,
  2472. body: 'some body',
  2473. },
  2474. )
  2475. =end
  2476. def signature_create(params = {})
  2477. switch_window_focus(params)
  2478. log('signature_create', params)
  2479. instance = params[:browser] || @browser
  2480. data = params[:data]
  2481. click(
  2482. browser: instance,
  2483. css: 'a[href="#manage"]',
  2484. mute_log: true,
  2485. )
  2486. click(
  2487. browser: instance,
  2488. css: '.content.active a[href="#channels/email"]',
  2489. mute_log: true,
  2490. )
  2491. click(
  2492. browser: instance,
  2493. css: '.content.active a[href="#c-signature"]',
  2494. mute_log: true,
  2495. )
  2496. sleep 4
  2497. click(
  2498. browser: instance,
  2499. css: '.content.active #c-signature a[data-type="new"]',
  2500. mute_log: true,
  2501. )
  2502. modal_ready(browser: instance)
  2503. set(
  2504. browser: instance,
  2505. css: '.modal input[name=name]',
  2506. value: data[:name],
  2507. )
  2508. set(
  2509. browser: instance,
  2510. css: '.modal [data-name=body]',
  2511. value: data[:body],
  2512. )
  2513. instance.find_elements(css: '.modal button.js-submit')[0].click
  2514. modal_disappear(browser: instance)
  2515. 11.times do
  2516. element = instance.find_elements(css: 'body')[0]
  2517. text = element.text
  2518. if text.match?(/#{Regexp.quote(data[:name])}/)
  2519. assert(true, 'signature created')
  2520. sleep 1
  2521. return true
  2522. end
  2523. sleep 1
  2524. end
  2525. screenshot(browser: instance, comment: 'signature_create_failed')
  2526. raise 'signature creation failed'
  2527. end
  2528. =begin
  2529. group_create(
  2530. browser: browser2,
  2531. data: {
  2532. name: 'some sla' + random,
  2533. signature: 'some signature bame',
  2534. member: [
  2535. {
  2536. login: 'some_user_login',
  2537. access: 'all',
  2538. },
  2539. ],
  2540. },
  2541. )
  2542. =end
  2543. def group_create(params = {})
  2544. switch_window_focus(params)
  2545. log('group_create', params)
  2546. instance = params[:browser] || @browser
  2547. data = params[:data]
  2548. click(
  2549. browser: instance,
  2550. css: 'a[href="#manage"]',
  2551. mute_log: true,
  2552. )
  2553. click(
  2554. browser: instance,
  2555. css: '.content.active a[href="#manage/groups"]',
  2556. mute_log: true,
  2557. )
  2558. click(
  2559. browser: instance,
  2560. css: '.content.active a[data-type="new"]',
  2561. mute_log: true,
  2562. )
  2563. modal_ready(browser: instance)
  2564. element = instance.find_elements(css: '.modal input[name=name]')[0]
  2565. element.clear
  2566. element.send_keys(data[:name])
  2567. element = instance.find_elements(css: '.modal select[name="email_address_id"]')[0]
  2568. dropdown = Selenium::WebDriver::Support::Select.new(element)
  2569. dropdown.select_by(:index, 1)
  2570. #dropdown.select_by(:text, action[:group])
  2571. if data[:signature]
  2572. element = instance.find_elements(css: '.modal select[name="signature_id"]')[0]
  2573. dropdown = Selenium::WebDriver::Support::Select.new(element)
  2574. dropdown.select_by(:text, data[:signature])
  2575. end
  2576. instance.find_elements(css: '.modal button.js-submit')[0].click
  2577. modal_disappear(browser: instance)
  2578. 11.times do
  2579. element = instance.find_elements(css: 'body')[0]
  2580. text = element.text
  2581. if text.match?(/#{Regexp.quote(data[:name])}/)
  2582. assert(true, 'group created')
  2583. modal_disappear(browser: instance) # wait until modal has gone
  2584. # add member
  2585. data[:member]&.each do |member|
  2586. instance.find_elements(css: 'a[href="#manage"]')[0].click
  2587. sleep 1
  2588. instance.find_elements(css: '.content.active a[href="#manage/users"]')[0].click
  2589. sleep 3
  2590. element = instance.find_elements(css: '.content.active [name="search"]')[0]
  2591. element.clear
  2592. element.send_keys(member[:login])
  2593. sleep 3
  2594. #instance.find_elements(:css => '.content.active table [data-id]')[0].click
  2595. instance.execute_script('$(".content.active table [data-id] td").first().click()')
  2596. modal_ready(browser: instance)
  2597. #instance.find_elements(:css => 'label:contains(" ' + action[:name] + '")')[0].click
  2598. instance.execute_script('$(".js-groupList tr:contains(\"' + data[:name] + '\") .js-groupListItem[value=' + member[:access] + ']").prop("checked", true)')
  2599. screenshot(browser: instance, comment: 'group_create_member')
  2600. instance.find_elements(css: '.modal button.js-submit')[0].click
  2601. modal_disappear(browser: instance)
  2602. end
  2603. end
  2604. sleep 1
  2605. return true
  2606. end
  2607. screenshot(browser: instance, comment: 'group_create_failed')
  2608. raise 'group creation failed'
  2609. end
  2610. =begin
  2611. role_create(
  2612. browser: browser2,
  2613. data: {
  2614. name: 'some role' + random,
  2615. default_at_signup: false,
  2616. permission: {
  2617. 'admin.group' => true,
  2618. 'preferences.password' => true,
  2619. },
  2620. member: [
  2621. 'some_user_login',
  2622. ],
  2623. },
  2624. )
  2625. =end
  2626. def role_create(params = {})
  2627. switch_window_focus(params)
  2628. log('role_create', params)
  2629. instance = params[:browser] || @browser
  2630. data = params[:data]
  2631. click(
  2632. browser: instance,
  2633. css: 'a[href="#manage"]',
  2634. mute_log: true,
  2635. )
  2636. click(
  2637. browser: instance,
  2638. css: '.content.active a[href="#manage/roles"]',
  2639. mute_log: true,
  2640. )
  2641. click(
  2642. browser: instance,
  2643. css: '.content.active a[data-type="new"]',
  2644. mute_log: true,
  2645. )
  2646. modal_ready(browser: instance)
  2647. element = instance.find_elements(css: '.modal input[name=name]')[0]
  2648. element.clear
  2649. element.send_keys(data[:name])
  2650. if data.key?(:default_at_signup)
  2651. element = instance.find_elements(css: '.modal select[name="default_at_signup"]')[0]
  2652. dropdown = Selenium::WebDriver::Support::Select.new(element)
  2653. if data[:default_at_signup] == true
  2654. dropdown.select_by(:text, 'yes')
  2655. else
  2656. dropdown.select_by(:text, 'no')
  2657. end
  2658. end
  2659. if data.key?(:permission)
  2660. data[:permission].each do |permission_name, permission_value|
  2661. if permission_value == false
  2662. uncheck(
  2663. browser: instance,
  2664. css: ".modal [data-permission-name=\"#{permission_name}\"]",
  2665. )
  2666. else
  2667. check(
  2668. browser: instance,
  2669. css: ".modal [data-permission-name=\"#{permission_name}\"]",
  2670. )
  2671. end
  2672. end
  2673. end
  2674. instance.find_elements(css: '.modal button.js-submit')[0].click
  2675. modal_disappear(browser: instance)
  2676. 11.times do
  2677. element = instance.find_elements(css: 'body')[0]
  2678. text = element.text
  2679. if text.match?(/#{Regexp.quote(data[:name])}/)
  2680. assert(true, 'role created')
  2681. modal_disappear(browser: instance) # wait until modal has gone
  2682. # add member
  2683. data[:member]&.each do |login|
  2684. instance.find_elements(css: 'a[href="#manage"]')[0].click
  2685. sleep 1
  2686. instance.find_elements(css: '.content.active a[href="#manage/users"]')[0].click
  2687. sleep 3
  2688. element = instance.find_elements(css: '.content.active [name="search"]')[0]
  2689. element.clear
  2690. element.send_keys(login)
  2691. sleep 3
  2692. #instance.find_elements(:css => '.content.active table [data-id]')[0].click
  2693. instance.execute_script('$(".content.active table [data-id] td").first().click()')
  2694. sleep 3
  2695. #instance.find_elements(:css => 'label:contains(" ' + action[:name] + '")')[0].click
  2696. instance.execute_script('$(\'label:contains(" ' + data[:name] + '")\').first().click()')
  2697. instance.find_elements(css: '.modal button.js-submit')[0].click
  2698. modal_disappear(browser: instance)
  2699. end
  2700. end
  2701. sleep 1
  2702. return true
  2703. end
  2704. screenshot(browser: instance, comment: 'role_create_failed')
  2705. raise 'role creation failed'
  2706. end
  2707. =begin
  2708. role_create(
  2709. browser: browser2,
  2710. data: {
  2711. name: 'some role' + random,
  2712. default_at_signup: false,
  2713. permission: {
  2714. 'admin.group' => true,
  2715. 'preferences.password' => true,
  2716. },
  2717. member: [
  2718. 'some_user_login',
  2719. ],
  2720. },
  2721. )
  2722. =end
  2723. def role_edit(params = {})
  2724. switch_window_focus(params)
  2725. log('role_edit', params)
  2726. instance = params[:browser] || @browser
  2727. data = params[:data]
  2728. click(
  2729. browser: instance,
  2730. css: 'a[href="#manage"]',
  2731. mute_log: true,
  2732. )
  2733. click(
  2734. browser: instance,
  2735. css: '.content.active a[href="#manage/roles"]',
  2736. mute_log: true,
  2737. )
  2738. instance.execute_script('$(\'.content.active table tr td:contains(" ' + data[:name] + '")\').first().click()')
  2739. modal_ready(browser: instance)
  2740. element = instance.find_elements(css: '.modal input[name=name]')[0]
  2741. element.clear
  2742. element.send_keys(data[:name])
  2743. if data.key?(:default_at_signup)
  2744. element = instance.find_elements(css: '.modal select[name="default_at_signup"]')[0]
  2745. dropdown = Selenium::WebDriver::Support::Select.new(element)
  2746. if data[:default_at_signup] == true
  2747. dropdown.select_by(:text, 'yes')
  2748. else
  2749. dropdown.select_by(:text, 'no')
  2750. end
  2751. end
  2752. if data.key?(:permission)
  2753. data[:permission].each do |permission_name, permission_value|
  2754. if permission_value == false
  2755. uncheck(
  2756. browser: instance,
  2757. css: ".modal [data-permission-name=\"#{permission_name}\"]",
  2758. )
  2759. else
  2760. check(
  2761. browser: instance,
  2762. css: ".modal [data-permission-name=\"#{permission_name}\"]",
  2763. )
  2764. end
  2765. end
  2766. end
  2767. if data.key?(:active)
  2768. element = instance.find_elements(css: '.modal select[name="active"]')[0]
  2769. dropdown = Selenium::WebDriver::Support::Select.new(element)
  2770. if data[:active] == true
  2771. dropdown.select_by(:text, 'active')
  2772. else
  2773. dropdown.select_by(:text, 'inactive')
  2774. end
  2775. end
  2776. instance.find_elements(css: '.modal button.js-submit')[0].click
  2777. modal_disappear(browser: instance)
  2778. 11.times do
  2779. element = instance.find_elements(css: 'body')[0]
  2780. text = element.text
  2781. if text.match?(/#{Regexp.quote(data[:name])}/)
  2782. assert(true, 'role created')
  2783. modal_disappear(browser: instance) # wait until modal has gone
  2784. # add member
  2785. data[:member]&.each do |login|
  2786. instance.find_elements(css: 'a[href="#manage"]')[0].click
  2787. sleep 1
  2788. instance.find_elements(css: '.content.active a[href="#manage/users"]')[0].click
  2789. sleep 3
  2790. element = instance.find_elements(css: '.content.active [name="search"]')[0]
  2791. element.clear
  2792. element.send_keys(login)
  2793. sleep 3
  2794. #instance.find_elements(:css => '.content.active table [data-id]')[0].click
  2795. instance.execute_script('$(".content.active table [data-id] td").first().click()')
  2796. sleep 3
  2797. #instance.find_elements(:css => 'label:contains(" ' + action[:name] + '")')[0].click
  2798. instance.execute_script('$(\'label:contains(" ' + data[:name] + '")\').first().click()')
  2799. instance.find_elements(css: '.modal button.js-submit')[0].click
  2800. modal_disappear(browser: instance)
  2801. end
  2802. end
  2803. sleep 1
  2804. return true
  2805. end
  2806. screenshot(browser: instance, comment: 'role_edit_failed')
  2807. raise 'role edit failed'
  2808. end
  2809. =begin
  2810. object_manager_attribute_create(
  2811. browser: browser2,
  2812. data: {
  2813. name: 'field_name' + random,
  2814. display: 'Display Name of Field',
  2815. data_type: 'Select',
  2816. data_option: {
  2817. options: {
  2818. 'aa' => 'AA',
  2819. 'bb' => 'BB',
  2820. },
  2821. default: 'abc',
  2822. },
  2823. },
  2824. error: 'already exists'
  2825. )
  2826. object_manager_attribute_create(
  2827. browser: browser2,
  2828. data: {
  2829. name: 'field_name' + random,
  2830. display: 'Display Name of Field',
  2831. data_type: 'Text',
  2832. data_option: {
  2833. default: 'abc',
  2834. },
  2835. },
  2836. error: 'already exists'
  2837. )
  2838. object_manager_attribute_create(
  2839. browser: browser2,
  2840. data: {
  2841. name: 'field_name' + random,
  2842. display: 'Display Name of Field',
  2843. data_type: 'Integer',
  2844. data_option: {
  2845. default: '15',
  2846. min: 1,
  2847. max: 999999,
  2848. },
  2849. },
  2850. error: 'already exists'
  2851. )
  2852. object_manager_attribute_create(
  2853. browser: browser2,
  2854. data: {
  2855. name: 'field_name' + random,
  2856. display: 'Display Name of Field',
  2857. data_type: 'Datetime',
  2858. data_option: {
  2859. future: true,
  2860. past: true,
  2861. diff: 24,
  2862. },
  2863. },
  2864. error: 'already exists'
  2865. )
  2866. object_manager_attribute_create(
  2867. browser: browser2,
  2868. data: {
  2869. name: 'field_name' + random,
  2870. display: 'Display Name of Field',
  2871. data_type: 'Date',
  2872. data_option: {
  2873. future: true,
  2874. past: true,
  2875. diff: 24,
  2876. },
  2877. },
  2878. error: 'already exists'
  2879. )
  2880. object_manager_attribute_create(
  2881. browser: browser2,
  2882. data: {
  2883. name: 'field_name' + random,
  2884. display: 'Display Name of Field',
  2885. data_type: 'Boolean',
  2886. data_option: {
  2887. options: {
  2888. true: 'YES',
  2889. false: 'NO',
  2890. }
  2891. default: undefined,
  2892. },
  2893. },
  2894. error: 'already exists'
  2895. )
  2896. =end
  2897. def object_manager_attribute_create(params = {})
  2898. switch_window_focus(params)
  2899. log('object_manager_attribute_create', params)
  2900. instance = params[:browser] || @browser
  2901. data = params[:data]
  2902. click(
  2903. browser: instance,
  2904. css: 'a[href="#manage"]',
  2905. mute_log: true,
  2906. )
  2907. click(
  2908. browser: instance,
  2909. css: '.content.active a[href="#system/object_manager"]',
  2910. mute_log: true,
  2911. )
  2912. sleep 4
  2913. click(
  2914. browser: instance,
  2915. css: '.content.active .js-new',
  2916. mute_log: true,
  2917. )
  2918. modal_ready(browser: instance)
  2919. element = instance.find_elements(css: '.modal input[name=name]')[0]
  2920. element.clear
  2921. element.send_keys(data[:name])
  2922. element = instance.find_elements(css: '.modal input[name=display]')[0]
  2923. element.clear
  2924. element.send_keys(data[:display])
  2925. select(
  2926. browser: instance,
  2927. css: '.modal select[name="data_type"]',
  2928. value: data[:data_type],
  2929. mute_log: true,
  2930. )
  2931. if data[:data_option]
  2932. if data[:data_option][:options]
  2933. if data[:data_type] == 'Boolean'
  2934. # rubocop:disable Lint/BooleanSymbol
  2935. element = instance.find_elements(css: '.modal .js-valueTrue').first
  2936. element.clear
  2937. element.send_keys(data[:data_option][:options][:true])
  2938. element = instance.find_elements(css: '.modal .js-valueFalse').first
  2939. element.clear
  2940. element.send_keys(data[:data_option][:options][:false])
  2941. # rubocop:enable Lint/BooleanSymbol
  2942. elsif data[:data_type] == 'Tree Select'
  2943. add_tree_options(
  2944. instance: instance,
  2945. options: data[:data_option][:options],
  2946. )
  2947. else
  2948. data[:data_option][:options].each do |key, value|
  2949. element = instance.find_elements(css: '.modal .js-Table .js-key').last
  2950. element.clear
  2951. element.send_keys(key)
  2952. element = instance.find_elements(css: '.modal .js-Table .js-value').last
  2953. element.clear
  2954. element.send_keys(value)
  2955. element = instance.find_elements(css: '.modal .js-Table .js-add')[0]
  2956. element.click
  2957. end
  2958. end
  2959. end
  2960. %i[default min max diff].each do |key|
  2961. next if !data[:data_option].key?(key)
  2962. element = instance.find_elements(css: ".modal [name=\"data_option::#{key}\"]").first
  2963. element.clear
  2964. element.send_keys(data[:data_option][key])
  2965. end
  2966. %i[future past].each do |key|
  2967. next if !data[:data_option].key?(key)
  2968. select(
  2969. browser: instance,
  2970. css: ".modal select[name=\"data_option::#{key}\"]",
  2971. value: data[:data_option][key],
  2972. mute_log: true,
  2973. )
  2974. end
  2975. end
  2976. instance.find_elements(css: '.modal button.js-submit')[0].click
  2977. if params[:error]
  2978. sleep 4
  2979. watch_for(
  2980. css: '.modal',
  2981. value: params[:error],
  2982. )
  2983. click(
  2984. browser: instance,
  2985. css: '.modal .js-close',
  2986. )
  2987. modal_disappear(browser: instance)
  2988. return
  2989. end
  2990. 11.times do
  2991. element = instance.find_elements(css: 'body')[0]
  2992. text = element.text
  2993. if text.match?(/#{Regexp.quote(data[:name])}/)
  2994. assert(true, 'object manager attribute created')
  2995. sleep 1
  2996. return true
  2997. end
  2998. sleep 1
  2999. end
  3000. screenshot(browser: instance, comment: 'object_manager_attribute_create_failed')
  3001. raise 'object manager attribute creation failed'
  3002. end
  3003. =begin
  3004. object_manager_attribute_update(
  3005. browser: browser2,
  3006. data: {
  3007. name: 'field_name' + random,
  3008. display: 'Display Name of Field',
  3009. data_type: 'Select',
  3010. data_option: {
  3011. options: {
  3012. 'aa' => 'AA',
  3013. 'bb' => 'BB',
  3014. },
  3015. default: 'abc',
  3016. },
  3017. },
  3018. error: 'already exists'
  3019. )
  3020. =end
  3021. def object_manager_attribute_update(params = {})
  3022. switch_window_focus(params)
  3023. log('object_manager_attribute_update', params)
  3024. instance = params[:browser] || @browser
  3025. data = params[:data]
  3026. click(
  3027. browser: instance,
  3028. css: 'a[href="#manage"]',
  3029. mute_log: true,
  3030. )
  3031. click(
  3032. browser: instance,
  3033. css: '.content.active a[href="#system/object_manager"]',
  3034. mute_log: true,
  3035. )
  3036. sleep 4
  3037. instance.execute_script("$(\".content.active td:contains('#{data[:name]}')\").first().click()")
  3038. modal_ready(browser: instance)
  3039. element = instance.find_elements(css: '.modal input[name=display]')[0]
  3040. element.clear
  3041. element.send_keys(data[:display])
  3042. select(
  3043. browser: instance,
  3044. css: '.modal select[name="data_type"]',
  3045. value: data[:data_type],
  3046. mute_log: true,
  3047. )
  3048. if data[:data_option]
  3049. if data[:data_option][:options]
  3050. if data[:data_type] == 'Boolean'
  3051. # rubocop:disable Lint/BooleanSymbol
  3052. element = instance.find_elements(css: '.modal .js-valueTrue').first
  3053. element.clear
  3054. element.send_keys(data[:data_option][:options][:true])
  3055. element = instance.find_elements(css: '.modal .js-valueFalse').first
  3056. element.clear
  3057. element.send_keys(data[:data_option][:options][:false])
  3058. # rubocop:enable Lint/BooleanSymbol
  3059. else
  3060. data[:data_option][:options].each do |key, value|
  3061. element = instance.find_elements(css: '.modal .js-Table .js-key').last
  3062. element.clear
  3063. element.send_keys(key)
  3064. element = instance.find_elements(css: '.modal .js-Table .js-value').last
  3065. element.clear
  3066. element.send_keys(value)
  3067. element = instance.find_elements(css: '.modal .js-Table .js-add')[0]
  3068. element.click
  3069. end
  3070. end
  3071. end
  3072. %i[default min max diff].each do |key|
  3073. next if !data[:data_option].key?(key)
  3074. element = instance.find_elements(css: ".modal [name=\"data_option::#{key}\"]").first
  3075. element.clear
  3076. element.send_keys(data[:data_option][key])
  3077. end
  3078. %i[future past].each do |key|
  3079. next if !data[:data_option].key?(key)
  3080. select(
  3081. browser: instance,
  3082. css: ".modal select[name=\"data_option::#{key}\"]",
  3083. value: data[:data_option][key],
  3084. mute_log: true,
  3085. )
  3086. end
  3087. end
  3088. instance.find_elements(css: '.modal button.js-submit')[0].click
  3089. if params[:error]
  3090. sleep 4
  3091. watch_for(
  3092. css: '.modal',
  3093. value: params[:error],
  3094. )
  3095. click(
  3096. browser: instance,
  3097. css: '.modal .js-close',
  3098. )
  3099. modal_disappear(browser: instance)
  3100. return
  3101. end
  3102. 11.times do
  3103. element = instance.find_elements(css: 'body')[0]
  3104. text = element.text
  3105. if text.match?(/#{Regexp.quote(data[:name])}/)
  3106. assert(true, 'object manager attribute updated')
  3107. sleep 1
  3108. return true
  3109. end
  3110. sleep 1
  3111. end
  3112. screenshot(browser: instance, comment: 'object_manager_attribute_update_failed')
  3113. raise 'object manager attribute update failed'
  3114. end
  3115. =begin
  3116. object_manager_attribute_delete(
  3117. browser: browser2,
  3118. data: {
  3119. name: 'field_name' + random,
  3120. },
  3121. )
  3122. =end
  3123. def object_manager_attribute_delete(params = {})
  3124. switch_window_focus(params)
  3125. log('object_manager_attribute_delete', params)
  3126. instance = params[:browser] || @browser
  3127. data = params[:data]
  3128. click(
  3129. browser: instance,
  3130. css: 'a[href="#manage"]',
  3131. mute_log: true,
  3132. )
  3133. click(
  3134. browser: instance,
  3135. css: '.content.active a[href="#system/object_manager"]',
  3136. mute_log: true,
  3137. )
  3138. sleep 4
  3139. instance = params[:browser] || @browser
  3140. data = params[:data]
  3141. r = instance.execute_script("$(\".content.active td:contains('#{data[:name]}')\").first().closest('tr').find('.js-delete').click()")
  3142. #p "rrr #{r.inspect}"
  3143. end
  3144. =begin
  3145. object_manager_attribute_discard_changes(
  3146. browser: browser2,
  3147. )
  3148. =end
  3149. def object_manager_attribute_discard_changes(params = {})
  3150. switch_window_focus(params)
  3151. log('object_manager_attribute_discard_changes', params)
  3152. instance = params[:browser] || @browser
  3153. click(
  3154. browser: instance,
  3155. css: 'a[href="#manage"]',
  3156. mute_log: true,
  3157. )
  3158. click(
  3159. browser: instance,
  3160. css: '.content.active a[href="#system/object_manager"]',
  3161. mute_log: true,
  3162. )
  3163. sleep 4
  3164. element = instance.find_elements(css: '.content.active .js-discard').first
  3165. element.click
  3166. watch_for_disappear(
  3167. browser: instance,
  3168. css: '.content.active .js-discard',
  3169. )
  3170. end
  3171. =begin
  3172. tags_verify(
  3173. browser: browser2,
  3174. tags: {
  3175. 'tag 1' => true,
  3176. 'tag 2' => true,
  3177. 'tag 3' => false,
  3178. },
  3179. )
  3180. =end
  3181. def tags_verify(params = {})
  3182. switch_window_focus(params)
  3183. log('tags_verify', params)
  3184. instance = params[:browser] || @browser
  3185. tags = instance.find_elements({ css: '.content.active .js-tag' })
  3186. assert(tags)
  3187. assert(tags[0])
  3188. tags_found = {}
  3189. params[:tags].each_key do |key|
  3190. tags_found[key] = false
  3191. end
  3192. tags.each do |element|
  3193. text = element.text
  3194. if tags_found.key?(text)
  3195. tags_found[text] = true
  3196. else
  3197. assert(false, "tag exists but is not in check to verify '#{text}'")
  3198. end
  3199. end
  3200. params[:tags].each do |key, value|
  3201. assert_equal(value, tags_found[key], "tag '#{key}'")
  3202. end
  3203. end
  3204. def quote(string)
  3205. string_quoted = string
  3206. string_quoted.gsub!(/&/, '&amp;')
  3207. string_quoted.gsub!(/</, '&lt;')
  3208. string_quoted.gsub!(/>/, '&gt;')
  3209. string_quoted
  3210. end
  3211. def switch_window_focus(params)
  3212. instance = params[:browser] || @browser
  3213. if instance != @last_used_browser
  3214. log('switch browser window focus', {})
  3215. instance.switch_to.window(instance.window_handles.first)
  3216. end
  3217. @last_used_browser = instance
  3218. end
  3219. def log(method, params = {})
  3220. begin
  3221. instance = params[:browser] || @browser
  3222. if instance
  3223. logs = instance.manage.logs.get(:browser)
  3224. logs.each do |log|
  3225. next if log.level == 'WARNING' && log.message =~ /Declaration\sdropped./ # ignore ff css warnings
  3226. time = Time.zone.parse(Time.zone.at(log.timestamp / 1000).to_datetime.to_s)
  3227. puts "#{time}/#{log.level}: #{log.message}"
  3228. end
  3229. end
  3230. rescue
  3231. # failed to get logs
  3232. end
  3233. return if !DEBUG
  3234. return if params[:mute_log]
  3235. puts "#{Time.zone.now}/#{method}: #{params.inspect}"
  3236. end
  3237. private
  3238. def add_tree_options(instance:, options:)
  3239. # first level entries have to get added in regular order
  3240. options.each_key.with_index do |option, index|
  3241. if index != 0
  3242. element = instance.find_elements(css: '.modal .js-treeTable .js-addRow')[index - 1]
  3243. element.click
  3244. end
  3245. element = instance.find_elements(css: '.modal .js-treeTable .js-key')[index]
  3246. element.clear
  3247. element.send_keys(option)
  3248. end
  3249. add_sub_tree_recursion(
  3250. instance: instance,
  3251. options: options,
  3252. )
  3253. end
  3254. def add_sub_tree_recursion(instance:, options:, offset: 0)
  3255. options.each_value.inject(offset) do |child_offset, children|
  3256. child_offset += 1
  3257. # put your recursion glasses on 8-)
  3258. add_sub_tree_options(
  3259. instance: instance,
  3260. options: children,
  3261. offset: child_offset,
  3262. )
  3263. end
  3264. end
  3265. def add_sub_tree_options(instance:, options:, offset:)
  3266. # sub level entries have to get added in reversed order
  3267. level_options = options.to_a.reverse.to_h.keys
  3268. level_options.each do |option|
  3269. # sub level entries have to get added via 'add child row' link
  3270. click_index = offset - 1
  3271. element = instance.find_elements(css: '.modal .js-treeTable .js-addChild')[click_index]
  3272. element.click
  3273. element = instance.find_elements(css: '.modal .js-treeTable .js-key')[offset]
  3274. element.clear
  3275. element.send_keys(option)
  3276. sleep 0.25
  3277. end
  3278. add_sub_tree_recursion(
  3279. instance: instance,
  3280. options: options,
  3281. offset: offset,
  3282. )
  3283. end
  3284. end