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