open-nomad/ui/tests/integration/components/task-group-row-test.js
2022-01-20 09:46:45 -05:00

188 lines
6.4 KiB
JavaScript

import { module, test } from 'qunit';
import { setupRenderingTest } from 'ember-qunit';
import { click, find, render, settled, waitUntil } from '@ember/test-helpers';
import hbs from 'htmlbars-inline-precompile';
import { startMirage } from 'nomad-ui/initializers/ember-cli-mirage';
import { initialize as fragmentSerializerInitializer } from 'nomad-ui/initializers/fragment-serializer';
import { componentA11yAudit } from 'nomad-ui/tests/helpers/a11y-audit';
const jobName = 'test-job';
const jobId = JSON.stringify([jobName, 'default']);
const countChange = () => {
const initial = find('[data-test-task-group-count]').textContent;
return () => find('[data-test-task-group-count]').textContent !== initial;
};
let managementToken;
let clientToken;
const makeJob = (server, props = {}) => {
// These tests require a job with particular task groups. This requires
// mild Mirage surgery.
const job = server.create('job', {
id: jobName,
groupCount: 0,
createAllocations: false,
shallow: true,
...props,
});
const noScalingGroup = server.create('task-group', {
job,
name: 'no-scaling',
shallow: true,
withScaling: false,
});
const scalingGroup = server.create('task-group', {
job,
count: 2,
name: 'scaling',
shallow: true,
withScaling: true,
});
job.update({
taskGroupIds: [noScalingGroup.id, scalingGroup.id],
});
};
module('Integration | Component | task group row', function (hooks) {
setupRenderingTest(hooks);
hooks.beforeEach(async function () {
fragmentSerializerInitializer(this.owner);
this.store = this.owner.lookup('service:store');
this.token = this.owner.lookup('service:token');
this.server = startMirage();
this.server.create('node');
managementToken = this.server.create('token');
clientToken = this.server.create('token');
window.localStorage.nomadTokenSecret = managementToken.secretId;
});
hooks.afterEach(function () {
this.server.shutdown();
window.localStorage.clear();
});
const commonTemplate = hbs`
<TaskGroupRow @taskGroup={{group}} />
`;
test('Task group row conditionally shows scaling buttons based on the presence of the scaling attr on the task group', async function (assert) {
makeJob(this.server, { noActiveDeployment: true });
this.token.fetchSelfTokenAndPolicies.perform();
await settled();
const job = await this.store.find('job', jobId);
this.set('group', job.taskGroups.findBy('name', 'no-scaling'));
await render(commonTemplate);
assert.notOk(find('[data-test-scale]'));
this.set('group', job.taskGroups.findBy('name', 'scaling'));
await settled();
assert.ok(find('[data-test-scale]'));
await componentA11yAudit(this.element, assert);
});
test('Clicking scaling buttons immediately updates the rendered count but debounces the scaling API request', async function (assert) {
makeJob(this.server, { noActiveDeployment: true });
this.token.fetchSelfTokenAndPolicies.perform();
await settled();
const job = await this.store.find('job', jobId);
this.set('group', job.taskGroups.findBy('name', 'scaling'));
await render(commonTemplate);
assert.equal(find('[data-test-task-group-count]').textContent, 2);
click('[data-test-scale="increment"]');
await waitUntil(countChange());
assert.equal(find('[data-test-task-group-count]').textContent, 3);
click('[data-test-scale="increment"]');
await waitUntil(countChange());
assert.equal(find('[data-test-task-group-count]').textContent, 4);
assert.notOk(
server.pretender.handledRequests.find(
(req) => req.method === 'POST' && req.url.endsWith('/scale')
)
);
await settled();
const scaleRequests = server.pretender.handledRequests.filter(
(req) => req.method === 'POST' && req.url.endsWith('/scale')
);
assert.equal(scaleRequests.length, 1);
assert.equal(JSON.parse(scaleRequests[0].requestBody).Count, 4);
});
test('When the current count is equal to the max count, the increment count button is disabled', async function (assert) {
makeJob(this.server, { noActiveDeployment: true });
this.token.fetchSelfTokenAndPolicies.perform();
await settled();
const job = await this.store.find('job', jobId);
const group = job.taskGroups.findBy('name', 'scaling');
group.set('count', group.scaling.max);
this.set('group', group);
await render(commonTemplate);
assert.ok(find('[data-test-scale="increment"]:disabled'));
await componentA11yAudit(this.element, assert);
});
test('When the current count is equal to the min count, the decrement count button is disabled', async function (assert) {
makeJob(this.server, { noActiveDeployment: true });
this.token.fetchSelfTokenAndPolicies.perform();
await settled();
const job = await this.store.find('job', jobId);
const group = job.taskGroups.findBy('name', 'scaling');
group.set('count', group.scaling.min);
this.set('group', group);
await render(commonTemplate);
assert.ok(find('[data-test-scale="decrement"]:disabled'));
await componentA11yAudit(this.element, assert);
});
test('When there is an active deployment, both scale buttons are disabled', async function (assert) {
makeJob(this.server, { activeDeployment: true });
this.token.fetchSelfTokenAndPolicies.perform();
await settled();
const job = await this.store.find('job', jobId);
this.set('group', job.taskGroups.findBy('name', 'scaling'));
await render(commonTemplate);
assert.ok(find('[data-test-scale="increment"]:disabled'));
assert.ok(find('[data-test-scale="decrement"]:disabled'));
await componentA11yAudit(this.element, assert);
});
test('When the current ACL token does not have the namespace:scale-job or namespace:submit-job policy rule', async function (assert) {
makeJob(this.server, { noActiveDeployment: true });
window.localStorage.nomadTokenSecret = clientToken.secretId;
this.token.fetchSelfTokenAndPolicies.perform();
await settled();
const job = await this.store.find('job', jobId);
this.set('group', job.taskGroups.findBy('name', 'scaling'));
await render(commonTemplate);
assert.ok(find('[data-test-scale="increment"]:disabled'));
assert.ok(find('[data-test-scale="decrement"]:disabled'));
assert.ok(
find('[data-test-scale-controls]').getAttribute('aria-label').includes("You aren't allowed")
);
});
});