mirror of
https://github.com/nasa/openmct.git
synced 2025-01-28 07:04:10 +00:00
a3fb84ad43
* fix: remove mystery webpack code * fix: remove type:module and specify exports - we aren't a module... yet * fix: rename webpack*.js to webpack*.mjs so we can use import/export. fix imports * fix: exports format * fix: woops, need to add `start` script back * chore: split e2e into its own module * fix: use normal Painterro import * fix: update e2e pathing * fix: copy over helper functions * chore: specify `cwd` for playwright configs so that openmct npm commands work as intended in any environment * chore: add pretest script to e2e package.json * chore: don't package e2e * refactor: tidy up webpack common config * chore: compile types to a single file * chore: fix visual test npm scripts * chore: fix import pathing * chore: define package exports, move test specific dependencies to the subpackage * chore: export test framework from openmct-e2e * chore: export baseFixtures also * chore: let `openmct` and `openmct-e2e` share `node_modules/` * chore: use `--workspace`, remove pretest script * Revert "fix: remove mystery webpack code" This reverts commit eb14d52569ffa27ab1a090b883694f4707b59cd0. * chore: update package-lock * chore: add `.npmignore` * fix: *js -> mjs
189 lines
6.4 KiB
JavaScript
189 lines
6.4 KiB
JavaScript
/*****************************************************************************
|
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
|
* as represented by the Administrator of the National Aeronautics and Space
|
|
* Administration. All rights reserved.
|
|
*
|
|
* Open MCT is licensed under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
* http://www.apache.org/licenses/LICENSE-2.0.
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
* Open MCT includes source code licensed under additional open source
|
|
* licenses. See the Open Source Licenses file (LICENSES.md) included with
|
|
* this source code distribution or the Licensing information page available
|
|
* at runtime from the About dialog for additional information.
|
|
*****************************************************************************/
|
|
import fs from 'fs';
|
|
|
|
import { createDomainObjectWithDefaults, createPlanFromJSON } from '../../../appActions.js';
|
|
import {
|
|
assertPlanActivities,
|
|
setBoundsToSpanAllActivities
|
|
} from '../../../helper/planningUtils.js';
|
|
import { expect, test } from '../../../pluginFixtures.js';
|
|
|
|
const testPlan1 = JSON.parse(
|
|
fs.readFileSync(
|
|
new URL('../../../test-data/examplePlans/ExamplePlan_Small1.json', import.meta.url)
|
|
)
|
|
);
|
|
const testPlan2 = JSON.parse(
|
|
fs.readFileSync(
|
|
new URL('../../../test-data/examplePlans/ExamplePlan_Small2.json', import.meta.url)
|
|
)
|
|
);
|
|
|
|
test.describe('Gantt Chart', () => {
|
|
let ganttChart;
|
|
let plan;
|
|
test.beforeEach(async ({ page }) => {
|
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
|
ganttChart = await createDomainObjectWithDefaults(page, {
|
|
type: 'Gantt Chart'
|
|
});
|
|
plan = await createPlanFromJSON(page, {
|
|
json: testPlan1,
|
|
parent: ganttChart.uuid
|
|
});
|
|
});
|
|
|
|
test('Displays all plan events', async ({ page }) => {
|
|
await page.goto(ganttChart.url);
|
|
|
|
await assertPlanActivities(page, testPlan1, ganttChart.url);
|
|
});
|
|
test('Replaces a plan with a new plan', async ({ page }) => {
|
|
await assertPlanActivities(page, testPlan1, ganttChart.url);
|
|
await createPlanFromJSON(page, {
|
|
json: testPlan2,
|
|
parent: ganttChart.uuid
|
|
});
|
|
const replaceModal = page
|
|
.getByRole('dialog')
|
|
.filter({ hasText: 'This action will replace the current Plan. Do you want to continue?' });
|
|
await expect(replaceModal).toBeVisible();
|
|
await page.getByRole('button', { name: 'Ok', exact: true }).click();
|
|
|
|
await assertPlanActivities(page, testPlan2, ganttChart.url);
|
|
});
|
|
test('Can select a single activity and display its details in the inspector', async ({
|
|
page
|
|
}) => {
|
|
test.slow();
|
|
await page.goto(ganttChart.url);
|
|
|
|
await setBoundsToSpanAllActivities(page, testPlan1, ganttChart.url);
|
|
|
|
const activities = Object.values(testPlan1).flat();
|
|
const activity = activities[0];
|
|
await page
|
|
.locator('g')
|
|
.filter({ hasText: new RegExp(activity.name) })
|
|
.click();
|
|
await page.getByRole('tab', { name: 'Activity' }).click();
|
|
|
|
const startDateTime = await page
|
|
.locator(
|
|
'.c-inspect-properties__label:has-text("Start DateTime")+.c-inspect-properties__value'
|
|
)
|
|
.innerText();
|
|
const endDateTime = await page
|
|
.locator('.c-inspect-properties__label:has-text("End DateTime")+.c-inspect-properties__value')
|
|
.innerText();
|
|
const duration = await page
|
|
.locator('.c-inspect-properties__label:has-text("duration")+.c-inspect-properties__value')
|
|
.innerText();
|
|
|
|
const expectedStartDate = new Date(activity.start).toISOString();
|
|
const actualStartDate = new Date(startDateTime).toISOString();
|
|
const expectedEndDate = new Date(activity.end).toISOString();
|
|
const actualEndDate = new Date(endDateTime).toISOString();
|
|
const expectedDuration = getPreciseDuration(activity.end - activity.start);
|
|
const actualDuration = duration;
|
|
|
|
expect(expectedStartDate).toEqual(actualStartDate);
|
|
expect(expectedEndDate).toEqual(actualEndDate);
|
|
expect(expectedDuration).toEqual(actualDuration);
|
|
});
|
|
test("Displays a Plan's draft status", async ({ page }) => {
|
|
test.info().annotations.push({
|
|
type: 'issue',
|
|
description: 'https://github.com/nasa/openmct/issues/6641'
|
|
});
|
|
|
|
// Mark the Plan's status as draft in the OpenMCT API
|
|
await page.evaluate(async (planObject) => {
|
|
await window.openmct.status.set(planObject.uuid, 'draft');
|
|
}, plan);
|
|
|
|
// Navigate to the Gantt Chart
|
|
await page.goto(ganttChart.url);
|
|
|
|
// Assert that the Plan's status is displayed as draft
|
|
expect(await page.locator('.u-contents.c-swimlane.is-status--draft').count()).toBe(
|
|
Object.keys(testPlan1).length
|
|
);
|
|
});
|
|
});
|
|
|
|
const ONE_SECOND = 1000;
|
|
const ONE_MINUTE = 60 * ONE_SECOND;
|
|
const ONE_HOUR = ONE_MINUTE * 60;
|
|
const ONE_DAY = ONE_HOUR * 24;
|
|
|
|
function normalizeAge(num) {
|
|
const hundredtized = num * 100;
|
|
const isWhole = hundredtized % 100 === 0;
|
|
|
|
return isWhole ? hundredtized / 100 : num;
|
|
}
|
|
|
|
function padLeadingZeros(num, numOfLeadingZeros) {
|
|
return num.toString().padStart(numOfLeadingZeros, '0');
|
|
}
|
|
|
|
function toDoubleDigits(num) {
|
|
return padLeadingZeros(num, 2);
|
|
}
|
|
|
|
function toTripleDigits(num) {
|
|
return padLeadingZeros(num, 3);
|
|
}
|
|
|
|
function getPreciseDuration(value, { excludeMilliSeconds, useDayFormat } = {}) {
|
|
let preciseDuration;
|
|
const ms = value || 0;
|
|
|
|
const duration = [
|
|
Math.floor(normalizeAge(ms / ONE_DAY)),
|
|
toDoubleDigits(Math.floor(normalizeAge((ms % ONE_DAY) / ONE_HOUR))),
|
|
toDoubleDigits(Math.floor(normalizeAge((ms % ONE_HOUR) / ONE_MINUTE))),
|
|
toDoubleDigits(Math.floor(normalizeAge((ms % ONE_MINUTE) / ONE_SECOND)))
|
|
];
|
|
if (!excludeMilliSeconds) {
|
|
duration.push(toTripleDigits(Math.floor(normalizeAge(ms % ONE_SECOND))));
|
|
}
|
|
|
|
if (useDayFormat) {
|
|
// Format days as XD
|
|
const days = duration.shift();
|
|
if (days > 0) {
|
|
preciseDuration = `${days}D ${duration.join(':')}`;
|
|
} else {
|
|
preciseDuration = duration.join(':');
|
|
}
|
|
} else {
|
|
const days = toDoubleDigits(duration.shift());
|
|
duration.unshift(days);
|
|
preciseDuration = duration.join(':');
|
|
}
|
|
|
|
return preciseDuration;
|
|
}
|