1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 describe('Create workflow tests', function () {
10 cy.getUser('admin', 'Admin', 'User', true, true)
11 .as('adminUser').then(function () {
12 adminUser = this.adminUser;
15 cy.getUser('activeuser', 'Active', 'User', false, true)
16 .as('activeUser').then(function () {
17 activeUser = this.activeUser;
22 it('can create project with nested data', function () {
23 cy.createGroup(adminUser.token, {
24 group_class: "project",
25 name: `Test project (${Math.floor(Math.random() * 999999)})`,
28 cy.get('@project1').then(() => {
29 cy.createGroup(adminUser.token, {
30 group_class: "project",
31 name: `Test project (${Math.floor(Math.random() * 999999)})`,
32 owner_uuid: this.project1.uuid,
36 cy.get('@project2').then(() => {
37 cy.createGroup(adminUser.token, {
38 group_class: "project",
39 name: `Test project (${Math.floor(Math.random() * 999999)})`,
40 owner_uuid: this.project2.uuid,
44 cy.get('@project3').then(() => {
45 cy.createWorkflow(adminUser.token, {
46 name: `TestWorkflow${Math.floor(Math.random() * 999999)}.cwl`,
47 definition: "{\n \"$graph\": [\n {\n \"class\": \"Workflow\",\n \"doc\": \"Reverse the lines in a document, then sort those lines.\",\n \"hints\": [\n {\n \"acrContainerImage\": \"99b0201f4cade456b4c9d343769a3b70+261\",\n \"class\": \"http://arvados.org/cwl#WorkflowRunnerResources\"\n }\n ],\n \"id\": \"#main\",\n \"inputs\": [\n {\n \"default\": null,\n \"doc\": \"The input file to be processed.\",\n \"id\": \"#main/input\",\n \"type\": \"File\"\n },\n {\n \"default\": true,\n \"doc\": \"If true, reverse (decending) sort\",\n \"id\": \"#main/reverse_sort\",\n \"type\": \"boolean\"\n }\n ],\n \"outputs\": [\n {\n \"doc\": \"The output with the lines reversed and sorted.\",\n \"id\": \"#main/output\",\n \"outputSource\": \"#main/sorted/output\",\n \"type\": \"File\"\n }\n ],\n \"steps\": [\n {\n \"id\": \"#main/rev\",\n \"in\": [\n {\n \"id\": \"#main/rev/input\",\n \"source\": \"#main/input\"\n }\n ],\n \"out\": [\n \"#main/rev/output\"\n ],\n \"run\": \"#revtool.cwl\"\n },\n {\n \"id\": \"#main/sorted\",\n \"in\": [\n {\n \"id\": \"#main/sorted/input\",\n \"source\": \"#main/rev/output\"\n },\n {\n \"id\": \"#main/sorted/reverse\",\n \"source\": \"#main/reverse_sort\"\n }\n ],\n \"out\": [\n \"#main/sorted/output\"\n ],\n \"run\": \"#sorttool.cwl\"\n }\n ]\n },\n {\n \"baseCommand\": \"rev\",\n \"class\": \"CommandLineTool\",\n \"doc\": \"Reverse each line using the `rev` command\",\n \"hints\": [\n {\n \"class\": \"ResourceRequirement\",\n \"ramMin\": 8\n }\n ],\n \"id\": \"#revtool.cwl\",\n \"inputs\": [\n {\n \"id\": \"#revtool.cwl/input\",\n \"inputBinding\": {},\n \"type\": \"File\"\n }\n ],\n \"outputs\": [\n {\n \"id\": \"#revtool.cwl/output\",\n \"outputBinding\": {\n \"glob\": \"output.txt\"\n },\n \"type\": \"File\"\n }\n ],\n \"stdout\": \"output.txt\"\n },\n {\n \"baseCommand\": \"sort\",\n \"class\": \"CommandLineTool\",\n \"doc\": \"Sort lines using the `sort` command\",\n \"hints\": [\n {\n \"class\": \"ResourceRequirement\",\n \"ramMin\": 8\n }\n ],\n \"id\": \"#sorttool.cwl\",\n \"inputs\": [\n {\n \"id\": \"#sorttool.cwl/reverse\",\n \"inputBinding\": {\n \"position\": 1,\n \"prefix\": \"-r\"\n },\n \"type\": \"boolean\"\n },\n {\n \"id\": \"#sorttool.cwl/input\",\n \"inputBinding\": {\n \"position\": 2\n },\n \"type\": \"File\"\n }\n ],\n \"outputs\": [\n {\n \"id\": \"#sorttool.cwl/output\",\n \"outputBinding\": {\n \"glob\": \"output.txt\"\n },\n \"type\": \"File\"\n }\n ],\n \"stdout\": \"output.txt\"\n }\n ],\n \"cwlVersion\": \"v1.0\"\n}",
51 cy.createCollection(adminUser.token, {
52 name: `Test collection ${Math.floor(Math.random() * 999999)}`,
53 owner_uuid: this.project3.uuid,
54 manifest_text: "./subdir 37b51d194a7513e45b56f6524f2d51f2+3 0:3:foo\n. 37b51d194a7513e45b56f6524f2d51f2+3 0:3:baz\n"
56 .as('testCollection');
59 cy.get('@testWorkflow').then(() => {
60 cy.loginAs(adminUser);
62 cy.get('[data-cy=side-panel-button]').click();
63 cy.get('[data-cy=side-panel-run-process]').click();
65 cy.get('.layout-pane')
66 .contains(this.testWorkflow.name)
69 cy.get('[data-cy=run-process-next-button]').click();
71 cy.get('[data-cy=new-process-panel]').contains('Run workflow').should('be.disabled');
73 cy.get('[data-cy=new-process-panel]')
75 cy.get('[name=name]').type(`Workflow name (${Math.floor(Math.random() * 999999)})`);
76 cy.contains('input').next().click();
79 cy.get('[data-cy=choose-a-file-dialog]').as('chooseFileDialog');
80 cy.get('@chooseFileDialog').contains('Home Projects').closest('ul').find('i').click();
82 cy.get('@project1').then((project1) => {
83 cy.get('@chooseFileDialog').find(`[data-id=${project1.uuid}]`).find('i').click();
86 cy.get('@project2').then((project2) => {
87 cy.get('@chooseFileDialog').find(`[data-id=${project2.uuid}]`).find('i').click();
90 cy.get('@project3').then((project3) => {
91 cy.get('@chooseFileDialog').find(`[data-id=${project3.uuid}]`).find('i').click();
94 cy.get('@testCollection').then((testCollection) => {
95 cy.get('@chooseFileDialog').find(`[data-id=${testCollection.uuid}]`).find('i').click();
98 cy.get('@chooseFileDialog').contains('baz').click();
100 cy.get('@chooseFileDialog').find('button').contains('Ok').click();
102 cy.get('[data-cy=new-process-panel]')
103 .find('button').contains('Run workflow').should('not.be.disabled');
107 ['workflow_with_array_fields.yaml', 'workflow_with_default_array_fields.yaml'].forEach((yamlfile) =>
108 it('can select multi files when creating workflow '+yamlfile, () => {
110 owningUser: activeUser,
111 projectName: 'myProject1',
115 cy.createCollection(adminUser.token, {
116 name: `Test collection ${Math.floor(Math.random() * 999999)}`,
117 owner_uuid: activeUser.user.uuid,
118 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n. 37b51d194a7513e45b56f6524f2d51f2+3 0:3:baz\n"
120 .as('testCollection');
122 cy.createCollection(adminUser.token, {
123 name: `Test collection ${Math.floor(Math.random() * 999999)}`,
124 owner_uuid: activeUser.user.uuid,
125 manifest_text: `. 37b51d194a7513e45b56f6524f2d51f2+3 0:3:buz\n`
127 .as('testCollection2');
129 cy.getAll('@myProject1', '@testCollection', '@testCollection2')
130 .then(function ([myProject1, testCollection, testCollection2]) {
131 cy.readFile('cypress/fixtures/'+yamlfile).then(workflow => {
132 cy.createWorkflow(adminUser.token, {
133 name: `TestWorkflow${Math.floor(Math.random() * 999999)}.cwl`,
134 definition: workflow,
135 owner_uuid: myProject1.uuid,
140 cy.loginAs(activeUser);
142 cy.get('main').contains(myProject1.name).click();
144 cy.get('[data-cy=side-panel-button]').click();
146 cy.get('#aside-menu-list').contains('Run a workflow').click();
148 cy.get('@testWorkflow')
149 .then((testWorkflow) => {
150 cy.get('main').contains(testWorkflow.name).click();
151 cy.get('[data-cy=run-process-next-button]').click();
153 cy.get('label').contains('foo').parent('div').find('input').click();
154 cy.get('div[role=dialog]')
156 // must use .then to avoid selecting instead of expanding https://github.com/cypress-io/cypress/issues/5529
157 cy.get('p').contains('Home Projects').closest('ul')
159 .then(el => el.click());
161 cy.get(`[data-id=${testCollection.uuid}]`)
165 cy.contains('bar').closest('[data-action=TOGGLE_ACTIVE]').parent().find('input[type=checkbox]').click();
166 cy.contains('baz').closest('[data-action=TOGGLE_ACTIVE]').parent().find('input[type=checkbox]').click();
168 cy.get('[data-cy=ok-button]').click();
171 cy.get('label').contains('bar').parent('div').find('input').click();
172 cy.get('div[role=dialog]')
174 // must use .then to avoid selecting instead of expanding https://github.com/cypress-io/cypress/issues/5529
175 cy.get('p').contains('Home Projects').closest('ul')
177 .then(el => el.click());
179 cy.get(`[data-id=${testCollection.uuid}]`)
180 .find('input[type=checkbox]').click();
182 cy.get(`[data-id=${testCollection2.uuid}]`)
183 .find('input[type=checkbox]').click();
185 cy.get('[data-cy=ok-button]').click();
189 cy.get('label').contains('foo').parent('div')
195 cy.get('label').contains('bar').parent('div')
197 cy.contains(testCollection.name);
198 cy.contains(testCollection2.name);
203 it('allows selecting collection subdirectories and reselects existing selections', () => {
205 owningUser: activeUser,
206 projectName: 'myProject1',
210 cy.createCollection(adminUser.token, {
211 name: `Test collection ${Math.floor(Math.random() * 999999)}`,
212 owner_uuid: activeUser.user.uuid,
213 manifest_text: "./subdir/dir1 d41d8cd98f00b204e9800998ecf8427e+0 0:0:\\056\n./subdir/dir2 d41d8cd98f00b204e9800998ecf8427e+0 0:0:\\056\n"
215 .as('testCollection');
217 cy.getAll('@myProject1', '@testCollection')
218 .then(function ([myProject1, testCollection]) {
219 cy.readFile('cypress/fixtures/workflow_directory_array.yaml').then(workflow => {
220 cy.createWorkflow(adminUser.token, {
221 name: `TestWorkflow${Math.floor(Math.random() * 999999)}.cwl`,
222 definition: workflow,
223 owner_uuid: myProject1.uuid,
228 cy.loginAs(activeUser);
230 cy.get('main').contains(myProject1.name).click();
232 cy.get('[data-cy=side-panel-button]').click();
234 cy.get('#aside-menu-list').contains('Run a workflow').click();
236 cy.get('@testWorkflow')
237 .then((testWorkflow) => {
238 cy.get('main').contains(testWorkflow.name).click();
239 cy.get('[data-cy=run-process-next-button]').click();
241 cy.get('label').contains('directoryInputName').parent('div').find('input').click();
242 cy.get('div[role=dialog]')
244 // must use .then to avoid selecting instead of expanding https://github.com/cypress-io/cypress/issues/5529
245 cy.get('p').contains('Home Projects').closest('ul')
247 .then(el => el.click());
249 cy.get(`[data-id=${testCollection.uuid}]`)
252 cy.get(`[data-id="${testCollection.uuid}/subdir"]`)
255 cy.contains('dir1').closest('[data-action=TOGGLE_ACTIVE]').parent().find('input[type=checkbox]').click();
256 cy.contains('dir2').closest('[data-action=TOGGLE_ACTIVE]').parent().find('input[type=checkbox]').click();
258 cy.get('[data-cy=ok-button]').click();
261 // Verify subdirectories were selected
262 cy.get('label').contains('directoryInputName').parent('div')
268 // Reopen tree picker and verify subdirectories are preselected
269 cy.get('label').contains('directoryInputName').parent('div').find('input').click();
270 cy.waitForDom().get('div[role=dialog]')
272 cy.contains('dir1').closest('[data-action=TOGGLE_ACTIVE]').parent().find('input[type=checkbox]').should('be.checked');
273 cy.contains('dir2').closest('[data-action=TOGGLE_ACTIVE]').parent().find('input[type=checkbox]').should('be.checked');
280 it('handles secret inputs', () => {
282 owningUser: activeUser,
283 projectName: 'myProject1',
287 cy.setupDockerImage("arvados/jobs").as("dockerImg");
289 cy.getAll('@myProject1').then(function ([myProject1]) {
290 cy.readFile('cypress/fixtures/workflow_with_secret_input.yaml').then(workflow => {
291 cy.createWorkflow(adminUser.token, {
292 name: `TestWorkflow${Math.floor(Math.random() * 999999)}.cwl`,
293 definition: workflow,
294 owner_uuid: myProject1.uuid,
299 cy.loginAs(activeUser);
301 cy.get('main').contains(myProject1.name).click();
303 cy.get('[data-cy=side-panel-button]').click();
305 cy.get('#aside-menu-list').contains('Run a workflow').click();
307 cy.get('@testWorkflow')
308 .then((testWorkflow) => {
309 cy.get('main').contains(testWorkflow.name).click();
310 cy.get('[data-cy=run-process-next-button]').click();
312 var foo = cy.get('label').contains('foo').parent('div').find('input');
313 foo.type("secret_value_xyz");
314 foo.should('have.attr', 'type').and('equal', 'password');
316 var bar = cy.get('label').contains('bar').parent('div').find('input');
317 bar.type("exposed_value_xyz");
318 bar.should('have.attr', 'type').and('equal', 'text');
320 cy.get('[data-cy=new-process-panel]').contains('Run workflow').click();
322 cy.get('[data-cy=process-io-card]').should('contain', 'exposed_value_xyz');
323 cy.get('[data-cy=process-io-card]').should('contain', 'Cannot display secret');
324 cy.get('[data-cy=process-io-card]').should('not.contain', 'secret_value_xyz');
326 cy.url().then((url) => {
327 let uuid = url.split('/').pop();
328 cy.getResource(activeUser.token, "container_requests", uuid).then((res) => {
329 expect(res.mounts["/var/lib/cwl/cwl.input.json"].content.bar).to.equal('exposed_value_xyz');
330 expect(res.mounts["/var/lib/cwl/cwl.input.json"].content.foo).to.deep.equal({$include: '/secrets/s0'});