1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 describe("Search tests", function () {
10 // Only set up common users once. These aren't set up as aliases because
11 // aliases are cleaned up after every test. Also it doesn't make sense
12 // to set the same users on beforeEach() over and over again, so we
13 // separate a little from Cypress' 'Best Practices' here.
14 cy.getUser("admin", "Admin", "User", true, true)
17 adminUser = this.adminUser;
19 cy.getUser("collectionuser1", "Collection", "User", false, true)
22 activeUser = this.activeUser;
26 it("can search for old collection versions", function () {
27 const colName = `Versioned Collection ${Math.floor(Math.random() * Math.floor(999999))}`;
29 let oldVersionUuid = "";
30 // Make sure no other collections with this name exist
31 cy.doRequest("GET", "/arvados/v1/collections", null, {
32 filters: `[["name", "=", "${colName}"]]`,
33 include_old_versions: true,
38 expect(this.collections).to.be.empty;
40 // Creates the collection using the admin token so we can set up
41 // a bogus manifest text without block signatures.
42 cy.createCollection(adminUser.token, {
44 owner_uuid: activeUser.user.uuid,
45 preserve_version: true,
46 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n",
48 .as("originalVersion")
50 // Change the file name to create a new version.
51 cy.updateCollection(adminUser.token, this.originalVersion.uuid, {
52 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:foo\n",
54 colUuid = this.originalVersion.uuid;
56 // Confirm that there are 2 versions of the collection
57 cy.doRequest("GET", "/arvados/v1/collections", null, {
58 filters: `[["name", "=", "${colName}"]]`,
59 include_old_versions: true,
64 expect(this.collections).to.have.lengthOf(2);
65 this.collections.map(function (aCollection) {
66 expect(aCollection.current_version_uuid).to.equal(colUuid);
67 if (aCollection.uuid !== aCollection.current_version_uuid) {
68 oldVersionUuid = aCollection.uuid;
71 cy.loginAs(activeUser);
72 const searchQuery = `${colName} type:arvados#collection`;
73 // Search for only collection's current version
74 cy.doSearch(`${searchQuery}`);
75 cy.get("[data-cy=search-results]").should("contain", "head version");
76 cy.get("[data-cy=search-results]").should("not.contain", "version 1");
77 // ...and then, include old versions.
78 cy.doSearch(`${searchQuery} is:pastVersion`);
79 cy.get("[data-cy=search-results]").should("contain", "head version");
80 cy.get("[data-cy=search-results]").should("contain", "version 1");
84 it("can display path of the selected item", function () {
85 const colName = `Collection ${Math.floor(Math.random() * Math.floor(999999))}`;
87 // Creates the collection using the admin token so we can set up
88 // a bogus manifest text without block signatures.
89 cy.createCollection(adminUser.token, {
91 owner_uuid: activeUser.user.uuid,
92 preserve_version: true,
93 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n",
95 cy.loginAs(activeUser);
99 cy.get("[data-cy=search-results]").should("contain", colName);
101 cy.get("[data-cy=search-results]").contains(colName).closest("tr").click();
103 cy.get("[data-cy=element-path]").should("contain", `/ Projects / ${colName}`);
107 it("can search items using quotes", function () {
108 const random = Math.floor(Math.random() * Math.floor(999999));
109 const colName = `Collection ${random}`;
110 const colName2 = `Collection test ${random}`;
112 // Creates the collection using the admin token so we can set up
113 // a bogus manifest text without block signatures.
114 cy.createCollection(adminUser.token, {
116 owner_uuid: activeUser.user.uuid,
117 preserve_version: true,
118 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n",
119 }).as("collection1");
121 cy.createCollection(adminUser.token, {
123 owner_uuid: activeUser.user.uuid,
124 preserve_version: true,
125 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n",
126 }).as("collection2");
128 cy.getAll("@collection1", "@collection2").then(function () {
129 cy.loginAs(activeUser);
131 cy.doSearch(colName);
132 cy.get("[data-cy=search-results] table tbody tr").should("have.length", 2);
134 cy.doSearch(`"${colName}"`);
135 cy.get("[data-cy=search-results] table tbody tr").should("have.length", 1);
139 it("can display owner of the item", function () {
140 const colName = `Collection ${Math.floor(Math.random() * Math.floor(999999))}`;
142 cy.createCollection(adminUser.token, {
144 owner_uuid: activeUser.user.uuid,
145 preserve_version: true,
146 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n",
147 }).then(function () {
148 cy.loginAs(activeUser);
150 cy.doSearch(colName);
152 cy.get("[data-cy=search-results]").should("contain", colName);
154 cy.get("[data-cy=search-results]")
158 cy.get("p").contains(activeUser.user.uuid).should("contain", activeUser.user.full_name);
163 // The following test is enabled on Electron only, as Chromium and Firefox
164 // require permissions to access the clipboard.
165 it("shows search context menu", { browser: 'electron' } , function () {
166 const colName = `Home Collection ${Math.floor(Math.random() * Math.floor(999999))}`;
167 const federatedColName = `Federated Collection ${Math.floor(Math.random() * Math.floor(999999))}`;
168 const federatedColUuid = "xxxxx-4zz18-000000000000000";
170 // Intercept config to insert remote cluster
171 cy.intercept({ method: "GET", hostname: "127.0.0.1", url: "**/arvados/v1/config?nocache=*" }, req => {
172 req.on('response', res => {
173 res.body.RemoteClusters = {
174 "*": res.body.RemoteClusters["*"],
177 Host: "xxxxx.fakecluster.tld",
186 // Fake remote cluster config
190 hostname: "xxxxx.fakecluster.tld",
191 url: "**/arvados/v1/config",
204 ActivateUsers: false,
212 Composer: { ExternalURL: "" },
213 Controller: { ExternalURL: "https://xxxxx.fakecluster.tld:34763/" },
214 DispatchCloud: { ExternalURL: "" },
215 DispatchLSF: { ExternalURL: "" },
216 DispatchSLURM: { ExternalURL: "" },
217 Health: { ExternalURL: "https://xxxxx.fakecluster.tld:42915/" },
218 Keepbalance: { ExternalURL: "" },
219 Keepproxy: { ExternalURL: "https://xxxxx.fakecluster.tld:46773/" },
220 Keepstore: { ExternalURL: "" },
221 RailsAPI: { ExternalURL: "" },
222 WebDAV: { ExternalURL: "https://xxxxx.fakecluster.tld:36041/" },
223 WebDAVDownload: { ExternalURL: "https://xxxxx.fakecluster.tld:42957/" },
224 WebShell: { ExternalURL: "" },
225 Websocket: { ExternalURL: "wss://xxxxx.fakecluster.tld:37121/websocket" },
226 Workbench1: { ExternalURL: "https://wb1.xxxxx.fakecluster.tld/" },
227 Workbench2: { ExternalURL: "https://wb2.xxxxx.fakecluster.tld/" },
230 default: { Default: true, Priority: 0 },
232 Users: { SupportEmailAddress: "arvados@example.com" },
239 cy.createCollection(adminUser.token, {
241 owner_uuid: activeUser.user.uuid,
242 preserve_version: true,
243 manifest_text: ". 37b51d194a7513e45b56f6524f2d51f2+3 0:3:bar\n",
244 }).then(function (testCollection) {
245 cy.loginAs(activeUser);
247 // Intercept search results to add federated result
248 cy.intercept({ method: "GET", url: "**/arvados/v1/groups/contents?*" }, req => {
249 req.on('response', res => {
253 ...res.body.items[0],
254 uuid: federatedColUuid,
255 portable_data_hash: "00000000000000000000000000000000+0",
256 name: federatedColName,
257 href: res.body.items[0].href.replace(testCollection.uuid, federatedColUuid),
260 res.body.items_available += 1;
264 cy.doSearch(colName);
267 cy.window().then(win => {
268 cy.stub(win, "open").as("Open");
271 // Check Copy link to clipboard
272 cy.get("[data-cy=search-results]").contains(colName).rightclick();
273 cy.get("[data-cy=context-menu]").within(ctx => {
274 // Check that there are 4 items in the menu
275 cy.get(ctx).children().should("have.length", 4);
276 cy.contains("API Details");
277 cy.contains("Copy link to clipboard");
278 cy.contains("Open in new tab");
279 cy.contains("View details");
281 cy.contains("Copy link to clipboard").click();
283 cy.window({ timeout: 15000 }).then(win =>
284 win.navigator.clipboard.readText().then(text => {
285 expect(text).to.match(new RegExp(`/collections/${testCollection.uuid}$`));
290 // Check open in new tab
291 cy.get("[data-cy=search-results]").contains(colName).rightclick();
292 cy.get("[data-cy=context-menu]").within(() => {
293 cy.contains("Open in new tab").click();
295 cy.get("@Open").should("have.been.calledOnceWith", `${window.location.origin}/collections/${testCollection.uuid}`);
298 // Check federated result Copy link to clipboard
299 cy.get("[data-cy=search-results]").contains(federatedColName).rightclick();
300 cy.get("[data-cy=context-menu]").within(() => {
301 cy.contains("Copy link to clipboard").click();
303 cy.window({ timeout: 15000 }).then(win =>
304 win.navigator.clipboard.readText().then(text => {
305 expect(text).to.equal(`https://wb2.xxxxx.fakecluster.tld/collections/${federatedColUuid}`);
309 // Check open in new tab
310 cy.get("[data-cy=search-results]").contains(federatedColName).rightclick();
311 cy.get("[data-cy=context-menu]").within(() => {
312 cy.contains("Open in new tab").click();
314 cy.get("@Open").should("have.been.calledWith", `https://wb2.xxxxx.fakecluster.tld/collections/${federatedColUuid}`);