34ee1f479a6a2a9a9eacd8402d79dd343bbaf08a
[arvados.git] / apps / workbench / test / integration_helper.rb
1 # Copyright (C) The Arvados Authors. All rights reserved.
2 #
3 # SPDX-License-Identifier: AGPL-3.0
4
5 require 'test_helper'
6 require 'capybara/rails'
7 require 'capybara/poltergeist'
8 require 'uri'
9 require 'yaml'
10
11 def available_port for_what
12   begin
13     Addrinfo.tcp("0.0.0.0", 0).listen do |srv|
14       port = srv.connect_address.ip_port
15       # Selenium needs an additional locking port, check if it's available
16       # and retry if necessary.
17       if for_what == 'selenium'
18         locking_port = port - 1
19         Addrinfo.tcp("0.0.0.0", locking_port).listen.close
20       end
21       STDERR.puts "Using port #{port} for #{for_what}"
22       return port
23     end
24   rescue Errno::EADDRINUSE, Errno::EACCES
25     retry
26   end
27 end
28
29 def selenium_opts
30   {
31     port: available_port('selenium'),
32     desired_capabilities: Selenium::WebDriver::Remote::Capabilities.firefox(
33       acceptInsecureCerts: true,
34     ),
35   }
36 end
37
38 def poltergeist_opts
39   {
40     phantomjs_options: ['--ignore-ssl-errors=true'],
41     port: available_port('poltergeist'),
42     window_size: [1200, 800],
43   }
44 end
45
46 Capybara.register_driver :poltergeist do |app|
47   Capybara::Poltergeist::Driver.new app, poltergeist_opts
48 end
49
50 Capybara.register_driver :poltergeist_debug do |app|
51   Capybara::Poltergeist::Driver.new app, poltergeist_opts.merge(inspector: true)
52 end
53
54 Capybara.register_driver :poltergeist_with_fake_websocket do |app|
55   js = File.expand_path '../support/fake_websocket.js', __FILE__
56   Capybara::Poltergeist::Driver.new app, poltergeist_opts.merge(extensions: [js])
57 end
58
59 Capybara.register_driver :poltergeist_without_file_api do |app|
60   js = File.expand_path '../support/remove_file_api.js', __FILE__
61   Capybara::Poltergeist::Driver.new app, poltergeist_opts.merge(extensions: [js])
62 end
63
64 Capybara.register_driver :selenium do |app|
65   Capybara::Selenium::Driver.new app, selenium_opts
66 end
67
68 Capybara.register_driver :selenium_with_download do |app|
69   profile = Selenium::WebDriver::Firefox::Profile.new
70   profile['browser.download.dir'] = DownloadHelper.path.to_s
71   profile['browser.download.downloadDir'] = DownloadHelper.path.to_s
72   profile['browser.download.defaultFolder'] = DownloadHelper.path.to_s
73   profile['browser.download.folderList'] = 2 # "save to user-defined location"
74   profile['browser.download.manager.showWhenStarting'] = false
75   profile['browser.helperApps.alwaysAsk.force'] = false
76   profile['browser.helperApps.neverAsk.saveToDisk'] = 'text/plain,application/octet-stream'
77   Capybara::Selenium::Driver.new app, selenium_opts.merge(profile: profile)
78 end
79
80 module WaitForAjax
81   # FIXME: Huge side effect here
82   # The following line changes the global default Capybara wait time, affecting
83   # every test which follows this one. This should be removed and the failing tests
84   # should have their individual wait times increased, if appropriate, using
85   # the using_wait_time(N) construct to temporarily change the wait time.
86   # Note: the below is especially bad because there are places that increase wait
87   # times using a multiplier e.g. using_wait_time(3 * Capybara.default_max_wait_time)
88   Capybara.default_max_wait_time = 10
89   def wait_for_ajax
90     timeout = 10
91     count = 0
92     while page.evaluate_script("jQuery.active").to_i > 0
93       count += 1
94       raise "AJAX request took more than #{timeout} seconds" if count > timeout * 10
95       sleep(0.1)
96     end
97   end
98
99 end
100
101 module AssertDomEvent
102   # Yield the supplied block, then wait for an event to arrive at a
103   # DOM element.
104   def assert_triggers_dom_event events, target='body'
105     magic = 'received-dom-event-' + rand(2**30).to_s(36)
106     page.execute_script <<eos
107       $('#{target}').one('#{events}', function() {
108         $('body').addClass('#{magic}');
109       });
110 eos
111     yield
112     assert_selector "body.#{magic}"
113     page.execute_script "$('body').removeClass('#{magic}');";
114   end
115 end
116
117 module HeadlessHelper
118   class HeadlessSingleton
119     @display = ENV['ARVADOS_TEST_HEADLESS_DISPLAY'] || rand(400)+100
120     STDERR.puts "Using display :#{@display} for headless tests"
121     def self.get
122       @headless ||= Headless.new reuse: false, display: @display
123     end
124   end
125
126   Capybara.default_driver = :rack_test
127
128   def self.included base
129     base.class_eval do
130       setup do
131         Capybara.use_default_driver
132         @headless = false
133       end
134
135       teardown do
136         if @headless
137           @headless.stop
138           @headless = false
139         end
140       end
141     end
142   end
143
144   def need_selenium reason=nil, driver=:selenium
145     Capybara.current_driver = driver
146     unless ENV['ARVADOS_TEST_HEADFUL'] or @headless
147       @headless = HeadlessSingleton.get
148       @headless.start
149     end
150   end
151
152   def need_javascript reason=nil
153     unless Capybara.current_driver == :selenium
154       Capybara.current_driver = :poltergeist
155     end
156   end
157 end
158
159 module KeepWebConfig
160   def getport service
161     File.read(File.expand_path("../../../../tmp/#{service}.port", __FILE__))
162   end
163
164   def use_keep_web_config
165     @kwport = getport 'keep-web-ssl'
166     @kwdport = getport 'keep-web-dl-ssl'
167     Rails.configuration.Services.WebDAV.ExternalURL = URI("https://localhost:#{@kwport}")
168     Rails.configuration.Services.WebDAVDownload.ExternalURL = URI("https://localhost:#{@kwdport}")
169   end
170 end
171
172 class ActionDispatch::IntegrationTest
173   # Make the Capybara DSL available in all integration tests
174   include Capybara::DSL
175   include ApiFixtureLoader
176   include WaitForAjax
177   include AssertDomEvent
178   include HeadlessHelper
179
180   @@API_AUTHS = self.api_fixture('api_client_authorizations')
181
182   def page_with_token(token, path='/')
183     # Generate a page path with an embedded API token.
184     # Typical usage: visit page_with_token('token_name', page)
185     # The token can be specified by the name of an api_client_authorizations
186     # fixture, or passed as a raw string.
187     api_token = ((@@API_AUTHS.include? token) ?
188                  @@API_AUTHS[token]['api_token'] : token)
189     path_parts = path.partition("#")
190     sep = (path_parts.first.include? '?') ? '&' : '?'
191     q_string = URI.encode_www_form('api_token' => api_token)
192     path_parts.insert(1, "#{sep}#{q_string}")
193     path_parts.join("")
194   end
195
196   # Find a page element, but return false instead of raising an
197   # exception if not found. Use this with assertions to explain that
198   # the error signifies a failed test rather than an unexpected error
199   # during a testing procedure.
200   def find?(*args)
201     begin
202       find(*args)
203     rescue Capybara::ElementNotFound
204       false
205     end
206   end
207
208   @@screenshot_count = 1
209   def screenshot
210     image_file = "./tmp/workbench-fail-#{@@screenshot_count}.png"
211     begin
212       page.save_screenshot image_file
213     rescue Capybara::NotSupportedByDriverError
214       # C'est la vie.
215     else
216       puts "Saved #{image_file}"
217       @@screenshot_count += 1
218     end
219   end
220
221   teardown do
222     if not passed?
223       screenshot
224     end
225     if Capybara.current_driver == :selenium
226       # Clearing localStorage crashes on a page where JS isn't
227       # executed. We also need to make sure we're clearing
228       # localStorage for the test server's origin, even if we finished
229       # the test on a different origin.
230       host = Capybara.current_session.server.host
231       port = Capybara.current_session.server.port
232       base = "http://#{host}:#{port}"
233       if page.evaluate_script("window.document.contentType") != "text/html" ||
234          !page.evaluate_script("window.location.toString()").start_with?(base)
235         visit "#{base}/404"
236       end
237       page.execute_script("window.localStorage.clear()")
238     else
239       page.driver.restart if defined?(page.driver.restart)
240     end
241     Capybara.reset_sessions!
242   end
243
244   def accept_alert
245     if Capybara.current_driver == :selenium
246       (0..9).each do
247         begin
248           page.driver.browser.switch_to.alert.accept
249           break
250         rescue Selenium::WebDriver::Error::NoSuchAlertError
251          sleep 0.1
252         end
253       end
254     else
255       # poltergeist returns true for confirm, so no need to accept
256     end
257   end
258 end
259
260 def upload_data_and_get_collection(data, user, filename, owner_uuid=nil)
261   token = api_token(user)
262   datablock = `echo -n #{data.shellescape} | ARVADOS_API_TOKEN=#{token.shellescape} arv-put --no-progress --raw -`.strip
263   assert $?.success?, $?
264   col = nil
265   use_token user do
266     mtxt = ". #{datablock} 0:#{data.length}:#{filename}\n"
267     if owner_uuid
268       col = Collection.create(manifest_text: mtxt, owner_uuid: owner_uuid)
269     else
270       col = Collection.create(manifest_text: mtxt)
271     end
272   end
273   return col
274 end