15872cc2d4
* Unknown status for allocations accounted for * Canary string removed * Test cleanup * Generate unknown in mirage * aacidentally oovervoowled * Update ui/app/components/allocation-status-bar.js Co-authored-by: Derek Strickland <1111455+DerekStrickland@users.noreply.github.com> * Disconnected state on job status in client * Renaming Disconnected to Unknown in the job-status-in-client * Unknown accounted for on job rows filtering and testsfix * Adding lostAllocs as a computed dependency * Unknown client status within acceptance test * Swatches updated and PR comments addressed * Unknown and disconnected added to test fixtures Co-authored-by: Derek Strickland <1111455+DerekStrickland@users.noreply.github.com>
317 lines
8.8 KiB
JavaScript
317 lines
8.8 KiB
JavaScript
/* eslint-disable qunit/require-expect */
|
|
import { currentURL } from '@ember/test-helpers';
|
|
import { module, test } from 'qunit';
|
|
import { setupApplicationTest } from 'ember-qunit';
|
|
import { setupMirage } from 'ember-cli-mirage/test-support';
|
|
import a11yAudit from 'nomad-ui/tests/helpers/a11y-audit';
|
|
import Allocations from 'nomad-ui/tests/pages/jobs/job/allocations';
|
|
|
|
let job;
|
|
let allocations;
|
|
|
|
const makeSearchAllocations = (server) => {
|
|
Array(10)
|
|
.fill(null)
|
|
.map((_, index) => {
|
|
server.create('allocation', {
|
|
id: index < 5 ? `ffffff-dddddd-${index}` : `111111-222222-${index}`,
|
|
shallow: true,
|
|
});
|
|
});
|
|
};
|
|
|
|
module('Acceptance | job allocations', function (hooks) {
|
|
setupApplicationTest(hooks);
|
|
setupMirage(hooks);
|
|
|
|
hooks.beforeEach(function () {
|
|
server.create('node');
|
|
|
|
job = server.create('job', {
|
|
noFailedPlacements: true,
|
|
createAllocations: false,
|
|
});
|
|
});
|
|
|
|
test('it passes an accessibility audit', async function (assert) {
|
|
server.createList('allocation', Allocations.pageSize - 1, {
|
|
shallow: true,
|
|
});
|
|
allocations = server.schema.allocations.where({ jobId: job.id }).models;
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
await a11yAudit(assert);
|
|
});
|
|
|
|
test('lists all allocations for the job', async function (assert) {
|
|
server.createList('allocation', Allocations.pageSize - 1, {
|
|
shallow: true,
|
|
});
|
|
allocations = server.schema.allocations.where({ jobId: job.id }).models;
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
|
|
assert.equal(
|
|
Allocations.allocations.length,
|
|
Allocations.pageSize - 1,
|
|
'Allocations are shown in a table'
|
|
);
|
|
|
|
const sortedAllocations = allocations.sortBy('modifyIndex').reverse();
|
|
|
|
Allocations.allocations.forEach((allocation, index) => {
|
|
const shortId = sortedAllocations[index].id.split('-')[0];
|
|
assert.equal(
|
|
allocation.shortId,
|
|
shortId,
|
|
`Allocation ${index} is ${shortId}`
|
|
);
|
|
});
|
|
|
|
assert.equal(document.title, `Job ${job.name} allocations - Nomad`);
|
|
});
|
|
|
|
test('allocations table is sortable', async function (assert) {
|
|
server.createList('allocation', Allocations.pageSize - 1);
|
|
allocations = server.schema.allocations.where({ jobId: job.id }).models;
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
await Allocations.sortBy('taskGroupName');
|
|
|
|
assert.equal(
|
|
currentURL(),
|
|
`/jobs/${job.id}/allocations?sort=taskGroupName`,
|
|
'the URL persists the sort parameter'
|
|
);
|
|
const sortedAllocations = allocations.sortBy('taskGroup').reverse();
|
|
Allocations.allocations.forEach((allocation, index) => {
|
|
const shortId = sortedAllocations[index].id.split('-')[0];
|
|
assert.equal(
|
|
allocation.shortId,
|
|
shortId,
|
|
`Allocation ${index} is ${shortId} with task group ${sortedAllocations[index].taskGroup}`
|
|
);
|
|
});
|
|
});
|
|
|
|
test('allocations table is searchable', async function (assert) {
|
|
makeSearchAllocations(server);
|
|
|
|
allocations = server.schema.allocations.where({ jobId: job.id }).models;
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
await Allocations.search('ffffff');
|
|
|
|
assert.equal(
|
|
Allocations.allocations.length,
|
|
5,
|
|
'List is filtered by search term'
|
|
);
|
|
});
|
|
|
|
test('when a search yields no results, the search box remains', async function (assert) {
|
|
makeSearchAllocations(server);
|
|
|
|
allocations = server.schema.allocations.where({ jobId: job.id }).models;
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
await Allocations.search('^nothing will ever match this long regex$');
|
|
|
|
assert.equal(
|
|
Allocations.emptyState.headline,
|
|
'No Matches',
|
|
'List is empty and the empty state is about search'
|
|
);
|
|
|
|
assert.ok(Allocations.hasSearchBox, 'Search box is still shown');
|
|
});
|
|
|
|
test('when the job for the allocations is not found, an error message is shown, but the URL persists', async function (assert) {
|
|
await Allocations.visit({ id: 'not-a-real-job' });
|
|
|
|
assert.equal(
|
|
server.pretender.handledRequests
|
|
.filter((request) => !request.url.includes('policy'))
|
|
.findBy('status', 404).url,
|
|
'/v1/job/not-a-real-job',
|
|
'A request to the nonexistent job is made'
|
|
);
|
|
assert.equal(
|
|
currentURL(),
|
|
'/jobs/not-a-real-job/allocations',
|
|
'The URL persists'
|
|
);
|
|
assert.ok(Allocations.error.isPresent, 'Error message is shown');
|
|
assert.equal(
|
|
Allocations.error.title,
|
|
'Not Found',
|
|
'Error message is for 404'
|
|
);
|
|
});
|
|
|
|
testFacet('Status', {
|
|
facet: Allocations.facets.status,
|
|
paramName: 'status',
|
|
expectedOptions: [
|
|
'Pending',
|
|
'Running',
|
|
'Complete',
|
|
'Failed',
|
|
'Lost',
|
|
'Unknown',
|
|
],
|
|
async beforeEach() {
|
|
['pending', 'running', 'complete', 'failed', 'lost', 'unknown'].forEach(
|
|
(s) => {
|
|
server.createList('allocation', 5, { clientStatus: s });
|
|
}
|
|
);
|
|
await Allocations.visit({ id: job.id });
|
|
},
|
|
filter: (alloc, selection) =>
|
|
alloc.jobId == job.id && selection.includes(alloc.clientStatus),
|
|
});
|
|
|
|
testFacet('Client', {
|
|
facet: Allocations.facets.client,
|
|
paramName: 'client',
|
|
expectedOptions(allocs) {
|
|
return Array.from(
|
|
new Set(
|
|
allocs
|
|
.filter((alloc) => alloc.jobId == job.id)
|
|
.mapBy('nodeId')
|
|
.map((id) => id.split('-')[0])
|
|
)
|
|
).sort();
|
|
},
|
|
async beforeEach() {
|
|
server.createList('node', 5);
|
|
server.createList('allocation', 20);
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
},
|
|
filter: (alloc, selection) =>
|
|
alloc.jobId == job.id && selection.includes(alloc.nodeId.split('-')[0]),
|
|
});
|
|
|
|
testFacet('Task Group', {
|
|
facet: Allocations.facets.taskGroup,
|
|
paramName: 'taskGroup',
|
|
expectedOptions(allocs) {
|
|
return Array.from(
|
|
new Set(
|
|
allocs.filter((alloc) => alloc.jobId == job.id).mapBy('taskGroup')
|
|
)
|
|
).sort();
|
|
},
|
|
async beforeEach() {
|
|
job = server.create('job', {
|
|
type: 'service',
|
|
status: 'running',
|
|
groupsCount: 5,
|
|
});
|
|
|
|
await Allocations.visit({ id: job.id });
|
|
},
|
|
filter: (alloc, selection) =>
|
|
alloc.jobId == job.id && selection.includes(alloc.taskGroup),
|
|
});
|
|
});
|
|
|
|
function testFacet(
|
|
label,
|
|
{ facet, paramName, beforeEach, filter, expectedOptions }
|
|
) {
|
|
test(`facet ${label} | the ${label} facet has the correct options`, async function (assert) {
|
|
await beforeEach();
|
|
await facet.toggle();
|
|
|
|
let expectation;
|
|
if (typeof expectedOptions === 'function') {
|
|
expectation = expectedOptions(server.db.allocations);
|
|
} else {
|
|
expectation = expectedOptions;
|
|
}
|
|
|
|
assert.deepEqual(
|
|
facet.options.map((option) => option.label.trim()),
|
|
expectation,
|
|
'Options for facet are as expected'
|
|
);
|
|
});
|
|
|
|
test(`facet ${label} | the ${label} facet filters the allocations list by ${label}`, async function (assert) {
|
|
let option;
|
|
|
|
await beforeEach();
|
|
|
|
await facet.toggle();
|
|
option = facet.options.objectAt(0);
|
|
await option.toggle();
|
|
|
|
const selection = [option.key];
|
|
const expectedAllocs = server.db.allocations
|
|
.filter((alloc) => filter(alloc, selection))
|
|
.sortBy('modifyIndex')
|
|
.reverse();
|
|
|
|
Allocations.allocations.forEach((alloc, index) => {
|
|
assert.equal(
|
|
alloc.id,
|
|
expectedAllocs[index].id,
|
|
`Allocation at ${index} is ${expectedAllocs[index].id}`
|
|
);
|
|
});
|
|
});
|
|
|
|
test(`facet ${label} | selecting multiple options in the ${label} facet results in a broader search`, async function (assert) {
|
|
const selection = [];
|
|
|
|
await beforeEach();
|
|
await facet.toggle();
|
|
|
|
const option1 = facet.options.objectAt(0);
|
|
const option2 = facet.options.objectAt(1);
|
|
await option1.toggle();
|
|
selection.push(option1.key);
|
|
await option2.toggle();
|
|
selection.push(option2.key);
|
|
|
|
const expectedAllocs = server.db.allocations
|
|
.filter((alloc) => filter(alloc, selection))
|
|
.sortBy('modifyIndex')
|
|
.reverse();
|
|
|
|
Allocations.allocations.forEach((alloc, index) => {
|
|
assert.equal(
|
|
alloc.id,
|
|
expectedAllocs[index].id,
|
|
`Allocation at ${index} is ${expectedAllocs[index].id}`
|
|
);
|
|
});
|
|
});
|
|
|
|
test(`facet ${label} | selecting options in the ${label} facet updates the ${paramName} query param`, async function (assert) {
|
|
const selection = [];
|
|
|
|
await beforeEach();
|
|
await facet.toggle();
|
|
|
|
const option1 = facet.options.objectAt(0);
|
|
const option2 = facet.options.objectAt(1);
|
|
await option1.toggle();
|
|
selection.push(option1.key);
|
|
await option2.toggle();
|
|
selection.push(option2.key);
|
|
|
|
assert.equal(
|
|
currentURL(),
|
|
`/jobs/${job.id}/allocations?${paramName}=${encodeURIComponent(
|
|
JSON.stringify(selection)
|
|
)}`,
|
|
'URL has the correct query param key and value'
|
|
);
|
|
});
|
|
}
|