+# Copyright (C) The Arvados Authors. All rights reserved.
+#
+# SPDX-License-Identifier: AGPL-3.0
+
require 'integration_helper'
class PipelineInstancesTest < ActionDispatch::IntegrationTest
DateTime.parse(utc).to_time
end
- if false
- # No need to test (or mention) these all the time. If they start
- # working (without need_selenium) then some real tests might not
- # need_selenium any more.
-
- test 'phantomjs DST' do
- skip '^^'
- t0s = '3/8/2015, 01:59 AM'
- t1s = '3/8/2015, 03:01 AM'
- t0 = parse_browser_timestamp t0s
- t1 = parse_browser_timestamp t1s
- assert_equal 120, t1-t0, "'#{t0s}' to '#{t1s}' was reported as #{t1-t0} seconds, should be 120"
- end
-
- test 'phantomjs DST 2' do
- skip '^^'
- t0s = '2015-03-08T10:43:00Z'
- t1s = '2015-03-09T03:43:00Z'
- t0 = parse_browser_timestamp page.evaluate_script("new Date('#{t0s}').toLocaleString()")
- t1 = parse_browser_timestamp page.evaluate_script("new Date('#{t1s}').toLocaleString()")
- assert_equal 17*3600, t1-t0, "'#{t0s}' to '#{t1s}' was reported as #{t1-t0} seconds, should be #{17*3600} (17 hours)"
- end
- end
-
- test 'Create and run a pipeline' do
- visit page_with_token('active_trustedclient', '/pipeline_templates')
- within('tr', text: 'Two Part Pipeline Template') do
- find('a,button', text: 'Run').click
- end
-
- # project chooser
- within('.modal-dialog') do #FIXME: source of 1 test error
- find('.selectable', text: 'A Project').click
- find('button', text: 'Choose').click
- end
-
- # This pipeline needs input. So, Run should be disabled
- page.assert_selector 'a.disabled,button.disabled', text: 'Run'
-
- instance_page = current_path
-
- # Add this collection to the project
- visit '/projects'
- find("#projects-menu").click
- find('.dropdown-menu a,button', text: 'A Project').click
- find('.btn', text: 'Add data').click
- find('.dropdown-menu a,button', text: 'Copy data from another project').click
- within('.modal-dialog') do
- wait_for_ajax
- first('span', text: 'foo_tag').click
- find('.btn', text: 'Copy').click
- end
- using_wait_time(Capybara.default_max_wait_time * 3) do
- wait_for_ajax
- end
-
- click_link 'Pipelines and processes'
- find('tr[data-kind="arvados#pipelineInstance"]', text: '(none)').
- find('a', text: 'Show').
- click
-
- assert find('p', text: 'Provide a value')
-
- find('div.form-group', text: 'Foo/bar pair').
- find('.btn', text: 'Choose').
- click
-
- within('.modal-dialog') do
- assert(has_text?("Foo/bar pair"),
- "pipeline input picker missing name of input")
- wait_for_ajax
- first('span', text: 'foo_tag').click
- find('button', text: 'OK').click
- end
- wait_for_ajax
-
- # The input, after being specified, should still be displayed (#3382)
- assert find('div.form-group', text: 'Foo/bar pair')
-
- # The input, after being specified, should still be editable (#3382)
- find('div.form-group', text: 'Foo/bar pair').
- find('.btn', text: 'Choose').click
-
- within('.modal-dialog') do
- assert(has_text?("Foo/bar pair"),
- "pipeline input picker missing name of input")
- wait_for_ajax
- first('span', text: 'foo_tag').click
- find('button', text: 'OK').click
- end
-
- # For good measure, check one last time that the input, after being specified twice, is still be displayed (#3382)
- assert find('div.form-group', text: 'Foo/bar pair')
-
- # Ensure that the collection's portable_data_hash, uuid and name
- # are saved in the desired places. (#4015)
-
- # foo_collection_in_aproject is the collection tagged with foo_tag.
- collection = api_fixture('collections', 'foo_collection_in_aproject')
- click_link 'Advanced'
- click_link 'API response'
- api_response = JSON.parse(find('div#advanced_api_response pre').text)
- input_params = api_response['components']['part-one']['script_parameters']['input']
- assert_equal input_params['value'], collection['portable_data_hash']
- assert_equal input_params['selection_name'], collection['name']
- assert_equal input_params['selection_uuid'], collection['uuid']
-
- # "Run" button is now enabled
- page.assert_no_selector 'a.disabled,button.disabled', text: 'Run'
-
- first('a,button', text: 'Run').click
-
- # Pipeline is running. We have a "Pause" button instead now.
- page.assert_selector 'a,button', text: 'Pause'
- find('a,button', text: 'Pause').click
-
- # Pipeline is stopped. It should now be in paused state and Runnable again.
- assert page.has_text? 'Paused'
- page.assert_no_selector 'a.disabled,button.disabled', text: 'Resume'
- page.assert_selector 'a,button', text: 'Re-run with latest'
- page.assert_selector 'a,button', text: 'Re-run options'
-
- # Since it is test env, no jobs are created to run. So, graph not visible
- assert page.has_no_text? 'Graph'
- end
-
- # Create a pipeline instance from within a project and run
- test 'Create pipeline inside a project and run' do
- visit page_with_token('active_trustedclient', '/projects')
-
- # Add collection to the project using Add data button
- find("#projects-menu").click
- find('.dropdown-menu a,button', text: 'A Project').click
- find('.btn', text: 'Add data').click
- find('.dropdown-menu a,button', text: 'Copy data from another project').click
- within('.modal-dialog') do
- wait_for_ajax
- first('span', text: 'foo_tag').click
- find('.btn', text: 'Copy').click
- end
- using_wait_time(Capybara.default_max_wait_time * 3) do
- wait_for_ajax
- end
-
- create_and_run_pipeline_in_aproject true, 'Two Part Pipeline Template', 'foo_collection_in_aproject', false
- end
-
- # Create a pipeline instance from outside of a project
- test 'Run a pipeline from dashboard' do
- visit page_with_token('active_trustedclient')
- create_and_run_pipeline_in_aproject false, 'Two Part Pipeline Template', 'foo_collection_in_aproject', false
- end
-
test 'view pipeline with job and see graph' do
visit page_with_token('active_trustedclient', '/pipeline_instances')
assert page.has_text? 'pipeline_with_job'
page.assert_selector "#provenance_graph"
end
- test 'pipeline description' do
- visit page_with_token('active_trustedclient', '/pipeline_instances')
- assert page.has_text? 'pipeline_with_job'
-
- find('a', text: 'pipeline_with_job').click
-
- within('.arv-description-as-subtitle') do
- find('.fa-pencil').click
- find('.editable-input textarea').set('*Textile description for pipeline instance*')
- find('.editable-submit').click
- end
- wait_for_ajax
-
- # verify description
- assert page.has_no_text? '*Textile description for pipeline instance*'
- assert page.has_text? 'Textile description for pipeline instance'
- end
-
test "JSON popup available for strange components" do
uuid = api_fixture("pipeline_instances")["components_is_jobspec"]["uuid"]
visit page_with_token("active", "/pipeline_instances/#{uuid}")
"did not land on pipeline instance page")
end
- PROJECT_WITH_SEARCH_COLLECTION = "A Subproject"
- def check_parameter_search(proj_name)
- create_pipeline_from("parameter_with_search", proj_name)
- search_text = api_fixture("pipeline_templates", "parameter_with_search",
- "components", "with-search",
- "script_parameters", "input", "search_for")
- first("a.btn,button", text: "Choose").click
- within(".modal-body") do
- if (proj_name != PROJECT_WITH_SEARCH_COLLECTION)
- # Switch finder modal to Subproject to find the Collection.
- click_on proj_name
- click_on PROJECT_WITH_SEARCH_COLLECTION
- end
- assert_equal(search_text, first("input").value,
- "parameter search not preseeded")
- assert(has_text?(api_fixture("collections")["baz_collection_name_in_asubproject"]["name"]),
- "baz Collection not in preseeded search results")
- end
- end
-
- test "Workbench respects search_for parameter in templates" do
- check_parameter_search(PROJECT_WITH_SEARCH_COLLECTION)
- end
-
- test "Workbench preserves search_for parameter after project switch" do
- check_parameter_search("A Project")
- end
-
- test "enter a float for a number pipeline input" do
- # Poltergeist either does not support the HTML 5 <input
- # type="number">, or interferes with the associated X-Editable
- # validation code. If the input field has type=number (forcing an
- # integer), this test will yield a false positive under
- # Poltergeist. --Brett, 2015-02-05
- need_selenium "for strict X-Editable input validation"
- create_pipeline_from("template_with_dataclass_number")
- INPUT_SELECTOR =
- ".editable[data-name='[components][work][script_parameters][input][value]']"
- find(INPUT_SELECTOR).click
- find(".editable-input input").set("12.34")
- find("#editable-submit").click
- assert_no_selector(".editable-popup")
- assert_selector(INPUT_SELECTOR, text: "12.34")
- end
-
- [
- [true, 'Two Part Pipeline Template', 'foo_collection_in_aproject', false],
- [false, 'Two Part Pipeline Template', 'foo_collection_in_aproject', false],
- [true, 'Two Part Template with dataclass File', 'foo_collection_in_aproject', true],
- [false, 'Two Part Template with dataclass File', 'foo_collection_in_aproject', true],
- [true, 'Two Part Pipeline Template', 'collection_with_no_name_in_aproject', false],
- ].each do |in_aproject, template_name, collection, choose_file|
- test "Run pipeline instance in #{in_aproject} with #{template_name} with #{collection} file #{choose_file}" do
- if in_aproject
- visit page_with_token 'active', \
- '/projects/'+api_fixture('groups')['aproject']['uuid']
- else
- visit page_with_token 'active', '/'
- end
-
- # need bigger modal size when choosing a file from collection
- if Capybara.current_driver == :selenium
- Capybara.current_session.driver.browser.manage.window.resize_to(1200, 800)
- end
-
- create_and_run_pipeline_in_aproject in_aproject, template_name, collection, choose_file
- instance_path = current_path
-
- # Pause the pipeline
- find('a,button', text: 'Pause').click
- assert page.has_text? 'Paused'
- page.assert_no_selector 'a.disabled,button.disabled', text: 'Resume'
- page.assert_selector 'a,button', text: 'Re-run with latest'
- page.assert_selector 'a,button', text: 'Re-run options'
-
- # Verify that the newly created instance is created in the right project.
- assert page.has_text? 'Home'
- if in_aproject
- assert page.has_text? 'A Project'
- else
- assert page.has_no_text? 'A Project'
- end
- end
- end
-
- [
- ['active', false, false, false],
- ['active', false, false, true],
- ['active', true, false, false],
- ['active', true, true, false],
- ['active', true, false, true],
- ['active', true, true, true],
- ['project_viewer', false, false, true],
- ['project_viewer', true, true, true],
- ].each do |user, with_options, choose_options, in_aproject|
- test "Rerun pipeline instance as #{user} using options #{with_options} #{choose_options} in #{in_aproject}" do
- if in_aproject
- path = '/pipeline_instances/'+api_fixture('pipeline_instances')['pipeline_owned_by_active_in_aproject']['uuid']
- else
- path = '/pipeline_instances/'+api_fixture('pipeline_instances')['pipeline_owned_by_active_in_home']['uuid']
- end
-
- visit page_with_token(user, path)
-
- page.assert_selector 'a,button', text: 'Re-run with latest'
- page.assert_selector 'a,button', text: 'Re-run options'
-
- if user == 'project_viewer' && in_aproject
- assert page.has_text? 'A Project'
- end
-
- # Now re-run the pipeline
- if with_options
- assert_triggers_dom_event 'shown.bs.modal' do
- find('a,button', text: 'Re-run options').click
- end
- within('.modal-dialog') do
- page.assert_selector 'a,button', text: 'Copy and edit inputs'
- page.assert_selector 'a,button', text: 'Run now'
- if choose_options
- find('button', text: 'Copy and edit inputs').click
- else
- find('button', text: 'Run now').click
- end
- end
- else
- find('a,button', text: 'Re-run with latest').click
- end
-
- # Verify that the newly created instance is created in the right
- # project. In case of project_viewer user, since the user cannot
- # write to the project, the pipeline should have been created in
- # the user's Home project.
- assert_not_equal path, current_path, 'Rerun instance path expected to be different'
- assert_text 'Home'
- if in_aproject && (user != 'project_viewer')
- assert_text 'A Project'
- else
- assert_no_text 'A Project'
- end
- end
- end
-
- # Create and run a pipeline for 'Two Part Pipeline Template' in 'A Project'
- def create_and_run_pipeline_in_aproject in_aproject, template_name, collection_fixture, choose_file=false
- # collection in aproject to be used as input
- collection = api_fixture('collections', collection_fixture)
-
- # create a pipeline instance
- find('.btn', text: 'Run a pipeline').click
- within('.modal-dialog') do
- find('.selectable', text: template_name).click
- find('.btn', text: 'Next: choose inputs').click
- end
-
- assert find('p', text: 'Provide a value')
-
- find('div.form-group', text: 'Foo/bar pair').
- find('.btn', text: 'Choose').
- click
-
- within('.modal-dialog') do
- if in_aproject
- assert_selector 'button.dropdown-toggle', text: 'A Project'
- wait_for_ajax
- else
- assert_selector 'button.dropdown-toggle', text: 'Home'
- wait_for_ajax
- click_button "Home"
- click_link "A Project"
- wait_for_ajax
- end
-
- if collection_fixture == 'foo_collection_in_aproject'
- first('span', text: 'foo_tag').click
- elsif collection['name']
- first('span', text: "#{collection['name']}").click
- else
- collection_uuid = collection['uuid']
- find("div[data-object-uuid=#{collection_uuid}]").click
- end
-
- if choose_file
- wait_for_ajax
- find('.preview-selectable', text: 'foo').click
- end
- find('button', text: 'OK').click
- end
-
- # The input, after being specified, should still be displayed (#3382)
- assert find('div.form-group', text: 'Foo/bar pair')
-
- # Ensure that the collection's portable_data_hash, uuid and name
- # are saved in the desired places. (#4015)
- click_link 'Advanced'
- click_link 'API response'
-
- api_response = JSON.parse(find('div#advanced_api_response pre').text)
- input_params = api_response['components']['part-one']['script_parameters']['input']
- assert_equal(input_params['selection_uuid'], collection['uuid'], "Not found expected input param uuid")
- if choose_file
- assert_equal(input_params['value'], collection['portable_data_hash']+'/foo', "Not found expected input file param value")
- assert_equal(input_params['selection_name'], collection['name']+'/foo', "Not found expected input file param name")
- else
- assert_equal(input_params['value'], collection['portable_data_hash'], "Not found expected input param value")
- assert_equal(input_params['selection_name'], collection['name'], "Not found expected input selection name")
- end
-
- # "Run" button present and enabled
- page.assert_no_selector 'a.disabled,button.disabled', text: 'Run'
- first('a,button', text: 'Run').click
-
- # Pipeline is running. We have a "Pause" button instead now.
- page.assert_no_selector 'a,button', text: 'Run'
- page.assert_no_selector 'a.disabled,button.disabled', text: 'Resume'
- page.assert_selector 'a,button', text: 'Pause'
-
- # Since it is test env, no jobs are created to run. So, graph not visible
- assert page.has_no_text? 'Graph'
- end
-
[
['user1_with_load', 'zzzzz-d1hrv-10pipelines0001', 0], # run time 0 minutes
['user1_with_load', 'zzzzz-d1hrv-10pipelines0010', 17*60*60 + 51*60], # run time 17 hours and 51 minutes
need_selenium 'to parse timestamps correctly across DST boundaries'
visit page_with_token(user, "/pipeline_instances/#{uuid}")
- assert page.has_text? 'This pipeline started at'
- page_text = page.text
-
+ regexp = "This pipeline started at (.+?)\\. "
if run_time
- match = /This pipeline started at (.*)\. It failed after (.*) at (.*)\. Check the Log/.match page_text
+ regexp += "It failed after (.+?) at (.+?)\\. Check the Log"
else
- match = /This pipeline started at (.*). It has been active for(.*)/.match page_text
+ regexp += "It has been active for \\d"
end
- assert_not_nil(match, 'Did not find text - This pipeline started at . . . ')
+ assert_match /#{regexp}/, page.text
- start_at = match[1]
- assert_not_nil(start_at, 'Did not find start_at time')
+ return if !run_time
- start_time = parse_browser_timestamp start_at
- if run_time
- finished_at = match[3]
- assert_not_nil(finished_at, 'Did not find finished_at time')
- finished_time = parse_browser_timestamp finished_at
- assert_equal(run_time, finished_time-start_time,
- "Time difference did not match for start_at #{start_at}, finished_at #{finished_at}, ran_for #{match[2]}")
- else
- match = /\d(.*)/.match match[2]
- assert_not_nil match, 'Did not find expected match for running component'
- end
+ # match again to capture (.*)
+ _, started, duration, finished = *(/#{regexp}/.match(page.text))
+ assert_equal(
+ run_time,
+ parse_browser_timestamp(finished) - parse_browser_timestamp(started),
+ "expected: #{run_time}, got: started #{started}, finished #{finished}, duration #{duration}")
end
end