d559072e48
* Text and code wrapping as a localStorage var * task-log uses wrapping and kb shortcut * Word wrap keyboard labels * Wrapper as a toggle not a button * Changelog and fixed an extra space trailing log lines * Moves toggle to inside * Acceptance tests for ww and toggle click
184 lines
5.4 KiB
JavaScript
184 lines
5.4 KiB
JavaScript
/**
|
|
* Copyright (c) HashiCorp, Inc.
|
|
* SPDX-License-Identifier: MPL-2.0
|
|
*/
|
|
|
|
/* eslint-disable qunit/require-expect */
|
|
import {
|
|
click,
|
|
currentURL,
|
|
findAll,
|
|
triggerKeyEvent,
|
|
} from '@ember/test-helpers';
|
|
import { run } from '@ember/runloop';
|
|
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 TaskLogs from 'nomad-ui/tests/pages/allocations/task/logs';
|
|
import percySnapshot from '@percy/ember';
|
|
import faker from 'nomad-ui/mirage/faker';
|
|
|
|
let allocation;
|
|
let task;
|
|
let job;
|
|
|
|
module('Acceptance | task logs', function (hooks) {
|
|
setupApplicationTest(hooks);
|
|
setupMirage(hooks);
|
|
|
|
hooks.beforeEach(async function () {
|
|
faker.seed(1);
|
|
server.create('agent');
|
|
server.create('node-pool');
|
|
server.create('node', 'forceIPv4');
|
|
job = server.create('job', { createAllocations: false });
|
|
|
|
allocation = server.create('allocation', {
|
|
jobId: job.id,
|
|
clientStatus: 'running',
|
|
});
|
|
task = server.db.taskStates.where({ allocationId: allocation.id })[0];
|
|
|
|
run.later(run, run.cancelTimers, 1000);
|
|
});
|
|
|
|
test('it passes an accessibility audit', async function (assert) {
|
|
await TaskLogs.visit({ id: allocation.id, name: task.name });
|
|
await a11yAudit(assert);
|
|
await percySnapshot(assert);
|
|
});
|
|
|
|
test('/allocation/:id/:task_name/logs should have a log component', async function (assert) {
|
|
await TaskLogs.visit({ id: allocation.id, name: task.name });
|
|
assert.equal(
|
|
currentURL(),
|
|
`/allocations/${allocation.id}/${task.name}/logs`,
|
|
'No redirect'
|
|
);
|
|
assert.ok(TaskLogs.hasTaskLog, 'Task log component found');
|
|
assert.ok(document.title.includes(`Task ${task.name}`));
|
|
});
|
|
|
|
test('the stdout log immediately starts streaming', async function (assert) {
|
|
await TaskLogs.visit({ id: allocation.id, name: task.name });
|
|
const node = server.db.nodes.find(allocation.nodeId);
|
|
const logUrlRegex = new RegExp(
|
|
`${node.httpAddr}/v1/client/fs/logs/${allocation.id}`
|
|
);
|
|
assert.ok(
|
|
server.pretender.handledRequests.filter((req) =>
|
|
logUrlRegex.test(req.url)
|
|
).length,
|
|
'Log requests were made'
|
|
);
|
|
});
|
|
|
|
test('logs can be word-wrapped', async function (assert) {
|
|
await TaskLogs.visit({ id: allocation.id, name: task.name });
|
|
|
|
assert.dom('[data-test-word-wrap-toggle]').isNotChecked();
|
|
assert.dom('[data-test-output]').doesNotHaveClass('wrapped');
|
|
|
|
run.later(() => {
|
|
run.cancelTimers();
|
|
}, 100);
|
|
await click('[data-test-word-wrap-toggle]');
|
|
assert.dom('[data-test-word-wrap-toggle]').isChecked();
|
|
assert.dom('[data-test-output]').hasClass('wrapped');
|
|
|
|
run.later(() => {
|
|
run.cancelTimers();
|
|
}, 100);
|
|
await click('[data-test-word-wrap-toggle]');
|
|
assert.dom('[data-test-word-wrap-toggle]').isNotChecked();
|
|
assert.dom('[data-test-output]').doesNotHaveClass('wrapped');
|
|
|
|
window.localStorage.clear();
|
|
});
|
|
|
|
test('logs in sidebar can be word-wrapped', async function (assert) {
|
|
await TaskLogs.visitParentJob({
|
|
id: job.id,
|
|
allocationId: allocation.id,
|
|
name: task.name,
|
|
});
|
|
|
|
run.later(() => {
|
|
run.cancelTimers();
|
|
}, 500);
|
|
|
|
const taskRow = [
|
|
...findAll('.task-sub-row').filter((row) => {
|
|
return row.textContent.includes(task.name);
|
|
}),
|
|
][0];
|
|
|
|
await click(taskRow.querySelector('button.logs-sidebar-trigger'));
|
|
|
|
assert.dom('[data-test-word-wrap-toggle]').isNotChecked();
|
|
assert.dom('[data-test-output]').doesNotHaveClass('wrapped');
|
|
|
|
run.later(() => {
|
|
run.cancelTimers();
|
|
}, 500);
|
|
|
|
// type "ww" to trigger word wrap
|
|
const W_KEY = 87;
|
|
triggerKeyEvent('.sidebar', 'keydown', W_KEY);
|
|
await triggerKeyEvent('.sidebar', 'keydown', W_KEY);
|
|
|
|
assert.dom('[data-test-word-wrap-toggle]').isChecked();
|
|
assert.dom('[data-test-output]').hasClass('wrapped');
|
|
|
|
run.later(() => {
|
|
run.cancelTimers();
|
|
}, 100);
|
|
|
|
triggerKeyEvent('.sidebar', 'keydown', W_KEY);
|
|
await triggerKeyEvent('.sidebar', 'keydown', W_KEY);
|
|
assert.dom('[data-test-word-wrap-toggle]').isNotChecked();
|
|
assert.dom('[data-test-output]').doesNotHaveClass('wrapped');
|
|
|
|
window.localStorage.clear();
|
|
});
|
|
|
|
test('logs are accessible in a sidebar context', async function (assert) {
|
|
await TaskLogs.visitParentJob({
|
|
id: job.id,
|
|
allocationId: allocation.id,
|
|
name: task.name,
|
|
});
|
|
assert.notOk(TaskLogs.sidebarIsPresent, 'Sidebar is not present');
|
|
|
|
run.later(() => {
|
|
run.cancelTimers();
|
|
}, 500);
|
|
|
|
const taskRow = [
|
|
...findAll('.task-sub-row').filter((row) => {
|
|
return row.textContent.includes(task.name);
|
|
}),
|
|
][0];
|
|
|
|
await click(taskRow.querySelector('button.logs-sidebar-trigger'));
|
|
|
|
assert.ok(TaskLogs.sidebarIsPresent, 'Sidebar is present');
|
|
assert
|
|
.dom('.task-context-sidebar h1.title')
|
|
.includesText(task.name, 'Sidebar title is correct');
|
|
assert
|
|
.dom('.task-context-sidebar h1.title')
|
|
.includesText(task.state, 'Task state is correctly displayed');
|
|
await percySnapshot(assert, {
|
|
percyCSS: `
|
|
.allocation-row td { display: none; }
|
|
.task-events table td:nth-child(1) { color: transparent; }
|
|
`,
|
|
});
|
|
|
|
await click('.sidebar button.close');
|
|
assert.notOk(TaskLogs.sidebarIsPresent, 'Sidebar is not present');
|
|
});
|
|
});
|