mirror of
https://github.com/nasa/openmct.git
synced 2024-12-18 20:57:53 +00:00
Merge branch 'master' into mct7175
This commit is contained in:
commit
89c463da52
@ -2,23 +2,23 @@ version: 2.1
|
|||||||
executors:
|
executors:
|
||||||
pw-focal-development:
|
pw-focal-development:
|
||||||
docker:
|
docker:
|
||||||
- image: mcr.microsoft.com/playwright:v1.36.2-focal
|
- image: mcr.microsoft.com/playwright:v1.39.0-focal
|
||||||
environment:
|
environment:
|
||||||
NODE_ENV: development # Needed to ensure 'dist' folder created and devDependencies installed
|
NODE_ENV: development # Needed to ensure 'dist' folder created and devDependencies installed
|
||||||
PERCY_POSTINSTALL_BROWSER: 'true' # Needed to store the percy browser in cache deps
|
PERCY_POSTINSTALL_BROWSER: "true" # Needed to store the percy browser in cache deps
|
||||||
PERCY_LOGLEVEL: 'debug' # Enable DEBUG level logging for Percy (Issue: https://github.com/nasa/openmct/issues/5742)
|
PERCY_LOGLEVEL: "debug" # Enable DEBUG level logging for Percy (Issue: https://github.com/nasa/openmct/issues/5742)
|
||||||
ubuntu:
|
ubuntu:
|
||||||
machine:
|
machine:
|
||||||
image: ubuntu-2204:current
|
image: ubuntu-2204:current
|
||||||
docker_layer_caching: true
|
docker_layer_caching: true
|
||||||
parameters:
|
parameters:
|
||||||
BUST_CACHE:
|
BUST_CACHE:
|
||||||
description: 'Set this with the CircleCI UI Trigger Workflow button (boolean = true) to bust the cache!'
|
description: "Set this with the CircleCI UI Trigger Workflow button (boolean = true) to bust the cache!"
|
||||||
default: false
|
default: false
|
||||||
type: boolean
|
type: boolean
|
||||||
commands:
|
commands:
|
||||||
build_and_install:
|
build_and_install:
|
||||||
description: 'All steps used to build and install. Will use cache if found'
|
description: "All steps used to build and install. Will use cache if found"
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
node-version:
|
||||||
type: string
|
type: string
|
||||||
@ -30,7 +30,7 @@ commands:
|
|||||||
node-version: << parameters.node-version >>
|
node-version: << parameters.node-version >>
|
||||||
- run: npm install --no-audit --progress=false
|
- run: npm install --no-audit --progress=false
|
||||||
restore_cache_cmd:
|
restore_cache_cmd:
|
||||||
description: 'Custom command for restoring cache with the ability to bust cache. When BUST_CACHE is set to true, jobs will not restore cache'
|
description: "Custom command for restoring cache with the ability to bust cache. When BUST_CACHE is set to true, jobs will not restore cache"
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
node-version:
|
||||||
type: string
|
type: string
|
||||||
@ -42,7 +42,7 @@ commands:
|
|||||||
- restore_cache:
|
- restore_cache:
|
||||||
key: deps--{{ arch }}--{{ .Branch }}--<< parameters.node-version >>--{{ checksum "package.json" }}-{{ checksum ".circleci/config.yml" }}
|
key: deps--{{ arch }}--{{ .Branch }}--<< parameters.node-version >>--{{ checksum "package.json" }}-{{ checksum ".circleci/config.yml" }}
|
||||||
save_cache_cmd:
|
save_cache_cmd:
|
||||||
description: 'Custom command for saving cache.'
|
description: "Custom command for saving cache."
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
node-version:
|
||||||
type: string
|
type: string
|
||||||
@ -53,7 +53,7 @@ commands:
|
|||||||
- ~/.npm
|
- ~/.npm
|
||||||
- node_modules
|
- node_modules
|
||||||
generate_and_store_version_and_filesystem_artifacts:
|
generate_and_store_version_and_filesystem_artifacts:
|
||||||
description: 'Track important packages and files'
|
description: "Track important packages and files"
|
||||||
steps:
|
steps:
|
||||||
- run: |
|
- run: |
|
||||||
[[ $EUID -ne 0 ]] && (sudo mkdir -p /tmp/artifacts && sudo chmod 777 /tmp/artifacts) || (mkdir -p /tmp/artifacts && chmod 777 /tmp/artifacts)
|
[[ $EUID -ne 0 ]] && (sudo mkdir -p /tmp/artifacts && sudo chmod 777 /tmp/artifacts) || (mkdir -p /tmp/artifacts && chmod 777 /tmp/artifacts)
|
||||||
@ -64,7 +64,7 @@ commands:
|
|||||||
- store_artifacts:
|
- store_artifacts:
|
||||||
path: /tmp/artifacts/
|
path: /tmp/artifacts/
|
||||||
generate_e2e_code_cov_report:
|
generate_e2e_code_cov_report:
|
||||||
description: 'Generate e2e code coverage artifacts and publish to codecov.io. Needed to that we can ignore the exit code status of the npm run test'
|
description: "Generate e2e code coverage artifacts and publish to codecov.io. Needed to that we can ignore the exit code status of the npm run test"
|
||||||
parameters:
|
parameters:
|
||||||
suite:
|
suite:
|
||||||
type: string
|
type: string
|
||||||
@ -105,7 +105,11 @@ jobs:
|
|||||||
node-version: <<parameters.node-version>>
|
node-version: <<parameters.node-version>>
|
||||||
- browser-tools/install-chrome:
|
- browser-tools/install-chrome:
|
||||||
replace-existing: false
|
replace-existing: false
|
||||||
- run: npm run test
|
- run:
|
||||||
|
command: |
|
||||||
|
mkdir -p dist/reports/tests/
|
||||||
|
TESTFILES=$(circleci tests glob "src/**/*Spec.js")
|
||||||
|
echo "$TESTFILES" | circleci tests run --command="xargs npm run test" --verbose
|
||||||
- run: npm run cov:unit:publish
|
- run: npm run cov:unit:publish
|
||||||
- save_cache_cmd:
|
- save_cache_cmd:
|
||||||
node-version: <<parameters.node-version>>
|
node-version: <<parameters.node-version>>
|
||||||
@ -120,21 +124,23 @@ jobs:
|
|||||||
- generate_and_store_version_and_filesystem_artifacts
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
e2e-test:
|
e2e-test:
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
|
||||||
type: string
|
|
||||||
suite: #stable or full
|
suite: #stable or full
|
||||||
type: string
|
type: string
|
||||||
executor: pw-focal-development
|
executor: pw-focal-development
|
||||||
parallelism: 4
|
parallelism: 7
|
||||||
steps:
|
steps:
|
||||||
- build_and_install:
|
- build_and_install:
|
||||||
node-version: <<parameters.node-version>>
|
node-version: lts/hydrogen
|
||||||
- when: #Only install chrome-beta when running the 'full' suite to save $$$
|
- when: #Only install chrome-beta when running the 'full' suite to save $$$
|
||||||
condition:
|
condition:
|
||||||
equal: ['full', <<parameters.suite>>]
|
equal: ["full", <<parameters.suite>>]
|
||||||
steps:
|
steps:
|
||||||
- run: npx playwright install chrome-beta
|
- run: npx playwright install chrome-beta
|
||||||
- run: SHARD="$((${CIRCLE_NODE_INDEX}+1))"; npm run test:e2e:<<parameters.suite>> -- --shard=${SHARD}/${CIRCLE_NODE_TOTAL}
|
- run:
|
||||||
|
command: |
|
||||||
|
mkdir test-results
|
||||||
|
TESTFILES=$(circleci tests glob "e2e/**/*.spec.js")
|
||||||
|
echo "$TESTFILES" | circleci tests run --command="xargs npm run test:e2e:<<parameters.suite>>" --verbose --split-by=timings
|
||||||
- when:
|
- when:
|
||||||
condition:
|
condition:
|
||||||
equal: [42, 42] # Always run codecov reports regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
equal: [42, 42] # Always run codecov reports regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
||||||
@ -155,14 +161,11 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- generate_and_store_version_and_filesystem_artifacts
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
e2e-couchdb:
|
e2e-couchdb:
|
||||||
parameters:
|
|
||||||
node-version:
|
|
||||||
type: string
|
|
||||||
executor: ubuntu
|
executor: ubuntu
|
||||||
steps:
|
steps:
|
||||||
- build_and_install:
|
- build_and_install:
|
||||||
node-version: <<parameters.node-version>>
|
node-version: lts/hydrogen
|
||||||
- run: npx playwright@1.36.2 install #Necessary for bare ubuntu machine
|
- run: npx playwright@1.39.0 install #Necessary for bare ubuntu machine
|
||||||
- run: |
|
- run: |
|
||||||
export $(cat src/plugins/persistence/couch/.env.ci | xargs)
|
export $(cat src/plugins/persistence/couch/.env.ci | xargs)
|
||||||
docker-compose -f src/plugins/persistence/couch/couchdb-compose.yaml up --detach
|
docker-compose -f src/plugins/persistence/couch/couchdb-compose.yaml up --detach
|
||||||
@ -189,15 +192,28 @@ jobs:
|
|||||||
equal: [42, 42] # Always generate version artifacts regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
equal: [42, 42] # Always generate version artifacts regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
||||||
steps:
|
steps:
|
||||||
- generate_and_store_version_and_filesystem_artifacts
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
perf-test:
|
mem-test:
|
||||||
parameters:
|
|
||||||
node-version:
|
|
||||||
type: string
|
|
||||||
executor: pw-focal-development
|
executor: pw-focal-development
|
||||||
steps:
|
steps:
|
||||||
- build_and_install:
|
- build_and_install:
|
||||||
node-version: <<parameters.node-version>>
|
node-version: lts/hydrogen
|
||||||
- run: npm run test:perf:memory
|
- run: npm run test:perf:memory
|
||||||
|
- store_test_results:
|
||||||
|
path: test-results/results.xml
|
||||||
|
- store_artifacts:
|
||||||
|
path: test-results
|
||||||
|
- store_artifacts:
|
||||||
|
path: html-test-results
|
||||||
|
- when:
|
||||||
|
condition:
|
||||||
|
equal: [42, 42] # Always run codecov reports regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
||||||
|
steps:
|
||||||
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
|
perf-test:
|
||||||
|
executor: pw-focal-development
|
||||||
|
steps:
|
||||||
|
- build_and_install:
|
||||||
|
node-version: lts/hydrogen
|
||||||
- run: npm run test:perf:localhost
|
- run: npm run test:perf:localhost
|
||||||
- run: npm run test:perf:contract
|
- run: npm run test:perf:contract
|
||||||
- store_test_results:
|
- store_test_results:
|
||||||
@ -211,16 +227,14 @@ jobs:
|
|||||||
equal: [42, 42] # Always run codecov reports regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
equal: [42, 42] # Always run codecov reports regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
||||||
steps:
|
steps:
|
||||||
- generate_and_store_version_and_filesystem_artifacts
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
visual-test:
|
visual-a11y-tests:
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
|
||||||
type: string
|
|
||||||
suite:
|
suite:
|
||||||
type: string # ci or full
|
type: string # ci or full
|
||||||
executor: pw-focal-development
|
executor: pw-focal-development
|
||||||
steps:
|
steps:
|
||||||
- build_and_install:
|
- build_and_install:
|
||||||
node-version: <<parameters.node-version>>
|
node-version: lts/hydrogen
|
||||||
- run: npm run test:e2e:visual:<<parameters.suite>>
|
- run: npm run test:e2e:visual:<<parameters.suite>>
|
||||||
- store_test_results:
|
- store_test_results:
|
||||||
path: test-results/results.xml
|
path: test-results/results.xml
|
||||||
@ -233,31 +247,28 @@ jobs:
|
|||||||
equal: [42, 42] # Always generate version artifacts regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
equal: [42, 42] # Always generate version artifacts regardless of test failure https://discuss.circleci.com/t/make-custom-command-run-always-with-when-always/38957/2
|
||||||
steps:
|
steps:
|
||||||
- generate_and_store_version_and_filesystem_artifacts
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
|
|
||||||
workflows:
|
workflows:
|
||||||
overall-circleci-commit-status: #These jobs run on every commit
|
overall-circleci-commit-status: #These jobs run on every commit
|
||||||
jobs:
|
jobs:
|
||||||
- lint:
|
- lint:
|
||||||
name: node16-lint
|
name: node20-lint
|
||||||
node-version: lts/gallium
|
node-version: lts/iron
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node18-chrome
|
name: node18-chrome
|
||||||
node-version: lts/hydrogen
|
node-version: lts/hydrogen
|
||||||
- e2e-test:
|
- e2e-test:
|
||||||
name: e2e-stable
|
name: e2e-stable
|
||||||
node-version: lts/hydrogen
|
|
||||||
suite: stable
|
suite: stable
|
||||||
- perf-test:
|
- visual-a11y-tests:
|
||||||
node-version: lts/hydrogen
|
|
||||||
- visual-test:
|
|
||||||
name: visual-test-ci
|
name: visual-test-ci
|
||||||
suite: ci
|
suite: ci
|
||||||
node-version: lts/hydrogen
|
|
||||||
|
|
||||||
the-nightly: #These jobs do not run on PRs, but against master at night
|
the-nightly: #These jobs do not run on PRs, but against master at night
|
||||||
jobs:
|
jobs:
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node16-chrome-nightly
|
name: node20-chrome-nightly
|
||||||
node-version: lts/gallium
|
node-version: lts/iron
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node18-chrome
|
name: node18-chrome
|
||||||
node-version: lts/hydrogen
|
node-version: lts/hydrogen
|
||||||
@ -265,19 +276,16 @@ workflows:
|
|||||||
node-version: lts/hydrogen
|
node-version: lts/hydrogen
|
||||||
- e2e-test:
|
- e2e-test:
|
||||||
name: e2e-full-nightly
|
name: e2e-full-nightly
|
||||||
node-version: lts/hydrogen
|
|
||||||
suite: full
|
suite: full
|
||||||
- perf-test:
|
- mem-test
|
||||||
node-version: lts/hydrogen
|
- perf-test
|
||||||
- visual-test:
|
- visual-a11y-tests:
|
||||||
name: visual-test-nightly
|
name: visual-test-nightly
|
||||||
suite: full
|
suite: full
|
||||||
node-version: lts/hydrogen
|
- e2e-couchdb
|
||||||
- e2e-couchdb:
|
|
||||||
node-version: lts/hydrogen
|
|
||||||
triggers:
|
triggers:
|
||||||
- schedule:
|
- schedule:
|
||||||
cron: '0 0 * * *'
|
cron: "0 0 * * *"
|
||||||
filters:
|
filters:
|
||||||
branches:
|
branches:
|
||||||
only:
|
only:
|
||||||
|
16
.cspell.json
16
.cspell.json
@ -43,7 +43,6 @@
|
|||||||
"sharded",
|
"sharded",
|
||||||
"perfromance",
|
"perfromance",
|
||||||
"MMOC",
|
"MMOC",
|
||||||
"deploysentinel",
|
|
||||||
"codegen",
|
"codegen",
|
||||||
"Unfortuantely",
|
"Unfortuantely",
|
||||||
"viewports",
|
"viewports",
|
||||||
@ -487,9 +486,20 @@
|
|||||||
"blockquote",
|
"blockquote",
|
||||||
"blockquotes",
|
"blockquotes",
|
||||||
"Blockquote",
|
"Blockquote",
|
||||||
"Blockquotes"
|
"Blockquotes",
|
||||||
|
"oger",
|
||||||
|
"lcovonly",
|
||||||
|
"gcov",
|
||||||
|
"WCAG",
|
||||||
|
"stackedplot",
|
||||||
|
"Andale",
|
||||||
|
"unnormalized",
|
||||||
|
"checksnapshots",
|
||||||
|
"specced",
|
||||||
|
"composables",
|
||||||
|
"countup"
|
||||||
],
|
],
|
||||||
"dictionaries": ["npm", "softwareTerms", "node", "html", "css", "bash", "en_US"],
|
"dictionaries": ["npm", "softwareTerms", "node", "html", "css", "bash", "en_US", "en-gb", "misc"],
|
||||||
"ignorePaths": [
|
"ignorePaths": [
|
||||||
"package.json",
|
"package.json",
|
||||||
"dist/**",
|
"dist/**",
|
||||||
|
@ -4,6 +4,10 @@
|
|||||||
# Requires Git > 2.23
|
# Requires Git > 2.23
|
||||||
# See https://git-scm.com/docs/git-blame#Documentation/git-blame.txt---ignore-revs-fileltfilegt
|
# See https://git-scm.com/docs/git-blame#Documentation/git-blame.txt---ignore-revs-fileltfilegt
|
||||||
|
|
||||||
|
# vue-eslint update 2019
|
||||||
|
14a0f84c1bcd56886d7c9e4e6afa8f7d292734e5
|
||||||
|
# eslint changes 2022
|
||||||
|
d80b6923541704ab925abf0047cbbc58735c27e2
|
||||||
# Copyright year update 2022
|
# Copyright year update 2022
|
||||||
4a9744e916d24122a81092f6b7950054048ba860
|
4a9744e916d24122a81092f6b7950054048ba860
|
||||||
# Copyright year update 2023
|
# Copyright year update 2023
|
||||||
|
8
.github/PULL_REQUEST_TEMPLATE.md
vendored
8
.github/PULL_REQUEST_TEMPLATE.md
vendored
@ -8,14 +8,16 @@ Closes <!--- Insert Issue Number(s) this PR addresses. Start by typing # will op
|
|||||||
|
|
||||||
* [ ] Have you followed the guidelines in our [Contributing document](https://github.com/nasa/openmct/blob/master/CONTRIBUTING.md)?
|
* [ ] Have you followed the guidelines in our [Contributing document](https://github.com/nasa/openmct/blob/master/CONTRIBUTING.md)?
|
||||||
* [ ] Have you checked to ensure there aren't other open [Pull Requests](https://github.com/nasa/openmct/pulls) for the same update/change?
|
* [ ] Have you checked to ensure there aren't other open [Pull Requests](https://github.com/nasa/openmct/pulls) for the same update/change?
|
||||||
* [ ] Is this change backwards compatible? For example, developers won't need to change how they are calling the API or how they've extended core plugins such as Tables or Plots.
|
* [ ] Is this a notable change that will require a special callout in the release notes [Notable Change](../docs/src/process/release.md) ? For example, will this break compatibility with existing APIs or projects which source these plugins?
|
||||||
|
|
||||||
### Author Checklist
|
### Author Checklist
|
||||||
|
|
||||||
* [ ] Changes address original issue?
|
* [ ] Changes address original issue?
|
||||||
* [ ] Tests included and/or updated with changes?
|
* [ ] Tests included and/or updated with changes?
|
||||||
* [ ] Command line build passes?
|
|
||||||
* [ ] Has this been smoke tested?
|
* [ ] Has this been smoke tested?
|
||||||
|
* [ ] Have you associated this PR with a `type:` label? Note: this is not necessarily the same as the original issue.
|
||||||
|
* [ ] Have you associated a milestone with this PR? Note: leave blank if unsure.
|
||||||
|
* [ ] Is this a breaking change to be called out in the release notes?
|
||||||
* [ ] Testing instructions included in associated issue OR is this a dependency/testcase change?
|
* [ ] Testing instructions included in associated issue OR is this a dependency/testcase change?
|
||||||
|
|
||||||
### Reviewer Checklist
|
### Reviewer Checklist
|
||||||
@ -25,5 +27,3 @@ Closes <!--- Insert Issue Number(s) this PR addresses. Start by typing # will op
|
|||||||
* [ ] Changes appear not to be breaking changes?
|
* [ ] Changes appear not to be breaking changes?
|
||||||
* [ ] Appropriate automated tests included?
|
* [ ] Appropriate automated tests included?
|
||||||
* [ ] Code style and in-line documentation are appropriate?
|
* [ ] Code style and in-line documentation are appropriate?
|
||||||
* [ ] Has associated issue been labelled unverified? (only applicable if this PR closes the issue)
|
|
||||||
* [ ] Has associated issue been labelled bug? (only applicable if this PR is for a bug fix)
|
|
||||||
|
4
.github/dependabot.yml
vendored
4
.github/dependabot.yml
vendored
@ -5,6 +5,7 @@ updates:
|
|||||||
schedule:
|
schedule:
|
||||||
interval: 'weekly'
|
interval: 'weekly'
|
||||||
open-pull-requests-limit: 10
|
open-pull-requests-limit: 10
|
||||||
|
rebase-strategy: 'disabled'
|
||||||
labels:
|
labels:
|
||||||
- 'pr:daveit'
|
- 'pr:daveit'
|
||||||
- 'pr:e2e'
|
- 'pr:e2e'
|
||||||
@ -28,10 +29,13 @@ updates:
|
|||||||
update-types: ['version-update:semver-patch']
|
update-types: ['version-update:semver-patch']
|
||||||
- dependency-name: '@types/lodash'
|
- dependency-name: '@types/lodash'
|
||||||
update-types: ['version-update:semver-patch']
|
update-types: ['version-update:semver-patch']
|
||||||
|
- dependency-name: 'marked'
|
||||||
|
update-types: ['version-update:semver-patch']
|
||||||
- package-ecosystem: 'github-actions'
|
- package-ecosystem: 'github-actions'
|
||||||
directory: '/'
|
directory: '/'
|
||||||
schedule:
|
schedule:
|
||||||
interval: 'daily'
|
interval: 'daily'
|
||||||
|
rebase-strategy: 'disabled'
|
||||||
labels:
|
labels:
|
||||||
- 'pr:daveit'
|
- 'pr:daveit'
|
||||||
- 'type:maintenance'
|
- 'type:maintenance'
|
||||||
|
5
.github/release.yml
vendored
5
.github/release.yml
vendored
@ -1,5 +1,8 @@
|
|||||||
changelog:
|
changelog:
|
||||||
categories:
|
categories:
|
||||||
|
- title: 💥 Notable Changes
|
||||||
|
labels:
|
||||||
|
- notable_change
|
||||||
- title: 🏕 Features
|
- title: 🏕 Features
|
||||||
labels:
|
labels:
|
||||||
- type:feature
|
- type:feature
|
||||||
@ -20,4 +23,4 @@ changelog:
|
|||||||
- dependencies
|
- dependencies
|
||||||
- title: 🐛 Bug Fixes
|
- title: 🐛 Bug Fixes
|
||||||
labels:
|
labels:
|
||||||
- '*'
|
- "*"
|
||||||
|
8
.github/workflows/codeql-analysis.yml
vendored
8
.github/workflows/codeql-analysis.yml
vendored
@ -27,18 +27,18 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v2
|
uses: github/codeql-action/init@v3
|
||||||
with:
|
with:
|
||||||
config-file: ./.github/codeql/codeql-config.yml
|
config-file: ./.github/codeql/codeql-config.yml
|
||||||
languages: javascript
|
languages: javascript
|
||||||
queries: security-and-quality
|
queries: security-and-quality
|
||||||
|
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v2
|
uses: github/codeql-action/autobuild@v3
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v2
|
uses: github/codeql-action/analyze@v3
|
||||||
|
7
.github/workflows/e2e-couchdb.yml
vendored
7
.github/workflows/e2e-couchdb.yml
vendored
@ -15,7 +15,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
timeout-minutes: 60
|
timeout-minutes: 60
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 'lts/hydrogen'
|
node-version: 'lts/hydrogen'
|
||||||
@ -37,7 +37,7 @@ jobs:
|
|||||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||||
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||||
|
|
||||||
- run: npx playwright@1.36.2 install
|
- run: npx playwright@1.39.0 install
|
||||||
|
|
||||||
- name: Start CouchDB Docker Container and Init with Setup Scripts
|
- name: Start CouchDB Docker Container and Init with Setup Scripts
|
||||||
run: |
|
run: |
|
||||||
@ -47,9 +47,8 @@ jobs:
|
|||||||
bash src/plugins/persistence/couch/setup-couchdb.sh
|
bash src/plugins/persistence/couch/setup-couchdb.sh
|
||||||
bash src/plugins/persistence/couch/replace-localstorage-with-couchdb-indexhtml.sh
|
bash src/plugins/persistence/couch/replace-localstorage-with-couchdb-indexhtml.sh
|
||||||
|
|
||||||
- name: Run CouchDB Tests and publish to deploysentinel
|
- name: Run CouchDB Tests
|
||||||
env:
|
env:
|
||||||
DEPLOYSENTINEL_API_KEY: ${{ secrets.DEPLOYSENTINEL_API_KEY }}
|
|
||||||
COMMIT_INFO_SHA: ${{github.event.pull_request.head.sha }}
|
COMMIT_INFO_SHA: ${{github.event.pull_request.head.sha }}
|
||||||
run: npm run test:e2e:couchdb
|
run: npm run test:e2e:couchdb
|
||||||
|
|
||||||
|
61
.github/workflows/e2e-flakefinder.yml
vendored
Normal file
61
.github/workflows/e2e-flakefinder.yml
vendored
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
name: 'pr:e2e:flakefinder'
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: master
|
||||||
|
workflow_dispatch:
|
||||||
|
pull_request:
|
||||||
|
types:
|
||||||
|
- labeled
|
||||||
|
- opened
|
||||||
|
schedule:
|
||||||
|
- cron: '0 0 * * *'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
e2e-flakefinder:
|
||||||
|
if: contains(github.event.pull_request.labels.*.name, 'pr:e2e:flakefinder') || github.event_name == 'schedule' || github.event_name == 'workflow_dispatch' || github.event.action == 'opened'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 120
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 'lts/hydrogen'
|
||||||
|
|
||||||
|
- name: Cache NPM dependencies
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.npm
|
||||||
|
key: ${{ runner.os }}-node-${{ hashFiles('**/package.json') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-node-
|
||||||
|
|
||||||
|
- run: npx playwright@1.39.0 install
|
||||||
|
- run: npm install --cache ~/.npm --no-audit --progress=false
|
||||||
|
|
||||||
|
- name: Run E2E Tests (Repeated 10 Times)
|
||||||
|
run: npm run test:e2e:stable -- --retries=0 --repeat-each=10 --max-failures=50
|
||||||
|
|
||||||
|
- name: Archive test results
|
||||||
|
if: success() || failure()
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
path: test-results
|
||||||
|
|
||||||
|
- name: Remove pr:e2e:flakefinder label (if present)
|
||||||
|
if: always()
|
||||||
|
uses: actions/github-script@v6
|
||||||
|
with:
|
||||||
|
script: |
|
||||||
|
const { owner, repo, number } = context.issue;
|
||||||
|
const labelToRemove = 'pr:e2e:flakefinder';
|
||||||
|
try {
|
||||||
|
await github.rest.issues.removeLabel({
|
||||||
|
owner,
|
||||||
|
repo,
|
||||||
|
issue_number: number,
|
||||||
|
name: labelToRemove
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
core.warning(`Failed to remove ' + labelToRemove + ' label: ${error.message}`);
|
||||||
|
}
|
58
.github/workflows/e2e-perf.yml
vendored
Normal file
58
.github/workflows/e2e-perf.yml
vendored
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
name: 'e2e-perf'
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches: master
|
||||||
|
workflow_dispatch:
|
||||||
|
pull_request:
|
||||||
|
types:
|
||||||
|
- labeled
|
||||||
|
- opened
|
||||||
|
schedule:
|
||||||
|
- cron: '0 0 * * *'
|
||||||
|
jobs:
|
||||||
|
e2e-full:
|
||||||
|
if: contains(github.event.pull_request.labels.*.name, 'pr:e2e:perf') || github.event_name == 'schedule' || github.event_name == 'workflow_dispatch'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
timeout-minutes: 60
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: 'lts/hydrogen'
|
||||||
|
|
||||||
|
- name: Cache NPM dependencies
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: ~/.npm
|
||||||
|
key: ${{ runner.os }}-node-${{ hashFiles('**/package.json') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-node-
|
||||||
|
|
||||||
|
- run: npx playwright@1.39.0 install
|
||||||
|
- run: npm install --cache ~/.npm --no-audit --progress=false
|
||||||
|
- run: npm run test:perf:localhost
|
||||||
|
- run: npm run test:perf:contract
|
||||||
|
- run: npm run test:perf:memory
|
||||||
|
- name: Archive test results
|
||||||
|
if: success() || failure()
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
path: test-results
|
||||||
|
|
||||||
|
- name: Remove pr:e2e:perf label (if present)
|
||||||
|
if: always()
|
||||||
|
uses: actions/github-script@v6
|
||||||
|
with:
|
||||||
|
script: |
|
||||||
|
const { owner, repo, number } = context.issue;
|
||||||
|
const labelToRemove = 'pr:e2e:perf';
|
||||||
|
try {
|
||||||
|
await github.rest.issues.removeLabel({
|
||||||
|
owner,
|
||||||
|
repo,
|
||||||
|
issue_number: number,
|
||||||
|
name: labelToRemove
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
core.warning(`Failed to remove ' + labelToRemove + ' label: ${error.message}`);
|
||||||
|
}
|
4
.github/workflows/e2e-pr.yml
vendored
4
.github/workflows/e2e-pr.yml
vendored
@ -20,7 +20,7 @@ jobs:
|
|||||||
- ubuntu-latest
|
- ubuntu-latest
|
||||||
- windows-latest
|
- windows-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 'lts/hydrogen'
|
node-version: 'lts/hydrogen'
|
||||||
@ -33,7 +33,7 @@ jobs:
|
|||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ runner.os }}-node-
|
${{ runner.os }}-node-
|
||||||
|
|
||||||
- run: npx playwright@1.36.2 install
|
- run: npx playwright@1.39.0 install
|
||||||
- run: npx playwright install chrome-beta
|
- run: npx playwright install chrome-beta
|
||||||
- run: npm install --cache ~/.npm --no-audit --progress=false
|
- run: npm install --cache ~/.npm --no-audit --progress=false
|
||||||
- run: npm run test:e2e:full -- --max-failures=40
|
- run: npm run test:e2e:full -- --max-failures=40
|
||||||
|
4
.github/workflows/npm-prerelease.yml
vendored
4
.github/workflows/npm-prerelease.yml
vendored
@ -11,7 +11,7 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: lts/hydrogen
|
node-version: lts/hydrogen
|
||||||
@ -26,7 +26,7 @@ jobs:
|
|||||||
needs: build
|
needs: build
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: lts/hydrogen
|
node-version: lts/hydrogen
|
||||||
|
4
.github/workflows/pr-platform.yml
vendored
4
.github/workflows/pr-platform.yml
vendored
@ -22,14 +22,14 @@ jobs:
|
|||||||
- macos-latest
|
- macos-latest
|
||||||
- windows-latest
|
- windows-latest
|
||||||
node_version:
|
node_version:
|
||||||
- lts/gallium
|
- lts/iron
|
||||||
- lts/hydrogen
|
- lts/hydrogen
|
||||||
architecture:
|
architecture:
|
||||||
- x64
|
- x64
|
||||||
|
|
||||||
name: Node ${{ matrix.node_version }} - ${{ matrix.architecture }} on ${{ matrix.os }}
|
name: Node ${{ matrix.node_version }} - ${{ matrix.architecture }} on ${{ matrix.os }}
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Setup node
|
- name: Setup node
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v4
|
||||||
|
22
.github/workflows/prcop.yml
vendored
22
.github/workflows/prcop.yml
vendored
@ -3,17 +3,17 @@ name: PRCop
|
|||||||
on:
|
on:
|
||||||
pull_request:
|
pull_request:
|
||||||
types:
|
types:
|
||||||
|
- labeled
|
||||||
|
- unlabeled
|
||||||
- opened
|
- opened
|
||||||
- reopened
|
- reopened
|
||||||
- edited
|
|
||||||
- synchronize
|
- synchronize
|
||||||
- ready_for_review
|
- edited
|
||||||
- review_requested
|
|
||||||
- review_request_removed
|
|
||||||
pull_request_review_comment:
|
pull_request_review_comment:
|
||||||
types:
|
types:
|
||||||
- created
|
- created
|
||||||
|
env:
|
||||||
|
LABELS: ${{ join( github.event.pull_request.labels.*.name, ' ' ) }}
|
||||||
jobs:
|
jobs:
|
||||||
prcop:
|
prcop:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
@ -24,3 +24,15 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
config-file: '.github/workflows/prcop-config.json'
|
config-file: '.github/workflows/prcop-config.json'
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
check-type-label:
|
||||||
|
name: Check type Label
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- if: contains( env.LABELS, 'type:' ) == false
|
||||||
|
run: exit 1
|
||||||
|
check-milestone:
|
||||||
|
name: Check Milestone
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- if: github.event.pull_request.milestone == null && contains( env.LABELS, 'no milestone' ) == false
|
||||||
|
run: exit 1
|
||||||
|
3
.gitignore
vendored
3
.gitignore
vendored
@ -15,6 +15,9 @@
|
|||||||
*.idea
|
*.idea
|
||||||
*.iml
|
*.iml
|
||||||
|
|
||||||
|
# VSCode
|
||||||
|
.vscode/settings.json
|
||||||
|
|
||||||
# Build output
|
# Build output
|
||||||
target
|
target
|
||||||
dist
|
dist
|
||||||
|
14
.vscode/extensions.json
vendored
Normal file
14
.vscode/extensions.json
vendored
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
// See https://go.microsoft.com/fwlink/?LinkId=827846 to learn about workspace recommendations.
|
||||||
|
// Extension identifier format: ${publisher}.${name}. Example: vscode.csharp
|
||||||
|
|
||||||
|
// List of extensions which should be recommended for users of this workspace.
|
||||||
|
"recommendations": [
|
||||||
|
"Vue.volar",
|
||||||
|
"Vue.vscode-typescript-vue-plugin",
|
||||||
|
"dbaeumer.vscode-eslint",
|
||||||
|
"rvest.vs-code-prettier-eslint"
|
||||||
|
],
|
||||||
|
// List of extensions recommended by VS Code that should not be recommended for users of this workspace.
|
||||||
|
"unwantedRecommendations": ["octref.vetur"]
|
||||||
|
}
|
@ -1,5 +1,3 @@
|
|||||||
/* global __dirname module */
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This is the OpenMCT common webpack file. It is imported by the other three webpack configurations:
|
This is the OpenMCT common webpack file. It is imported by the other three webpack configurations:
|
||||||
- webpack.prod.js - the production configuration for OpenMCT (default)
|
- webpack.prod.js - the production configuration for OpenMCT (default)
|
||||||
@ -8,27 +6,28 @@ This is the OpenMCT common webpack file. It is imported by the other three webpa
|
|||||||
There are separate npm scripts to use these configurations, though simply running `npm install`
|
There are separate npm scripts to use these configurations, though simply running `npm install`
|
||||||
will use the default production configuration.
|
will use the default production configuration.
|
||||||
*/
|
*/
|
||||||
const path = require('path');
|
import { execSync } from 'node:child_process';
|
||||||
const packageDefinition = require('../package.json');
|
import fs from 'node:fs';
|
||||||
const CopyWebpackPlugin = require('copy-webpack-plugin');
|
import path from 'node:path';
|
||||||
const webpack = require('webpack');
|
import { fileURLToPath } from 'node:url';
|
||||||
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
|
|
||||||
|
|
||||||
const { VueLoaderPlugin } = require('vue-loader');
|
import CopyWebpackPlugin from 'copy-webpack-plugin';
|
||||||
|
import MiniCssExtractPlugin from 'mini-css-extract-plugin';
|
||||||
|
import { VueLoaderPlugin } from 'vue-loader';
|
||||||
|
import webpack from 'webpack';
|
||||||
let gitRevision = 'error-retrieving-revision';
|
let gitRevision = 'error-retrieving-revision';
|
||||||
let gitBranch = 'error-retrieving-branch';
|
let gitBranch = 'error-retrieving-branch';
|
||||||
|
|
||||||
|
const packageDefinition = JSON.parse(fs.readFileSync(new URL('../package.json', import.meta.url)));
|
||||||
|
|
||||||
try {
|
try {
|
||||||
gitRevision = require('child_process').execSync('git rev-parse HEAD').toString().trim();
|
gitRevision = execSync('git rev-parse HEAD').toString().trim();
|
||||||
gitBranch = require('child_process')
|
gitBranch = execSync('git rev-parse --abbrev-ref HEAD').toString().trim();
|
||||||
.execSync('git rev-parse --abbrev-ref HEAD')
|
|
||||||
.toString()
|
|
||||||
.trim();
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.warn(err);
|
console.warn(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
const projectRootDir = path.resolve(__dirname, '..');
|
const projectRootDir = fileURLToPath(new URL('../', import.meta.url));
|
||||||
|
|
||||||
/** @type {import('webpack').Configuration} */
|
/** @type {import('webpack').Configuration} */
|
||||||
const config = {
|
const config = {
|
||||||
@ -56,6 +55,7 @@ const config = {
|
|||||||
filename: '[name].js',
|
filename: '[name].js',
|
||||||
path: path.resolve(projectRootDir, 'dist'),
|
path: path.resolve(projectRootDir, 'dist'),
|
||||||
library: 'openmct',
|
library: 'openmct',
|
||||||
|
libraryExport: 'default',
|
||||||
libraryTarget: 'umd',
|
libraryTarget: 'umd',
|
||||||
publicPath: '',
|
publicPath: '',
|
||||||
hashFunction: 'xxhash64',
|
hashFunction: 'xxhash64',
|
||||||
@ -65,19 +65,18 @@ const config = {
|
|||||||
alias: {
|
alias: {
|
||||||
'@': path.join(projectRootDir, 'src'),
|
'@': path.join(projectRootDir, 'src'),
|
||||||
legacyRegistry: path.join(projectRootDir, 'src/legacyRegistry'),
|
legacyRegistry: path.join(projectRootDir, 'src/legacyRegistry'),
|
||||||
saveAs: 'file-saver/src/FileSaver.js',
|
|
||||||
csv: 'comma-separated-values',
|
csv: 'comma-separated-values',
|
||||||
EventEmitter: 'eventemitter3',
|
EventEmitter: 'eventemitter3',
|
||||||
bourbon: 'bourbon.scss',
|
bourbon: 'bourbon.scss',
|
||||||
'plotly-basic': 'plotly.js-basic-dist',
|
'plotly-basic': 'plotly.js-basic-dist-min',
|
||||||
'plotly-gl2d': 'plotly.js-gl2d-dist',
|
'plotly-gl2d': 'plotly.js-gl2d-dist-min',
|
||||||
'd3-scale': path.join(projectRootDir, 'node_modules/d3-scale/dist/d3-scale.min.js'),
|
printj: 'printj/printj.mjs',
|
||||||
printj: path.join(projectRootDir, 'node_modules/printj/dist/printj.min.js'),
|
|
||||||
styles: path.join(projectRootDir, 'src/styles'),
|
styles: path.join(projectRootDir, 'src/styles'),
|
||||||
MCT: path.join(projectRootDir, 'src/MCT'),
|
MCT: path.join(projectRootDir, 'src/MCT'),
|
||||||
testUtils: path.join(projectRootDir, 'src/utils/testUtils.js'),
|
testUtils: path.join(projectRootDir, 'src/utils/testUtils.js'),
|
||||||
objectUtils: path.join(projectRootDir, 'src/api/objects/object-utils.js'),
|
objectUtils: path.join(projectRootDir, 'src/api/objects/object-utils.js'),
|
||||||
utils: path.join(projectRootDir, 'src/utils')
|
utils: path.join(projectRootDir, 'src/utils'),
|
||||||
|
vue: 'vue/dist/vue.esm-bundler'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
@ -183,4 +182,4 @@ const config = {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
||||||
|
@ -1,12 +1,10 @@
|
|||||||
/* global module */
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This file extends the webpack.dev.js config to add babel istanbul coverage.
|
This file extends the webpack.dev.js config to add babel istanbul coverage.
|
||||||
OpenMCT Continuous Integration servers use this configuration to add code coverage
|
OpenMCT Continuous Integration servers use this configuration to add code coverage
|
||||||
information to pull requests.
|
information to pull requests.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const config = require('./webpack.dev');
|
import config from './webpack.dev.js';
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
const CI = process.env.CI === 'true';
|
const CI = process.env.CI === 'true';
|
||||||
|
|
||||||
@ -34,4 +32,4 @@ config.module.rules.push({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
||||||
|
@ -1,18 +1,16 @@
|
|||||||
/* global __dirname module */
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This configuration should be used for development purposes. It contains full source map, a
|
This configuration should be used for development purposes. It contains full source map, a
|
||||||
devServer (which be invoked using by `npm start`), and a non-minified Vue.js distribution.
|
devServer (which be invoked using by `npm start`), and a non-minified Vue.js distribution.
|
||||||
If OpenMCT is to be used for a production server, use webpack.prod.js instead.
|
If OpenMCT is to be used for a production server, use webpack.prod.js instead.
|
||||||
*/
|
*/
|
||||||
const path = require('path');
|
import path from 'path';
|
||||||
const webpack = require('webpack');
|
import webpack from 'webpack';
|
||||||
const { merge } = require('webpack-merge');
|
import { merge } from 'webpack-merge';
|
||||||
|
import { fileURLToPath } from 'node:url';
|
||||||
|
|
||||||
const common = require('./webpack.common');
|
import common from './webpack.common.js';
|
||||||
const projectRootDir = path.resolve(__dirname, '..');
|
|
||||||
|
|
||||||
module.exports = merge(common, {
|
export default merge(common, {
|
||||||
mode: 'development',
|
mode: 'development',
|
||||||
watchOptions: {
|
watchOptions: {
|
||||||
// Since we use require.context, webpack is watching the entire directory.
|
// Since we use require.context, webpack is watching the entire directory.
|
||||||
@ -42,7 +40,7 @@ module.exports = merge(common, {
|
|||||||
},
|
},
|
||||||
watchFiles: ['**/*.css'],
|
watchFiles: ['**/*.css'],
|
||||||
static: {
|
static: {
|
||||||
directory: path.join(__dirname, '..', '/dist'),
|
directory: fileURLToPath(new URL('../dist', import.meta.url)),
|
||||||
publicPath: '/dist',
|
publicPath: '/dist',
|
||||||
watch: false
|
watch: false
|
||||||
}
|
}
|
||||||
|
@ -1,17 +1,14 @@
|
|||||||
/* global __dirname module */
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This configuration should be used for production installs.
|
This configuration should be used for production installs.
|
||||||
It is the default webpack configuration.
|
It is the default webpack configuration.
|
||||||
*/
|
*/
|
||||||
const path = require('path');
|
|
||||||
const webpack = require('webpack');
|
|
||||||
const { merge } = require('webpack-merge');
|
|
||||||
|
|
||||||
const common = require('./webpack.common');
|
import webpack from 'webpack';
|
||||||
const projectRootDir = path.resolve(__dirname, '..');
|
import { merge } from 'webpack-merge';
|
||||||
|
|
||||||
module.exports = merge(common, {
|
import common from './webpack.common.js';
|
||||||
|
|
||||||
|
export default merge(common, {
|
||||||
mode: 'production',
|
mode: 'production',
|
||||||
plugins: [
|
plugins: [
|
||||||
new webpack.DefinePlugin({
|
new webpack.DefinePlugin({
|
||||||
|
180
API.md
180
API.md
@ -94,6 +94,9 @@ well as assets such as html, css, and images necessary for the UI.
|
|||||||
|
|
||||||
## Starting an Open MCT application
|
## Starting an Open MCT application
|
||||||
|
|
||||||
|
> [!WARNING]
|
||||||
|
> Open MCT provides a development server via `webpack-dev-server` (`npm start`). **This should be used for development purposes only and should never be deployed to a production environment**.
|
||||||
|
|
||||||
To start a minimally functional Open MCT application, it is necessary to
|
To start a minimally functional Open MCT application, it is necessary to
|
||||||
include the Open MCT distributable, enable some basic plugins, and bootstrap
|
include the Open MCT distributable, enable some basic plugins, and bootstrap
|
||||||
the application. The tutorials walk through the process of getting Open MCT up
|
the application. The tutorials walk through the process of getting Open MCT up
|
||||||
@ -590,35 +593,108 @@ MinMax queries are issued by plots, and may be issued by other types as well. T
|
|||||||
#### Telemetry Formats
|
#### Telemetry Formats
|
||||||
|
|
||||||
Telemetry format objects define how to interpret and display telemetry data.
|
Telemetry format objects define how to interpret and display telemetry data.
|
||||||
They have a simple structure:
|
They have a simple structure, provided here as a TypeScript interface:
|
||||||
|
|
||||||
- `key`: A `string` that uniquely identifies this formatter.
|
```ts
|
||||||
- `format`: A `function` that takes a raw telemetry value, and returns a
|
interface Formatter {
|
||||||
human-readable `string` representation of that value. It has one required
|
key: string; // A string that uniquely identifies this formatter.
|
||||||
argument, and three optional arguments that provide context and can be used
|
|
||||||
for returning scaled representations of a value. An example of this is
|
format: (
|
||||||
representing time values in a scale such as the time conductor scale. There
|
value: any, // The raw telemetry value in its native type.
|
||||||
are multiple ways of representing a point in time, and by providing a minimum
|
minValue?: number, // An optional argument specifying the minimum displayed value.
|
||||||
scale value, maximum scale value, and a count, it's possible to provide more
|
maxValue?: number, // An optional argument specifying the maximum displayed value.
|
||||||
useful representations of time given the provided limitations.
|
count?: number // An optional argument specifying the number of displayed values.
|
||||||
- `value`: The raw telemetry value in its native type.
|
) => string; // Returns a human-readable string representation of the provided value.
|
||||||
- `minValue`: An **optional** argument specifying the minimum displayed
|
|
||||||
value.
|
parse: (
|
||||||
- `maxValue`: An **optional** argument specifying the maximum displayed
|
value: string | any // A string representation of a telemetry value or an already-parsed value.
|
||||||
value.
|
) => any; // Returns the value in its native type. This function should be idempotent.
|
||||||
- `count`: An **optional** argument specifying the number of displayed
|
|
||||||
values.
|
validate: (value: string) => boolean; // Takes a string representation of a telemetry value and returns a boolean indicating whether the provided string can be parsed.
|
||||||
- `parse`: A `function` that takes a `string` representation of a telemetry
|
}
|
||||||
value, and returns the value in its native type. **Note** parse might receive an already-parsed value. This function should be idempotent.
|
```
|
||||||
- `validate`: A `function` that takes a `string` representation of a telemetry
|
|
||||||
value, and returns a `boolean` value indicating whether the provided string
|
##### Built-in Formats
|
||||||
can be parsed.
|
|
||||||
|
Open MCT on its own defines a handful of built-in formats:
|
||||||
|
|
||||||
|
###### **Number Format (default):**
|
||||||
|
|
||||||
|
Applied to data with `format: 'number'`
|
||||||
|
```js
|
||||||
|
valueMetadata = {
|
||||||
|
format: 'number'
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
```ts
|
||||||
|
interface NumberFormatter extends Formatter {
|
||||||
|
parse: (x: any) => number;
|
||||||
|
format: (x: number) => string;
|
||||||
|
validate: (value: any) => boolean;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
###### **String Format**:
|
||||||
|
|
||||||
|
Applied to data with `format: 'string'`
|
||||||
|
```js
|
||||||
|
valueMetadata = {
|
||||||
|
format: 'string'
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
```
|
||||||
|
```ts
|
||||||
|
interface StringFormatter extends Formatter {
|
||||||
|
parse: (value: any) => string;
|
||||||
|
format: (value: string) => string;
|
||||||
|
validate: (value: any) => boolean;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
###### **Enum Format**:
|
||||||
|
Applied to data with `format: 'enum'`
|
||||||
|
```js
|
||||||
|
valueMetadata = {
|
||||||
|
format: 'enum',
|
||||||
|
enumerations: [
|
||||||
|
{
|
||||||
|
value: 1,
|
||||||
|
string: 'APPLE'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 2,
|
||||||
|
string: 'PEAR',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: 3,
|
||||||
|
string: 'ORANGE'
|
||||||
|
}]
|
||||||
|
// ...
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
Creates a two-way mapping between enum string and value to be used in the `parse` and `format` methods.
|
||||||
|
Ex:
|
||||||
|
- `formatter.parse('APPLE') === 1;`
|
||||||
|
- `formatter.format(1) === 'APPLE';`
|
||||||
|
|
||||||
|
```ts
|
||||||
|
interface EnumFormatter extends Formatter {
|
||||||
|
parse: (value: string) => string;
|
||||||
|
format: (value: number) => string;
|
||||||
|
validate: (value: any) => boolean;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
##### Registering Formats
|
##### Registering Formats
|
||||||
|
|
||||||
|
Formats implement the following interface (provided here as TypeScript for simplicity):
|
||||||
|
|
||||||
|
|
||||||
Formats are registered with the Telemetry API using the `addFormat` function. eg.
|
Formats are registered with the Telemetry API using the `addFormat` function. eg.
|
||||||
|
|
||||||
``` javascript
|
```javascript
|
||||||
openmct.telemetry.addFormat({
|
openmct.telemetry.addFormat({
|
||||||
key: 'number-to-string',
|
key: 'number-to-string',
|
||||||
format: function (number) {
|
format: function (number) {
|
||||||
@ -1228,3 +1304,61 @@ View provider Example:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Visibility-Based Rendering in View Providers
|
||||||
|
|
||||||
|
To enhance performance and resource efficiency in OpenMCT, a visibility-based rendering feature has been added. This feature is designed to defer the execution of rendering logic for views that are not currently visible. It ensures that views are only updated when they are in the viewport, similar to how modern browsers handle rendering of inactive tabs but optimized for the OpenMCT tabbed display. It also works when views are scrolled outside the viewport (e.g., in a Display Layout).
|
||||||
|
|
||||||
|
### Overview
|
||||||
|
|
||||||
|
The show function is responsible for the rendering of a view. An [Intersection Observer](https://developer.mozilla.org/en-US/docs/Web/API/Intersection_Observer_API) is used internally to determine whether the view is visible. This observer drives the visibility-based rendering feature, accessed via the `renderWhenVisible` function provided in the `viewOptions` parameter.
|
||||||
|
|
||||||
|
### Implementing Visibility-Based Rendering
|
||||||
|
|
||||||
|
The `renderWhenVisible` function is passed to the show function as part of the `viewOptions` object. This function can be used for all rendering logic that would otherwise be executed within a `requestAnimationFrame` call. When called, `renderWhenVisible` will either execute the provided function immediately (via `requestAnimationFrame`) if the view is currently visible, or defer its execution until the view becomes visible.
|
||||||
|
|
||||||
|
Additionally, `renderWhenVisible` returns a boolean value indicating whether the provided function was executed immediately (`true`) or deferred (`false`).
|
||||||
|
|
||||||
|
Monitoring of visibility begins after the first call to `renderWhenVisible` is made.
|
||||||
|
|
||||||
|
Here’s the signature for the show function:
|
||||||
|
|
||||||
|
`show(element, isEditing, viewOptions)`
|
||||||
|
|
||||||
|
* `element` (HTMLElement) - The DOM element where the view should be rendered.
|
||||||
|
* `isEditing` (boolean) - Indicates whether the view is in editing mode.
|
||||||
|
* `viewOptions` (Object) - An object with configuration options for the view, including:
|
||||||
|
* `renderWhenVisible` (Function) - This function wraps the `requestAnimationFrame` and only triggers the provided render logic when the view is visible in the viewport.
|
||||||
|
|
||||||
|
### Example
|
||||||
|
|
||||||
|
An OpenMCT view provider might implement the show function as follows:
|
||||||
|
|
||||||
|
```js
|
||||||
|
// Define your view provider
|
||||||
|
const myViewProvider = {
|
||||||
|
// ... other properties and methods ...
|
||||||
|
show: function (element, isEditing, viewOptions) {
|
||||||
|
// Callback for rendering view content
|
||||||
|
const renderCallback = () => {
|
||||||
|
// Your view rendering logic goes here
|
||||||
|
};
|
||||||
|
|
||||||
|
// Use the renderWhenVisible function to ensure rendering only happens when view is visible
|
||||||
|
const wasRenderedImmediately = viewOptions.renderWhenVisible(renderCallback);
|
||||||
|
|
||||||
|
// Optionally handle the immediate rendering return value
|
||||||
|
if (wasRenderedImmediately) {
|
||||||
|
console.debug('🪞 Rendering triggered immediately as the view is visible.');
|
||||||
|
} else {
|
||||||
|
console.debug('🛑 Rendering has been deferred until the view becomes visible.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
Note that `renderWhenVisible` defers rendering while the view is not visible and caters to the latest execution call. This provides responsiveness for dynamic content while ensuring performance optimizations.
|
||||||
|
|
||||||
|
Ensure your view logic is prepared to handle potentially multiple deferrals if using this API, as only the last call to renderWhenVisible will be queued for execution upon the view becoming visible.
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
# Open MCT License
|
# Open MCT License
|
||||||
|
|
||||||
Open MCT, Copyright (c) 2014-2023, United States Government as represented by the Administrator of the National Aeronautics and Space Administration. All rights reserved.
|
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.
|
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.
|
||||||
|
|
||||||
|
18
README.md
18
README.md
@ -1,4 +1,4 @@
|
|||||||
# Open MCT [![license](https://img.shields.io/badge/license-Apache%202.0-blue.svg)](http://www.apache.org/licenses/LICENSE-2.0) [![codecov](https://codecov.io/gh/nasa/openmct/branch/master/graph/badge.svg?token=7DQIipp3ej)](https://codecov.io/gh/nasa/openmct) [![This project is using Percy.io for visual regression testing.](https://percy.io/static/images/percy-badge.svg)](https://percy.io/b2e34b17/openmct) [![npm version](https://img.shields.io/npm/v/openmct.svg)](https://www.npmjs.com/package/openmct)
|
# Open MCT [![license](https://img.shields.io/badge/license-Apache%202.0-blue.svg)](http://www.apache.org/licenses/LICENSE-2.0) [![codecov](https://codecov.io/gh/nasa/openmct/branch/master/graph/badge.svg?token=7DQIipp3ej)](https://codecov.io/gh/nasa/openmct) [![This project is using Percy.io for visual regression testing.](https://percy.io/static/images/percy-badge.svg)](https://percy.io/b2e34b17/openmct) [![npm version](https://img.shields.io/npm/v/openmct.svg)](https://www.npmjs.com/package/openmct) ![CodeQL](https://github.com/nasa/openmct/workflows/CodeQL/badge.svg)
|
||||||
|
|
||||||
Open MCT (Open Mission Control Technologies) is a next-generation mission control framework for visualization of data on desktop and mobile devices. It is developed at NASA's Ames Research Center, and is being used by NASA for data analysis of spacecraft missions, as well as planning and operation of experimental rover systems. As a generalizable and open source framework, Open MCT could be used as the basis for building applications for planning, operation, and analysis of any systems producing telemetry data.
|
Open MCT (Open Mission Control Technologies) is a next-generation mission control framework for visualization of data on desktop and mobile devices. It is developed at NASA's Ames Research Center, and is being used by NASA for data analysis of spacecraft missions, as well as planning and operation of experimental rover systems. As a generalizable and open source framework, Open MCT could be used as the basis for building applications for planning, operation, and analysis of any systems producing telemetry data.
|
||||||
|
|
||||||
@ -127,6 +127,8 @@ Each test suite generates a report in CircleCI. For a complete overview of testi
|
|||||||
|
|
||||||
Our code coverage is generated during the runtime of our unit, e2e, and visual tests. The combination of those reports is published to [codecov.io](https://app.codecov.io/gh/nasa/openmct/)
|
Our code coverage is generated during the runtime of our unit, e2e, and visual tests. The combination of those reports is published to [codecov.io](https://app.codecov.io/gh/nasa/openmct/)
|
||||||
|
|
||||||
|
For more on the specifics of our code coverage setup, [see](TESTING.md#code-coverage)
|
||||||
|
|
||||||
# Glossary
|
# Glossary
|
||||||
|
|
||||||
Certain terms are used throughout Open MCT with consistent meanings
|
Certain terms are used throughout Open MCT with consistent meanings
|
||||||
@ -182,3 +184,17 @@ You might still be using legacy API if your source code
|
|||||||
|
|
||||||
### What should I do if I am using legacy API?
|
### What should I do if I am using legacy API?
|
||||||
Please refer to [the modern Open MCT API](https://nasa.github.io/openmct/documentation/). Post any questions to the [Discussions section](https://github.com/nasa/openmct/discussions) of the Open MCT GitHub repository.
|
Please refer to [the modern Open MCT API](https://nasa.github.io/openmct/documentation/). Post any questions to the [Discussions section](https://github.com/nasa/openmct/discussions) of the Open MCT GitHub repository.
|
||||||
|
|
||||||
|
## Related Repos
|
||||||
|
|
||||||
|
> [!NOTE]
|
||||||
|
> Although Open MCT functions as a standalone project, it is primarily an extensible framework intended to be used as a dependency with users' own plugins and packaging. Furthermore, Open MCT is intended to be used with an HTTP server such as Apache or Nginx. A great example of hosting Open MCT with Apache is `openmct-quickstart` and can be found in the table below.
|
||||||
|
|
||||||
|
| Repository | Description |
|
||||||
|
| --- | --- |
|
||||||
|
| [openmct-tutorial](https://github.com/nasa/openmct-tutorial) | A great place for beginners to learn how to use and extend Open MCT. |
|
||||||
|
| [openmct-quickstart](https://github.com/scottbell/openmct-quickstart) | A working example of Open MCT integrated with Apache HTTP server, YAMCS telemetry, and Couch DB for persistence.
|
||||||
|
| [Open MCT YAMCS Plugin](https://github.com/akhenry/openmct-yamcs) | Plugin for integrating YAMCS telemetry and command server with Open MCT. |
|
||||||
|
| [openmct-performance](https://github.com/unlikelyzero/openmct-performance) | Resources for performance testing Open MCT. |
|
||||||
|
| [openmct-as-a-dependency](https://github.com/unlikelyzero/openmct-as-a-dependency) | An advanced guide for users on how to build, develop, and test Open MCT when it's used as a dependency. |
|
||||||
|
|
||||||
|
82
TESTING.md
82
TESTING.md
@ -37,14 +37,84 @@ Documentation located [here](./e2e/README.md)
|
|||||||
|
|
||||||
## Code Coverage
|
## Code Coverage
|
||||||
|
|
||||||
* 100% statement coverage is achievable and desirable.
|
It's up to the individual developer as to whether they want to add line coverage in the form of a unit test or e2e test.
|
||||||
|
|
||||||
Codecov.io will combine each of the above commands with [Codecov.io Flags](https://docs.codecov.com/docs/flags). Effectively, this allows us to combine multiple reports which are run at various stages of our CI Pipeline or run as part of a parallel process.
|
Line Code Coverage is generated by our unit tests and e2e tests, then combined by ([Codecov.io Flags](https://docs.codecov.com/docs/flags)), and finally reported in GitHub PRs by Codecov.io's PR Bot. This workflow gives a comprehensive (if flawed) view of line coverage.
|
||||||
|
|
||||||
|
### Karma-istanbul
|
||||||
|
|
||||||
|
Line coverage is generated by our `karma-coverage-istanbul-reporter` package as defined in our `karma.conf.js` file:
|
||||||
|
|
||||||
|
```js
|
||||||
|
coverageIstanbulReporter: {
|
||||||
|
fixWebpackSourcePaths: true,
|
||||||
|
skipFilesWithNoCoverage: true,
|
||||||
|
dir: 'coverage/unit', //Sets coverage file to be consumed by codecov.io
|
||||||
|
reports: ['lcovonly']
|
||||||
|
},
|
||||||
|
```
|
||||||
|
|
||||||
|
Once the file is generated, it can be published to codecov with
|
||||||
|
|
||||||
|
```json
|
||||||
|
"cov:unit:publish": "codecov --disable=gcov -f ./coverage/unit/lcov.info -F unit",
|
||||||
|
```
|
||||||
|
|
||||||
|
### e2e
|
||||||
|
The e2e line coverage is a bit more complex than the karma implementation. This is the general sequence of events:
|
||||||
|
|
||||||
|
1. Each e2e suite will start webpack with the ```npm run start:coverage``` command with config `webpack.coverage.js` and the `babel-plugin-istanbul` plugin to generate code coverage during e2e test execution using our custom [baseFixture](./baseFixtures.js).
|
||||||
|
1. During testcase execution, each e2e shard will generate its piece of the larger coverage suite. **This coverage file is not merged**. The raw coverage file is stored in a `.nyc_report` directory.
|
||||||
|
1. [nyc](https://github.com/istanbuljs/nyc) converts this directory into a `lcov` file with the following command `npm run cov:e2e:report`
|
||||||
|
1. Most of the tests are run in the '@stable' configuration and focus on chrome/ubuntu at a single resolution. This coverage is published to codecov with `npm run cov:e2e:stable:publish`.
|
||||||
|
1. The rest of our coverage only appears when run against `@unstable` tests, persistent datastore (couchdb), non-ubuntu machines, and non-chrome browsers with the `npm run cov:e2e:full:publish` flag. Since this happens about once a day, we have leveraged codecov.io's carryforward flag to report on lines covered outside of each commit on an individual PR.
|
||||||
|
|
||||||
This e2e coverage is combined with our unit test report to give a comprehensive (if flawed) view of line coverage.
|
|
||||||
|
|
||||||
### Limitations in our code coverage reporting
|
### Limitations in our code coverage reporting
|
||||||
|
Our code coverage implementation has some known limitations:
|
||||||
|
- [Variability](https://github.com/nasa/openmct/issues/5811)
|
||||||
|
- [Accuracy](https://github.com/nasa/openmct/issues/7015)
|
||||||
|
- [Vue instrumentation gaps](https://github.com/nasa/openmct/issues/4973)
|
||||||
|
|
||||||
Our code coverage implementation has two known limitations:
|
## Troubleshooting CI
|
||||||
- [Variability and accuracy](https://github.com/nasa/openmct/issues/5811)
|
The following is an evolving guide to troubleshoot CI and PR issues.
|
||||||
- [Vue instrumentation](https://github.com/nasa/openmct/issues/4973)
|
|
||||||
|
### Github Checks failing
|
||||||
|
There are a few reasons that your GitHub PR could be failing beyond simple failed tests.
|
||||||
|
* Required Checks. We're leveraging required checks in GitHub so that we can quickly and precisely control what becomes and informational failure vs a hard requirement. The only way to determine the difference between a required vs information check is check for the `(Required)` emblem next to the step details in GitHub Checks.
|
||||||
|
* Not all required checks are run per commit. You may need to manually trigger addition GitHub checks with a `pr:<label>` label added to your PR.
|
||||||
|
|
||||||
|
### Flaky tests
|
||||||
|
|
||||||
|
(CircleCI's test insights feature)[https://circleci.com/blog/introducing-test-insights-with-flaky-test-detection/] collects historical data about the individual test results for both unit and e2e tests. Note: only a 14 day window of flake is available.
|
||||||
|
|
||||||
|
### Local=Pass and CI=Fail
|
||||||
|
Although rare, it is possible that your test can pass locally but fail in CI.
|
||||||
|
|
||||||
|
#### Busting Cache
|
||||||
|
In certain circumstances, the CircleCI cache can become stale. In order to bust the cache, we've implemented a runtime boolean parameter in Circle CI creatively name BUST_CACHE. To execute:
|
||||||
|
1. Navigate to the branch in Circle CI believed to have stale cache.
|
||||||
|
1. Click on the 'Trigger Pipeline' button.
|
||||||
|
1. Add Parameter -> Parameter Type = boolean , Name = BUST_CACHE ,Value = true
|
||||||
|
1. Click 'Trigger Pipeline'
|
||||||
|
|
||||||
|
#### Run tests in the same container as CI
|
||||||
|
|
||||||
|
In extreme cases, tests can fail due to the constraints of running within a container. To execute tests in exactly the same way as run in CircleCI.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
// Replace {X.X.X} with the current Playwright version
|
||||||
|
// from our package.json or circleCI configuration file
|
||||||
|
docker run --rm --network host --cpus="2" -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:v{X.X.X}-focal /bin/bash
|
||||||
|
npm install
|
||||||
|
```
|
||||||
|
|
||||||
|
At this point, you're running inside the same container and with 2 cpu cores. You can specify the unit tests:
|
||||||
|
```sh
|
||||||
|
npm run test
|
||||||
|
```
|
||||||
|
or e2e tests:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npx playwright test --config=e2e/playwright-ci.config.js --project=chrome --grep <the testcase name>
|
||||||
|
```
|
@ -1,7 +1,7 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
#*****************************************************************************
|
#*****************************************************************************
|
||||||
#* Open MCT, Copyright (c) 2014-2023, United States Government
|
#* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
#* as represented by the Administrator of the National Aeronautics and Space
|
#* as represented by the Administrator of the National Aeronautics and Space
|
||||||
#* Administration. All rights reserved.
|
#* Administration. All rights reserved.
|
||||||
#*
|
#*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<!--
|
<!--
|
||||||
Open MCT, Copyright (c) 2014-2023, United States Government
|
Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
as represented by the Administrator of the National Aeronautics and Space
|
as represented by the Administrator of the National Aeronautics and Space
|
||||||
Administration. All rights reserved.
|
Administration. All rights reserved.
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
30
docs/src/process/release.md
Normal file
30
docs/src/process/release.md
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
|
||||||
|
# Release of NASA Open MCT NPM Package
|
||||||
|
|
||||||
|
This document outlines the process and key considerations for releasing a new version of the NASA Open MCT project as an NPM (Node Package Manager) package.
|
||||||
|
|
||||||
|
## 1. Pre-requisites
|
||||||
|
|
||||||
|
Before releasing a new version of the NASA Open MCT NPM package, ensure all dependencies are updated, and comprehensive tests are performed. This ensures compatibility and performance of the Open MCT within the Node.js ecosystem.
|
||||||
|
|
||||||
|
## 2. Versioning
|
||||||
|
|
||||||
|
Versioning is a critical step for package release. The Open MCT team follows [Semantic Versioning (SemVer)](https://semver.org) that consists of three major components: MAJOR.MINOR.PATCH. These ensure a structured process for updating, bug fixes, backward compatibility, and software progress.
|
||||||
|
|
||||||
|
## 3. Changelog Maintenance
|
||||||
|
|
||||||
|
A comprehensive changelog file, `CHANGELOG.md`, documents any changes, adding a high level of transparencies for anyone desiring to look into the status of new and past progress. It includes the summation of any major new enhancements, changes, bug fixes, and the credits to the users responsible for each unique progress.
|
||||||
|
|
||||||
|
## 4. Notable Changes Labels on GitHub PRs
|
||||||
|
|
||||||
|
For the Open MCT package, we leverage GitHub's Pull Request (PR) mechanisms extensively, with three important PR labels dedicated to signifying 'notable_changes':
|
||||||
|
|
||||||
|
- **Breaking Change** Highlights the integration of changes that are suspected to break, or without a doubt will break, backward compatibility. These should signal to users the upgrade might be seamless only if dependency and integration factors are properly managed, if not, one should expect to manage atypical technical snags.
|
||||||
|
- **API change** Signifies when a contribution makes any complete or under layer changes to the communication or its supporting access processes. This label flags required see-through insight on how the web-based control panel sees and manipulates any value and or network logs.
|
||||||
|
- **Default Behavior Change:** In the incident an update either adjusts a form to or integrates a not previously kept setting or plugin. i.e. autoscale is enabled by default when working with plots.
|
||||||
|
|
||||||
|
## 6. Community & Contributions
|
||||||
|
|
||||||
|
A flat community and the rounded center are kept in continuous celebration, with the given station open for two open-specifying dialogues, research, and all-for development probing. State the ownership for a handed looped, a welcome for even structure-core and architectural draft and impend.
|
||||||
|
|
||||||
|
Thank you for your collaboration and commitment to moving the project onto a text big club.
|
@ -22,3 +22,12 @@ snapshot:
|
|||||||
.c-ne__embed__time{
|
.c-ne__embed__time{
|
||||||
opacity: 0 !important;
|
opacity: 0 !important;
|
||||||
}
|
}
|
||||||
|
/* Time Conductor Start Time */
|
||||||
|
.c-compact-tc__setting-value{
|
||||||
|
opacity: 0 !important;
|
||||||
|
}
|
||||||
|
/* Chart Area for Plots */
|
||||||
|
.gl-plot-chart-area{
|
||||||
|
opacity: 0 !important;
|
||||||
|
}
|
||||||
|
|
@ -22,3 +22,11 @@ snapshot:
|
|||||||
.c-ne__embed__time{
|
.c-ne__embed__time{
|
||||||
opacity: 0 !important;
|
opacity: 0 !important;
|
||||||
}
|
}
|
||||||
|
/* Time Conductor Start Time */
|
||||||
|
.c-compact-tc__setting-value{
|
||||||
|
opacity: 0 !important;
|
||||||
|
}
|
||||||
|
/* Chart Area for Plots */
|
||||||
|
.gl-plot-chart-area{
|
||||||
|
opacity: 0 !important;
|
||||||
|
}
|
107
e2e/README.md
107
e2e/README.md
@ -51,11 +51,13 @@ Next, you should walk through our implementation of Playwright in Open MCT:
|
|||||||
|
|
||||||
## Types of e2e Testing
|
## Types of e2e Testing
|
||||||
|
|
||||||
e2e testing describes the layer at which a test is performed without prescribing the assertions which are made. Generally, when writing an e2e test, we have three choices to make on an assertion strategy:
|
e2e testing describes the layer at which a test is performed without prescribing the assertions which are made. Generally, when writing an e2e test, we have five choices to make on an assertion strategy:
|
||||||
|
|
||||||
1. Functional - Verifies the functional correctness of the application. Sometimes interchanged with e2e or regression testing.
|
1. Functional - Verifies the functional correctness of the application. Sometimes interchanged with e2e or regression testing.
|
||||||
2. Visual - Verifies the "look and feel" of the application and can only detect _undesirable changes when compared to a previous baseline_.
|
2. Visual - Verifies the "look and feel" of the application and can only detect _undesirable changes when compared to a previous baseline_.
|
||||||
3. Snapshot - Similar to Visual in that it captures the "look" of the application and can only detect _undesirable changes when compared to a previous baseline_. **Generally not preferred due to advanced setup necessary.**
|
3. Snapshot - Similar to Visual in that it captures the "look" of the application and can only detect _undesirable changes when compared to a previous baseline_. **Generally not preferred due to advanced setup necessary.**
|
||||||
|
4. Accessibility - Verifies that the application meets the accessibility standards defined by the [WCAG organization](https://www.w3.org/WAI/standards-guidelines/wcag/).
|
||||||
|
5. Performance - Verifies that application provides a performant experience. Like Snapshot testing, these tests are generally not recommended due to their difficulty in providing a consistent result.
|
||||||
|
|
||||||
When choosing between the different testing strategies, think only about the assertion that is made at the end of the series of test steps. "I want to verify that the Timer plugin functions correctly" vs "I want to verify that the Timer plugin does not look different than originally designed".
|
When choosing between the different testing strategies, think only about the assertion that is made at the end of the series of test steps. "I want to verify that the Timer plugin functions correctly" vs "I want to verify that the Timer plugin does not look different than originally designed".
|
||||||
|
|
||||||
@ -107,7 +109,7 @@ For those interested in the mechanics of snapshot testing with Playwright, you c
|
|||||||
// from our package.json or circleCI configuration file
|
// from our package.json or circleCI configuration file
|
||||||
docker run --rm --network host -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:v{X.X.X}-focal /bin/bash
|
docker run --rm --network host -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:v{X.X.X}-focal /bin/bash
|
||||||
npm install
|
npm install
|
||||||
npx playwright test --config=e2e/playwright-ci.config.js --project=chrome --grep @snapshot
|
npm run test:e2e:checksnapshots
|
||||||
```
|
```
|
||||||
|
|
||||||
### Updating Snapshots
|
### Updating Snapshots
|
||||||
@ -132,6 +134,41 @@ npm install
|
|||||||
npm run test:e2e:updatesnapshots
|
npm run test:e2e:updatesnapshots
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Once that's done, you'll need to run the following to verify that the changes do not cause more problems:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm run test:e2e:checksnapshots
|
||||||
|
```
|
||||||
|
|
||||||
|
## Automated Accessibility (a11y) Testing
|
||||||
|
|
||||||
|
Open MCT incorporates accessibility testing through two primary methods to ensure its compliance with accessibility standards:
|
||||||
|
|
||||||
|
1. **Usage of Playwright's Locator Strategy**: Open MCT utilizes Playwright's locator strategy, specifically the [page.getByRole('') function](https://playwright.dev/docs/api/class-framelocator#frame-locator-get-by-role), to ensure that web elements are accessible via assistive technologies. This approach focuses on the accessibility of elements rather than full adherence to a11y guidelines, which is covered in the second method.
|
||||||
|
|
||||||
|
2. **Enforcing a11y Guidelines with Playwright Axe Plugin**: To rigorously enforce a11y guideline compliance, Open MCT employs the [playwright axe plugin](https://playwright.dev/docs/accessibility-testing). This is achieved through the `scanForA11yViolations` function within the visual testing suite. This method not only benefits from the existing coverage of the visual tests but also targets specific a11y issues, such as `color-contrast` violations, which are particularly pertinent in the context of visual testing.
|
||||||
|
|
||||||
|
### a11y Standards (WCAG and Section 508)
|
||||||
|
|
||||||
|
Playwright axe supports a wide range of [WCAG Standards](https://playwright.dev/docs/accessibility-testing#scanning-for-wcag-violations) to test against. Open MCT is testing against the [Section 508](https://www.section508.gov/test/testing-overview/) accessibility guidelines with the intent to support higher standards over time. As of 2024, Section508 requirements now map completely to WCAG 2.0 AA. In the future, Section 508 requirements may map to WCAG 2.1 AA.
|
||||||
|
|
||||||
|
### Reading an a11y test failure
|
||||||
|
|
||||||
|
When an a11y test fails, the result must be interpreted in the html test report or the a11y report json artifact stored in the `/test-results/` folder. The json structure should be parsed for `"violations"` by `"id"` and identified `"target"`. Example provided for the 'color-contrast-enhanced' violation.
|
||||||
|
|
||||||
|
```json
|
||||||
|
"violations":
|
||||||
|
{
|
||||||
|
"id": "color-contrast-enhanced",
|
||||||
|
"impact": "serious",
|
||||||
|
"html": "<span class=\"label c-indicator__label\">0 Snapshots <button aria-label=\"Show Snapshots\">Show</button></span>",
|
||||||
|
"target": [
|
||||||
|
".s-status-off > .label.c-indicator__label"
|
||||||
|
],
|
||||||
|
"failureSummary": "Fix any of the following:\n Element has insufficient color contrast of 6.51 (foreground color: #aaaaaa, background color: #262626, font size: 8.1pt (10.8px), font weight: normal). Expected contrast ratio of 7:1"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
## Performance Testing
|
## Performance Testing
|
||||||
|
|
||||||
The open source performance tests function in three ways which match their naming and folder structure:
|
The open source performance tests function in three ways which match their naming and folder structure:
|
||||||
@ -142,6 +179,8 @@ The open source performance tests function in three ways which match their namin
|
|||||||
|
|
||||||
These tests are expected to become blocking and gating with assertions as we extend the capabilities of Playwright.
|
These tests are expected to become blocking and gating with assertions as we extend the capabilities of Playwright.
|
||||||
|
|
||||||
|
In addition to the explicit definition of performance tests, we also ensure that our test timeout timing is "tight" to catch performance regressions detectable by action timeouts. i.e. [Notebooks load much slower than they used to #6459](https://github.com/nasa/openmct/issues/6459)
|
||||||
|
|
||||||
## Test Architecture and CI
|
## Test Architecture and CI
|
||||||
|
|
||||||
### Architecture
|
### Architecture
|
||||||
@ -161,8 +200,8 @@ Our file structure follows the type of type of testing being excercised at the e
|
|||||||
|`./tests/performance/` | Performance tests which should be run on every commit.|
|
|`./tests/performance/` | Performance tests which should be run on every commit.|
|
||||||
|`./tests/performance/contract/` | A subset of performance tests which are designed to provide a contract between the open source tests which are run on every commit and the downstream tests which are run post merge and with other frameworks.|
|
|`./tests/performance/contract/` | A subset of performance tests which are designed to provide a contract between the open source tests which are run on every commit and the downstream tests which are run post merge and with other frameworks.|
|
||||||
|`./tests/performance/memory` | A subset of performance tests which are designed to test for memory leaks.|
|
|`./tests/performance/memory` | A subset of performance tests which are designed to test for memory leaks.|
|
||||||
|`./tests/visual/` | Visual tests.|
|
|`./tests/visual-a11y/` | Visual tests and accessibility tests.|
|
||||||
|`./tests/visual/component/` | Visual tests which are only run against a single component.|
|
|`./tests/visual-a11y/component/` | Visual and accessibility tests which are only run against a single component.|
|
||||||
|`./appActions.js` | Contains common methods which can be leveraged by test case authors to quickly move through the application when writing new tests.|
|
|`./appActions.js` | Contains common methods which can be leveraged by test case authors to quickly move through the application when writing new tests.|
|
||||||
|`./baseFixture.js` | Contains base fixtures which only extend default `@playwright/test` functionality. The expectation is that these fixtures will be removed as the native Playwright API improves|
|
|`./baseFixture.js` | Contains base fixtures which only extend default `@playwright/test` functionality. The expectation is that these fixtures will be removed as the native Playwright API improves|
|
||||||
|
|
||||||
@ -180,7 +219,7 @@ Open MCT is leveraging the [config file](https://playwright.dev/docs/test-config
|
|||||||
|`./playwright-local.config.js` | Used when running locally|
|
|`./playwright-local.config.js` | Used when running locally|
|
||||||
|`./playwright-performance.config.js` | Used when running performance tests in CI or locally|
|
|`./playwright-performance.config.js` | Used when running performance tests in CI or locally|
|
||||||
|`./playwright-performance-devmode.config.js` | Used when running performance tests in CI or locally|
|
|`./playwright-performance-devmode.config.js` | Used when running performance tests in CI or locally|
|
||||||
|`./playwright-visual.config.js` | Used to run the visual tests in CI or locally|
|
|`./playwright-visual-a11y.config.js` | Used to run the visual and a11y tests in CI or locally|
|
||||||
|
|
||||||
#### Test Tags
|
#### Test Tags
|
||||||
|
|
||||||
@ -191,9 +230,10 @@ Current list of test tags:
|
|||||||
|Test Tag|Description|
|
|Test Tag|Description|
|
||||||
|:-:|-|
|
|:-:|-|
|
||||||
|`@ipad` | Test case or test suite is compatible with Playwright's iPad support and Open MCT's read-only mobile view (i.e. no create button).|
|
|`@ipad` | Test case or test suite is compatible with Playwright's iPad support and Open MCT's read-only mobile view (i.e. no create button).|
|
||||||
|
|`@a11y` | Test case or test suite to execute playwright-axe accessibility checks and generate a11y reports.|
|
||||||
|`@gds` | Denotes a GDS Test Case used in the VIPER Mission.|
|
|`@gds` | Denotes a GDS Test Case used in the VIPER Mission.|
|
||||||
|`@addInit` | Initializes the browser with an injected and artificial state. Useful for loading non-default plugins. Likely will not work outside of `npm start`.|
|
|`@addInit` | Initializes the browser with an injected and artificial state. Useful for loading non-default plugins. Likely will not work outside of `npm start`.|
|
||||||
|`@localStorage` | Captures or generates session storage to manipulate browser state. Useful for excluding in tests which require a persistent backend (i.e. CouchDB).|
|
|`@localStorage` | Captures or generates session storage to manipulate browser state. Useful for excluding in tests which require a persistent backend (i.e. CouchDB). See [note](#utilizing-localstorage)|
|
||||||
|`@snapshot` | Uses Playwright's snapshot functionality to record a copy of the DOM for direct comparison. Must be run inside of the playwright container.|
|
|`@snapshot` | Uses Playwright's snapshot functionality to record a copy of the DOM for direct comparison. Must be run inside of the playwright container.|
|
||||||
|`@unstable` | A new test or test which is known to be flaky.|
|
|`@unstable` | A new test or test which is known to be flaky.|
|
||||||
|`@2p` | Indicates that multiple users are involved, or multiple tabs/pages are used. Useful for testing multi-user interactivity.|
|
|`@2p` | Indicates that multiple users are involved, or multiple tabs/pages are used. Useful for testing multi-user interactivity.|
|
||||||
@ -216,7 +256,7 @@ CircleCI
|
|||||||
- Stable e2e tests against ubuntu and chrome
|
- Stable e2e tests against ubuntu and chrome
|
||||||
- Performance tests against ubuntu and chrome
|
- Performance tests against ubuntu and chrome
|
||||||
- e2e tests are linted
|
- e2e tests are linted
|
||||||
- Visual tests are run in a single resolution on the default `espresso` theme
|
- Visual and a11y tests are run in a single resolution on the default `espresso` theme
|
||||||
|
|
||||||
#### 2. Per-Merge Testing
|
#### 2. Per-Merge Testing
|
||||||
|
|
||||||
@ -232,7 +272,7 @@ Nightly Testing in Circle CI
|
|||||||
- Full e2e suite against ubuntu and chrome, firefox, and an MMOC resolution profile
|
- Full e2e suite against ubuntu and chrome, firefox, and an MMOC resolution profile
|
||||||
- Performance tests against ubuntu and chrome
|
- Performance tests against ubuntu and chrome
|
||||||
- CouchDB suite
|
- CouchDB suite
|
||||||
- Visual Tests are run in the full profile
|
- Visual and a11y Tests are run in the full profile
|
||||||
|
|
||||||
Github Actions / Workflow
|
Github Actions / Workflow
|
||||||
|
|
||||||
@ -352,6 +392,28 @@ By adhering to this principle, we can create tests that are both robust and refl
|
|||||||
1. Avoid repeated setup to test a single assertion. Write longer tests with multiple soft assertions.
|
1. Avoid repeated setup to test a single assertion. Write longer tests with multiple soft assertions.
|
||||||
This ensures that your changes will be picked up with large refactors.
|
This ensures that your changes will be picked up with large refactors.
|
||||||
|
|
||||||
|
##### Utilizing LocalStorage
|
||||||
|
1. In order to save test runtime in the case of tests that require a decent amount of initial setup (such as in the case of testing complex displays), you may use [Playwright's `storageState` feature](https://playwright.dev/docs/api/class-browsercontext#browser-context-storage-state) to generate and load localStorage states.
|
||||||
|
1. To generate a localStorage state to be used in a test:
|
||||||
|
- Add an e2e test to our generateLocalStorageData suite which sets the initial state (creating/configuring objects, etc.), saving it in the `test-data` folder:
|
||||||
|
```js
|
||||||
|
// Save localStorage for future test execution
|
||||||
|
await context.storageState({
|
||||||
|
path: path.join(__dirname, '../../../e2e/test-data/display_layout_with_child_layouts.json')
|
||||||
|
});
|
||||||
|
```
|
||||||
|
- Load the state from file at the beginning of the desired test suite (within the `test.describe()`). (NOTE: the storage state will be used for each test in the suite, so you may need to create a new suite):
|
||||||
|
```js
|
||||||
|
const LOCALSTORAGE_PATH = path.resolve(
|
||||||
|
__dirname,
|
||||||
|
'../../../../test-data/display_layout_with_child_layouts.json'
|
||||||
|
);
|
||||||
|
test.use({
|
||||||
|
storageState: path.resolve(__dirname, LOCALSTORAGE_PATH)
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
### How to write a great test
|
### How to write a great test
|
||||||
|
|
||||||
- Avoid using css locators to find elements to the page. Use modern web accessible locators like `getByRole`
|
- Avoid using css locators to find elements to the page. Use modern web accessible locators like `getByRole`
|
||||||
@ -383,7 +445,7 @@ By adhering to this principle, we can create tests that are both robust and refl
|
|||||||
5. **Hide the Tree and Inspector**: Generally, your test will not require comparisons involving the tree and inspector. These aspects are covered in component-specific tests (explained below). To exclude them from the comparison by default, navigate to the root of the main view with the tree and inspector hidden:
|
5. **Hide the Tree and Inspector**: Generally, your test will not require comparisons involving the tree and inspector. These aspects are covered in component-specific tests (explained below). To exclude them from the comparison by default, navigate to the root of the main view with the tree and inspector hidden:
|
||||||
- `await page.goto('./#/browse/mine?hideTree=true&hideInspector=true')`
|
- `await page.goto('./#/browse/mine?hideTree=true&hideInspector=true')`
|
||||||
|
|
||||||
6. **Component-Specific Tests**: If you wish to focus on a particular component, use the `/visual/component/` folder and limit the scope of the comparison to that component. For instance:
|
6. **Component-Specific Tests**: If you wish to focus on a particular component, use the `/visual-a11y/component/` folder and limit the scope of the comparison to that component. For instance:
|
||||||
```js
|
```js
|
||||||
await percySnapshot(page, `Tree Pane w/ single level expanded (theme: ${theme})`, {
|
await percySnapshot(page, `Tree Pane w/ single level expanded (theme: ${theme})`, {
|
||||||
scope: treePane
|
scope: treePane
|
||||||
@ -424,7 +486,7 @@ The following contains a list of tips and tricks which don't exactly fit into a
|
|||||||
It is possible to override the browser's clock in order to control time-based elements. Since this can cause unwanted behavior (i.e. Tree not rendering), only use this sparingly. To do this, use the `overrideClock` fixture as such:
|
It is possible to override the browser's clock in order to control time-based elements. Since this can cause unwanted behavior (i.e. Tree not rendering), only use this sparingly. To do this, use the `overrideClock` fixture as such:
|
||||||
|
|
||||||
```js
|
```js
|
||||||
const { test, expect } = require('../../pluginFixtures.js');
|
import { test, expect } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('foo test suite', () => {
|
test.describe('foo test suite', () => {
|
||||||
|
|
||||||
@ -468,15 +530,7 @@ Our e2e code coverage is captured and combined with our unit test coverage. For
|
|||||||
|
|
||||||
#### Generating e2e code coverage
|
#### Generating e2e code coverage
|
||||||
|
|
||||||
Code coverage is collected during test execution using our custom [baseFixture](./baseFixtures.js). The raw coverage files are stored in a `.nyc_report` directory to be converted into a lcov file with the following [nyc](https://github.com/istanbuljs/nyc) command:
|
Please read more about our code coverage [here](../TESTING.md#code-coverage)
|
||||||
|
|
||||||
```npm run cov:e2e:report```
|
|
||||||
|
|
||||||
At this point, the nyc linecov report can be published to [codecov.io](https://about.codecov.io/) with the following command:
|
|
||||||
|
|
||||||
```npm run cov:e2e:stable:publish``` for the stable suite running in ubuntu.
|
|
||||||
or
|
|
||||||
```npm run cov:e2e:full:publish``` for the full suite running against all available platforms.
|
|
||||||
|
|
||||||
## Other
|
## Other
|
||||||
|
|
||||||
@ -526,10 +580,19 @@ A single e2e test in Open MCT is extended to run:
|
|||||||
- How is Open MCT extending default Playwright functionality?
|
- How is Open MCT extending default Playwright functionality?
|
||||||
- What about Component Testing?
|
- What about Component Testing?
|
||||||
|
|
||||||
### Troubleshooting
|
### Writing Tests
|
||||||
|
|
||||||
|
Playwright provides 3 supported methods of debugging and authoring tests:
|
||||||
|
- A 'watch mode' for running tests locally and debugging on the fly
|
||||||
|
- A 'debug mode' for debugging tests and writing assertions against tests
|
||||||
|
- A 'VSCode plugin' for debugging tests within the VSCode IDE.
|
||||||
|
|
||||||
|
Generally, we encourage folks to use the watch mode and provide a script `npm run test:e2e:watch` which launches the launch mode ui and enables hot reloading on the dev server.
|
||||||
|
|
||||||
|
### e2e Troubleshooting
|
||||||
|
|
||||||
|
Please follow the general guide troubleshooting in [the general troubleshooting doc](../TESTING.md#troubleshooting-ci)
|
||||||
|
|
||||||
- Why is my test failing on CI and not locally?
|
|
||||||
- How can I view the failing tests on CI?
|
|
||||||
- Tests won't start because 'Error: <http://localhost:8080/># is already used...'
|
- Tests won't start because 'Error: <http://localhost:8080/># is already used...'
|
||||||
This error will appear when running the tests locally. Sometimes, the webserver is left in an orphaned state and needs to be cleaned up. To clear up the orphaned webserver, execute the following from your Terminal:
|
This error will appear when running the tests locally. Sometimes, the webserver is left in an orphaned state and needs to be cleaned up. To clear up the orphaned webserver, execute the following from your Terminal:
|
||||||
```lsof -n -i4TCP:8080 | awk '{print$2}' | tail -1 | xargs kill -9```
|
```lsof -n -i4TCP:8080 | awk '{print$2}' | tail -1 | xargs kill -9```
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -54,9 +54,9 @@
|
|||||||
* @property {import('../src/api/notifications/NotificationAPI').NotificationOptions} [notificationOptions] additional options
|
* @property {import('../src/api/notifications/NotificationAPI').NotificationOptions} [notificationOptions] additional options
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const Buffer = require('buffer').Buffer;
|
import { expect } from '@playwright/test';
|
||||||
const genUuid = require('uuid').v4;
|
import { Buffer } from 'buffer';
|
||||||
const { expect } = require('@playwright/test');
|
import { v4 as genUuid } from 'uuid';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This common function creates a domain object with the default options. It is the preferred way of creating objects
|
* This common function creates a domain object with the default options. It is the preferred way of creating objects
|
||||||
@ -81,7 +81,7 @@ async function createDomainObjectWithDefaults(
|
|||||||
await page.goto(`${parentUrl}`);
|
await page.goto(`${parentUrl}`);
|
||||||
|
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click the object specified by 'type'
|
// Click the object specified by 'type'
|
||||||
await page.click(`li[role='menuitem']:text("${type}")`);
|
await page.click(`li[role='menuitem']:text("${type}")`);
|
||||||
@ -108,7 +108,7 @@ async function createDomainObjectWithDefaults(
|
|||||||
// Click OK button and wait for Navigate event
|
// Click OK button and wait for Navigate event
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
page.waitForLoadState(),
|
page.waitForLoadState(),
|
||||||
page.click('[aria-label="Save"]'),
|
await page.getByRole('button', { name: 'Save' }).click(),
|
||||||
// Wait for Save Banner to appear
|
// Wait for Save Banner to appear
|
||||||
page.waitForSelector('.c-message-banner__message')
|
page.waitForSelector('.c-message-banner__message')
|
||||||
]);
|
]);
|
||||||
@ -120,8 +120,8 @@ async function createDomainObjectWithDefaults(
|
|||||||
|
|
||||||
if (await _isInEditMode(page, uuid)) {
|
if (await _isInEditMode(page, uuid)) {
|
||||||
// Save (exit edit mode)
|
// Save (exit edit mode)
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
await page.locator('li[title="Save and Finish Editing"]').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@ -182,7 +182,7 @@ async function createPlanFromJSON(page, { name, json, parent = 'mine' }) {
|
|||||||
await page.goto(`${parentUrl}`);
|
await page.goto(`${parentUrl}`);
|
||||||
|
|
||||||
// Click the Create button
|
// Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click 'Plan' menu option
|
// Click 'Plan' menu option
|
||||||
await page.click(`li:text("Plan")`);
|
await page.click(`li:text("Plan")`);
|
||||||
@ -231,7 +231,7 @@ async function createExampleTelemetryObject(page, parent = 'mine') {
|
|||||||
|
|
||||||
await page.goto(`${parentUrl}`);
|
await page.goto(`${parentUrl}`);
|
||||||
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
await page.locator('li:has-text("Sine Wave Generator")').click();
|
await page.locator('li:has-text("Sine Wave Generator")').click();
|
||||||
|
|
||||||
@ -284,7 +284,7 @@ async function navigateToObjectWithFixedTimeBounds(page, url, start, end) {
|
|||||||
*/
|
*/
|
||||||
async function openObjectTreeContextMenu(page, url) {
|
async function openObjectTreeContextMenu(page, url) {
|
||||||
await page.goto(url);
|
await page.goto(url);
|
||||||
await page.click('button[title="Show selected item in tree"]');
|
await page.getByLabel('Show selected item in tree').click();
|
||||||
await page.locator('.is-navigated-object').click({
|
await page.locator('.is-navigated-object').click({
|
||||||
button: 'right'
|
button: 'right'
|
||||||
});
|
});
|
||||||
@ -644,8 +644,7 @@ async function renameObjectFromContextMenu(page, url, newName) {
|
|||||||
await page.click('[aria-label="Save"]');
|
await page.click('[aria-label="Save"]');
|
||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-undef
|
export {
|
||||||
module.exports = {
|
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
createExampleTelemetryObject,
|
createExampleTelemetryObject,
|
||||||
createNotification,
|
createNotification,
|
||||||
@ -653,16 +652,16 @@ module.exports = {
|
|||||||
expandEntireTree,
|
expandEntireTree,
|
||||||
expandTreePaneItemByName,
|
expandTreePaneItemByName,
|
||||||
getCanvasPixels,
|
getCanvasPixels,
|
||||||
getHashUrlToDomainObject,
|
|
||||||
getFocusedObjectUuid,
|
getFocusedObjectUuid,
|
||||||
|
getHashUrlToDomainObject,
|
||||||
navigateToObjectWithFixedTimeBounds,
|
navigateToObjectWithFixedTimeBounds,
|
||||||
openObjectTreeContextMenu,
|
openObjectTreeContextMenu,
|
||||||
|
renameObjectFromContextMenu,
|
||||||
|
setEndOffset,
|
||||||
setFixedTimeMode,
|
setFixedTimeMode,
|
||||||
|
setIndependentTimeConductorBounds,
|
||||||
setRealTimeMode,
|
setRealTimeMode,
|
||||||
setStartOffset,
|
setStartOffset,
|
||||||
setEndOffset,
|
|
||||||
setTimeConductorBounds,
|
setTimeConductorBounds,
|
||||||
setIndependentTimeConductorBounds,
|
waitForPlotsToRender
|
||||||
waitForPlotsToRender,
|
|
||||||
renameObjectFromContextMenu
|
|
||||||
};
|
};
|
||||||
|
96
e2e/avpFixtures.js
Normal file
96
e2e/avpFixtures.js
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* avpFixtures.js
|
||||||
|
*
|
||||||
|
* @file This module provides custom fixtures specifically tailored for Accessibility, Visual, and Performance (AVP) tests.
|
||||||
|
* These fixtures extend the base functionality of the Playwright fixtures and appActions, and are designed to be
|
||||||
|
* generalized across all plugins. They offer functionalities like scanning for accessibility violations, integrating
|
||||||
|
* with axe-core, and more.
|
||||||
|
*
|
||||||
|
* IMPORTANT NOTE: This fixture file is not intended to be extended further by other fixtures. If you find yourself
|
||||||
|
* needing to do so, please consult the documentation and consider creating a specialized fixture or modifying the
|
||||||
|
* existing ones.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import AxeBuilder from '@axe-core/playwright';
|
||||||
|
import fs from 'fs';
|
||||||
|
import path from 'path';
|
||||||
|
|
||||||
|
import { expect, test } from './pluginFixtures.js';
|
||||||
|
|
||||||
|
// Constants for repeated values
|
||||||
|
const TEST_RESULTS_DIR = './test-results';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Scans for accessibility violations on a page and writes a report to disk if violations are found.
|
||||||
|
* Automatically asserts that no violations should be present.
|
||||||
|
*
|
||||||
|
* @typedef {object} GenerateReportOptions
|
||||||
|
* @property {string} [reportName] - The name for the report file.
|
||||||
|
*
|
||||||
|
* @param {import('playwright').Page} page - The page object from Playwright.
|
||||||
|
* @param {string} testCaseName - The name of the test case.
|
||||||
|
* @param {GenerateReportOptions} [options={}] - The options for the report generation.
|
||||||
|
*
|
||||||
|
* @returns {Promise<object|null>} Returns the accessibility scan results if violations are found,
|
||||||
|
* otherwise returns null.
|
||||||
|
*/
|
||||||
|
/* eslint-disable no-undef */
|
||||||
|
export async function scanForA11yViolations(page, testCaseName, options = {}) {
|
||||||
|
const builder = new AxeBuilder({ page });
|
||||||
|
builder.withTags(['wcag2aa']);
|
||||||
|
// https://github.com/dequelabs/axe-core/blob/develop/doc/rule-descriptions.md
|
||||||
|
const accessibilityScanResults = await builder.analyze();
|
||||||
|
|
||||||
|
// Assert that no violations should be present
|
||||||
|
expect(
|
||||||
|
accessibilityScanResults.violations,
|
||||||
|
`Accessibility violations found in test case: ${testCaseName}`
|
||||||
|
).toEqual([]);
|
||||||
|
|
||||||
|
// Check if there are any violations
|
||||||
|
if (accessibilityScanResults.violations.length > 0) {
|
||||||
|
let reportName = options.reportName || testCaseName;
|
||||||
|
let sanitizedReportName = reportName.replace(/\//g, '_');
|
||||||
|
const reportPath = path.join(TEST_RESULTS_DIR, `${sanitizedReportName}.json`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (!fs.existsSync(TEST_RESULTS_DIR)) {
|
||||||
|
fs.mkdirSync(TEST_RESULTS_DIR);
|
||||||
|
}
|
||||||
|
|
||||||
|
fs.writeFileSync(reportPath, JSON.stringify(accessibilityScanResults, null, 2));
|
||||||
|
console.log(`Accessibility report with violations saved successfully as ${reportPath}`);
|
||||||
|
return accessibilityScanResults;
|
||||||
|
} catch (err) {
|
||||||
|
console.error(`Error writing the accessibility report to file ${reportPath}:`, err);
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('No accessibility violations found, no report generated.');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { expect, test };
|
@ -1,6 +1,6 @@
|
|||||||
/* eslint-disable no-undef */
|
/* eslint-disable no-undef */
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -28,12 +28,12 @@
|
|||||||
* GitHub issues.
|
* GitHub issues.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const base = require('@playwright/test');
|
import { expect, request, test } from '@playwright/test';
|
||||||
const { expect, request } = base;
|
import fs from 'fs';
|
||||||
const fs = require('fs');
|
import path from 'path';
|
||||||
const path = require('path');
|
import sinon from 'sinon';
|
||||||
const { v4: uuid } = require('uuid');
|
import { fileURLToPath } from 'url';
|
||||||
const sinon = require('sinon');
|
import { v4 as uuid } from 'uuid';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Takes a `ConsoleMessage` and returns a formatted string. Used to enable console log error detection.
|
* Takes a `ConsoleMessage` and returns a formatted string. Used to enable console log error detection.
|
||||||
@ -68,7 +68,7 @@ function waitForAnimations(locator) {
|
|||||||
*/
|
*/
|
||||||
const istanbulCLIOutput = path.join(process.cwd(), '.nyc_output');
|
const istanbulCLIOutput = path.join(process.cwd(), '.nyc_output');
|
||||||
|
|
||||||
exports.test = base.test.extend({
|
const extendedTest = test.extend({
|
||||||
/**
|
/**
|
||||||
* This allows the test to manipulate the browser clock. This is useful for Visual and Snapshot tests which need
|
* This allows the test to manipulate the browser clock. This is useful for Visual and Snapshot tests which need
|
||||||
* the Time Indicator Clock to be in a specific state.
|
* the Time Indicator Clock to be in a specific state.
|
||||||
@ -97,7 +97,7 @@ exports.test = base.test.extend({
|
|||||||
async ({ context, clockOptions }, use) => {
|
async ({ context, clockOptions }, use) => {
|
||||||
if (clockOptions !== undefined) {
|
if (clockOptions !== undefined) {
|
||||||
await context.addInitScript({
|
await context.addInitScript({
|
||||||
path: path.join(__dirname, '../', './node_modules/sinon/pkg/sinon.js')
|
path: fileURLToPath(new URL('../node_modules/sinon/pkg/sinon.js', import.meta.url))
|
||||||
});
|
});
|
||||||
await context.addInitScript((options) => {
|
await context.addInitScript((options) => {
|
||||||
window.__clock = sinon.useFakeTimers(options);
|
window.__clock = sinon.useFakeTimers(options);
|
||||||
@ -201,6 +201,4 @@ exports.test = base.test.extend({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
exports.expect = expect;
|
export { expect, request, extendedTest as test, waitForAnimations };
|
||||||
exports.request = request;
|
|
||||||
exports.waitForAnimations = waitForAnimations;
|
|
||||||
|
@ -15,4 +15,5 @@ export const MISSION_TIME = 1732413600000; // Saturday, November 23, 2024 6:00:0
|
|||||||
* - hides the tree and inspector to prevent visual noise
|
* - hides the tree and inspector to prevent visual noise
|
||||||
* - sets the time bounds to a fixed range
|
* - sets the time bounds to a fixed range
|
||||||
*/
|
*/
|
||||||
export const VISUAL_URL = './#/browse/mine?tc.mode=fixed&tc.startBound=1693592063607&tc.endBound=1693593893607&tc.timeSystem=utc&view=grid&hideInspector=true&hideTree=true';
|
export const VISUAL_URL =
|
||||||
|
'./#/browse/mine?tc.mode=fixed&tc.startBound=1693592063607&tc.endBound=1693593893607&tc.timeSystem=utc&view=grid&hideInspector=true&hideTree=true';
|
||||||
|
30
e2e/helper/addInitDataVisualization.js
Normal file
30
e2e/helper/addInitDataVisualization.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
// This should be used to install the Example User
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
openmct.install(openmct.plugins.example.ExampleDataVisualizationSourcePlugin());
|
||||||
|
openmct.install(
|
||||||
|
openmct.plugins.InspectorDataVisualization({ type: 'exampleDataVisualizationSource' })
|
||||||
|
);
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,14 +19,15 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
/* global __dirname */
|
import { fileURLToPath } from 'url';
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
*/
|
*/
|
||||||
async function navigateToFaultManagementWithExample(page) {
|
async function navigateToFaultManagementWithExample(page) {
|
||||||
await page.addInitScript({ path: path.join(__dirname, './', 'addInitExampleFaultProvider.js') });
|
await page.addInitScript({
|
||||||
|
path: fileURLToPath(new URL('./addInitExampleFaultProvider.js', import.meta.url))
|
||||||
|
});
|
||||||
|
|
||||||
await navigateToFaultItemInTree(page);
|
await navigateToFaultItemInTree(page);
|
||||||
}
|
}
|
||||||
@ -36,7 +37,7 @@ async function navigateToFaultManagementWithExample(page) {
|
|||||||
*/
|
*/
|
||||||
async function navigateToFaultManagementWithStaticExample(page) {
|
async function navigateToFaultManagementWithStaticExample(page) {
|
||||||
await page.addInitScript({
|
await page.addInitScript({
|
||||||
path: path.join(__dirname, './', 'addInitExampleFaultProviderStatic.js')
|
path: fileURLToPath(new URL('./addInitExampleFaultProviderStatic.js', import.meta.url))
|
||||||
});
|
});
|
||||||
|
|
||||||
await navigateToFaultItemInTree(page);
|
await navigateToFaultItemInTree(page);
|
||||||
@ -46,7 +47,9 @@ async function navigateToFaultManagementWithStaticExample(page) {
|
|||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
*/
|
*/
|
||||||
async function navigateToFaultManagementWithoutExample(page) {
|
async function navigateToFaultManagementWithoutExample(page) {
|
||||||
await page.addInitScript({ path: path.join(__dirname, './', 'addInitFaultManagementPlugin.js') });
|
await page.addInitScript({
|
||||||
|
path: fileURLToPath(new URL('./addInitFaultManagementPlugin.js', import.meta.url))
|
||||||
|
});
|
||||||
|
|
||||||
await navigateToFaultItemInTree(page);
|
await navigateToFaultItemInTree(page);
|
||||||
}
|
}
|
||||||
@ -265,29 +268,28 @@ async function openFaultRowMenu(page, rowNumber) {
|
|||||||
.click();
|
.click();
|
||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-undef
|
export {
|
||||||
module.exports = {
|
|
||||||
navigateToFaultManagementWithExample,
|
|
||||||
navigateToFaultManagementWithStaticExample,
|
|
||||||
navigateToFaultManagementWithoutExample,
|
|
||||||
navigateToFaultItemInTree,
|
|
||||||
acknowledgeFault,
|
acknowledgeFault,
|
||||||
shelveMultipleFaults,
|
|
||||||
acknowledgeMultipleFaults,
|
acknowledgeMultipleFaults,
|
||||||
shelveFault,
|
|
||||||
changeViewTo,
|
changeViewTo,
|
||||||
sortFaultsBy,
|
|
||||||
enterSearchTerm,
|
|
||||||
clearSearch,
|
clearSearch,
|
||||||
selectFaultItem,
|
enterSearchTerm,
|
||||||
getHighestSeverity,
|
|
||||||
getLowestSeverity,
|
|
||||||
getFaultResultCount,
|
|
||||||
getFault,
|
getFault,
|
||||||
getFaultByName,
|
getFaultByName,
|
||||||
getFaultName,
|
getFaultName,
|
||||||
getFaultSeverity,
|
|
||||||
getFaultNamespace,
|
getFaultNamespace,
|
||||||
|
getFaultResultCount,
|
||||||
|
getFaultSeverity,
|
||||||
getFaultTriggerTime,
|
getFaultTriggerTime,
|
||||||
openFaultRowMenu
|
getHighestSeverity,
|
||||||
|
getLowestSeverity,
|
||||||
|
navigateToFaultItemInTree,
|
||||||
|
navigateToFaultManagementWithExample,
|
||||||
|
navigateToFaultManagementWithoutExample,
|
||||||
|
navigateToFaultManagementWithStaticExample,
|
||||||
|
openFaultRowMenu,
|
||||||
|
selectFaultItem,
|
||||||
|
shelveFault,
|
||||||
|
shelveMultipleFaults,
|
||||||
|
sortFaultsBy
|
||||||
};
|
};
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,11 +20,11 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { createDomainObjectWithDefaults } = require('../appActions');
|
import { createDomainObjectWithDefaults } from '../appActions.js';
|
||||||
|
|
||||||
const NOTEBOOK_DROP_AREA = '.c-notebook__drag-area';
|
const NOTEBOOK_DROP_AREA = '.c-notebook__drag-area';
|
||||||
const CUSTOM_NAME = 'CUSTOM_NAME';
|
const CUSTOM_NAME = 'CUSTOM_NAME';
|
||||||
const path = require('path');
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
@ -49,7 +49,7 @@ async function dragAndDropEmbed(page, notebookObject) {
|
|||||||
// Navigate to notebook
|
// Navigate to notebook
|
||||||
await page.goto(notebookObject.url);
|
await page.goto(notebookObject.url);
|
||||||
// Expand the tree to reveal the notebook
|
// Expand the tree to reveal the notebook
|
||||||
await page.click('button[title="Show selected item in tree"]');
|
await page.getByLabel('Show selected item in tree').click();
|
||||||
// Drag and drop the SWG into the notebook
|
// Drag and drop the SWG into the notebook
|
||||||
await page.dragAndDrop(`text=${swg.name}`, NOTEBOOK_DROP_AREA);
|
await page.dragAndDrop(`text=${swg.name}`, NOTEBOOK_DROP_AREA);
|
||||||
await commitEntry(page);
|
await commitEntry(page);
|
||||||
@ -69,7 +69,9 @@ async function commitEntry(page) {
|
|||||||
*/
|
*/
|
||||||
async function startAndAddRestrictedNotebookObject(page) {
|
async function startAndAddRestrictedNotebookObject(page) {
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
await page.addInitScript({ path: path.join(__dirname, 'addInitRestrictedNotebook.js') });
|
await page.addInitScript({
|
||||||
|
path: fileURLToPath(new URL('./addInitRestrictedNotebook.js', import.meta.url))
|
||||||
|
});
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
return createDomainObjectWithDefaults(page, {
|
return createDomainObjectWithDefaults(page, {
|
||||||
@ -138,12 +140,11 @@ async function createNotebookEntryAndTags(page, iterations = 1) {
|
|||||||
return notebook;
|
return notebook;
|
||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line no-undef
|
export {
|
||||||
module.exports = {
|
createNotebookAndEntry,
|
||||||
enterTextEntry,
|
|
||||||
dragAndDropEmbed,
|
|
||||||
startAndAddRestrictedNotebookObject,
|
|
||||||
lockPage,
|
|
||||||
createNotebookEntryAndTags,
|
createNotebookEntryAndTags,
|
||||||
createNotebookAndEntry
|
dragAndDropEmbed,
|
||||||
|
enterTextEntry,
|
||||||
|
lockPage,
|
||||||
|
startAndAddRestrictedNotebookObject
|
||||||
};
|
};
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,7 +20,7 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
import { expect } from '../pluginFixtures';
|
import { expect } from '../pluginFixtures.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Asserts that the number of activities in the plan view matches the number of
|
* Asserts that the number of activities in the plan view matches the number of
|
||||||
@ -81,6 +81,30 @@ function activitiesWithinTimeBounds(start1, end1, start2, end2) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Asserts that the swim lanes / groups in the plan view matches the order of
|
||||||
|
* groups in the plan data.
|
||||||
|
* @param {import('@playwright/test').Page} page the page
|
||||||
|
* @param {object} plan The raw plan json to assert against
|
||||||
|
* @param {string} objectUrl The URL of the object to assert against (plan or gantt chart)
|
||||||
|
*/
|
||||||
|
export async function assertPlanOrderedSwimLanes(page, plan, objectUrl) {
|
||||||
|
// Switch to the plan view
|
||||||
|
await page.goto(`${objectUrl}?view=plan.view`);
|
||||||
|
const planGroups = await page
|
||||||
|
.locator('.c-plan__contents > div > .c-swimlane__lane-label .c-object-label__name')
|
||||||
|
.all();
|
||||||
|
|
||||||
|
const groups = plan.Groups;
|
||||||
|
|
||||||
|
for (let i = 0; i < groups.length; i++) {
|
||||||
|
// Assert that the order of groups in the plan view matches the order of
|
||||||
|
// groups in the plan data
|
||||||
|
const groupName = await planGroups[i].innerText();
|
||||||
|
expect(groupName).toEqual(groups[i].name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Navigate to the plan view, switch to fixed time mode,
|
* Navigate to the plan view, switch to fixed time mode,
|
||||||
* and set the bounds to span all activities.
|
* and set the bounds to span all activities.
|
||||||
@ -89,17 +113,35 @@ function activitiesWithinTimeBounds(start1, end1, start2, end2) {
|
|||||||
* @param {string} planObjectUrl
|
* @param {string} planObjectUrl
|
||||||
*/
|
*/
|
||||||
export async function setBoundsToSpanAllActivities(page, planJson, planObjectUrl) {
|
export async function setBoundsToSpanAllActivities(page, planJson, planObjectUrl) {
|
||||||
const activities = Object.values(planJson).flat();
|
|
||||||
// Get the earliest start value
|
// Get the earliest start value
|
||||||
const start = Math.min(...activities.map((activity) => activity.start));
|
const start = getEarliestStartTime(planJson);
|
||||||
// Get the latest end value
|
// Get the latest end value
|
||||||
const end = Math.max(...activities.map((activity) => activity.end));
|
const end = getLatestEndTime(planJson);
|
||||||
// Set the start and end bounds to the earliest start and latest end
|
// Set the start and end bounds to the earliest start and latest end
|
||||||
await page.goto(
|
await page.goto(
|
||||||
`${planObjectUrl}?tc.mode=fixed&tc.startBound=${start}&tc.endBound=${end}&tc.timeSystem=utc&view=plan.view`
|
`${planObjectUrl}?tc.mode=fixed&tc.startBound=${start}&tc.endBound=${end}&tc.timeSystem=utc&view=plan.view`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {object} planJson
|
||||||
|
* @returns {number}
|
||||||
|
*/
|
||||||
|
export function getEarliestStartTime(planJson) {
|
||||||
|
const activities = Object.values(planJson).flat();
|
||||||
|
return Math.min(...activities.map((activity) => activity.start));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {object} planJson
|
||||||
|
* @returns {number}
|
||||||
|
*/
|
||||||
|
export function getLatestEndTime(planJson) {
|
||||||
|
const activities = Object.values(planJson).flat();
|
||||||
|
return Math.max(...activities.map((activity) => activity.end));
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Uses the Open MCT API to set the status of a plan to 'draft'.
|
* Uses the Open MCT API to set the status of a plan to 'draft'.
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
@ -110,3 +152,23 @@ export async function setDraftStatusForPlan(page, plan) {
|
|||||||
await window.openmct.status.set(planObject.uuid, 'draft');
|
await window.openmct.status.set(planObject.uuid, 'draft');
|
||||||
}, plan);
|
}, plan);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function addPlanGetInterceptor(page) {
|
||||||
|
await page.waitForLoadState('load');
|
||||||
|
await page.evaluate(async () => {
|
||||||
|
await window.openmct.objects.addGetInterceptor({
|
||||||
|
appliesTo: (identifier, domainObject) => {
|
||||||
|
return domainObject && domainObject.type === 'plan';
|
||||||
|
},
|
||||||
|
invoke: (identifier, object) => {
|
||||||
|
if (object) {
|
||||||
|
object.sourceMap = {
|
||||||
|
orderedGroups: 'Groups'
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return object;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
189
e2e/helper/plotTagsUtils.js
Normal file
189
e2e/helper/plotTagsUtils.js
Normal file
@ -0,0 +1,189 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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 { waitForPlotsToRender } from '../appActions.js';
|
||||||
|
import { expect } from '../pluginFixtures.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a canvas and a set of points, tags the points on the canvas.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {HTMLCanvasElement} canvas a telemetry item with a plot
|
||||||
|
* @param {Number} xEnd a telemetry item with a plot
|
||||||
|
* @param {Number} yEnd a telemetry item with a plot
|
||||||
|
* @returns {Promise}
|
||||||
|
*/
|
||||||
|
export async function createTags({ page, canvas, xEnd = 700, yEnd = 520 }) {
|
||||||
|
await canvas.hover({ trial: true });
|
||||||
|
|
||||||
|
//Alt+Shift Drag Start to select some points to tag
|
||||||
|
await page.keyboard.down('Alt');
|
||||||
|
await page.keyboard.down('Shift');
|
||||||
|
|
||||||
|
await canvas.dragTo(canvas, {
|
||||||
|
sourcePosition: {
|
||||||
|
x: 1,
|
||||||
|
y: 1
|
||||||
|
},
|
||||||
|
targetPosition: {
|
||||||
|
x: xEnd,
|
||||||
|
y: yEnd
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
//Alt Drag End
|
||||||
|
await page.keyboard.up('Alt');
|
||||||
|
await page.keyboard.up('Shift');
|
||||||
|
|
||||||
|
//Wait for canvas to stabilize.
|
||||||
|
await canvas.hover({ trial: true });
|
||||||
|
|
||||||
|
// add some tags
|
||||||
|
await page.getByText('Annotations').click();
|
||||||
|
await page.getByRole('button', { name: /Add Tag/ }).click();
|
||||||
|
await page.getByPlaceholder('Type to select tag').click();
|
||||||
|
await page.getByText('Driving').click();
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: /Add Tag/ }).click();
|
||||||
|
await page.getByPlaceholder('Type to select tag').click();
|
||||||
|
await page.getByText('Science').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a telemetry item (e.g., a Sine Wave Generator) with a plot, tests that the plot can be tagged.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {import('../../../../appActions').CreatedObjectInfo} telemetryItem a telemetry item with a plot
|
||||||
|
* @returns {Promise}
|
||||||
|
*/
|
||||||
|
export async function testTelemetryItem(page, telemetryItem) {
|
||||||
|
// Check that telemetry item also received the tag
|
||||||
|
await page.goto(telemetryItem.url);
|
||||||
|
|
||||||
|
await expect(page.getByText('No tags to display for this item')).toBeVisible();
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
//Wait for canvas to stabilize.
|
||||||
|
await waitForPlotsToRender(page);
|
||||||
|
|
||||||
|
await expect(canvas).toBeInViewport();
|
||||||
|
await canvas.hover({ trial: true });
|
||||||
|
|
||||||
|
// click on the tagged plot point
|
||||||
|
await canvas.click({
|
||||||
|
position: {
|
||||||
|
x: 100,
|
||||||
|
y: 100
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Driving')).toBeHidden();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given a page, tests that tags are searchable, deletable, and persist across reloads.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @returns {Promise}
|
||||||
|
*/
|
||||||
|
export async function basicTagsTests(page) {
|
||||||
|
// Search for Driving
|
||||||
|
await page.getByRole('searchbox', { name: 'Search Input' }).click();
|
||||||
|
|
||||||
|
// Clicking elsewhere should cause annotation selection to be cleared
|
||||||
|
await expect(page.getByText('No tags to display for this item')).toBeVisible();
|
||||||
|
//
|
||||||
|
await page.getByRole('searchbox', { name: 'Search Input' }).fill('driv');
|
||||||
|
|
||||||
|
// Always click on the first Sine Wave result
|
||||||
|
await page
|
||||||
|
.getByLabel('Search Result')
|
||||||
|
.getByText(/Sine Wave/)
|
||||||
|
.first()
|
||||||
|
.click();
|
||||||
|
|
||||||
|
// Delete Driving Tag
|
||||||
|
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
||||||
|
await page.locator('[aria-label="Remove tag Driving"]').click();
|
||||||
|
|
||||||
|
// Search for Science Tag
|
||||||
|
await page.getByRole('searchbox', { name: 'Search Input' }).click();
|
||||||
|
await page.getByRole('searchbox', { name: 'Search Input' }).fill('sc');
|
||||||
|
|
||||||
|
//Expect Science Tag to be present and and Driving Tags to be deleted
|
||||||
|
await expect(page.getByLabel('Search Result').first()).toContainText('Science');
|
||||||
|
await expect(page.getByLabel('Search Result').first()).not.toContainText('Driving');
|
||||||
|
|
||||||
|
// Search for Driving Tag and expect nothing found
|
||||||
|
await page.getByRole('searchbox', { name: 'Search Input' }).click();
|
||||||
|
await page.getByRole('searchbox', { name: 'Search Input' }).fill('driv');
|
||||||
|
await expect(page.getByText('No results found')).toBeVisible();
|
||||||
|
|
||||||
|
await page.reload({ waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
await waitForPlotsToRender(page);
|
||||||
|
|
||||||
|
//Navigate to the Inspector and check that all tags have been removed
|
||||||
|
await expect(page.getByRole('tab', { name: 'Annotations' })).not.toHaveClass(/is-current/);
|
||||||
|
await page.getByRole('tab', { name: 'Annotations' }).click();
|
||||||
|
await expect(page.getByRole('tab', { name: 'Annotations' })).toHaveClass(/is-current/);
|
||||||
|
await expect(page.getByText('No tags to display for this item')).toBeVisible();
|
||||||
|
|
||||||
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
// click on the tagged plot point
|
||||||
|
await canvas.click({
|
||||||
|
position: {
|
||||||
|
x: 100,
|
||||||
|
y: 100
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
//Expect Science to be visible but Driving to be hidden
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Driving')).toBeHidden();
|
||||||
|
|
||||||
|
//Click elsewhere
|
||||||
|
await page.locator('body').click();
|
||||||
|
//Click on tagged plot point again
|
||||||
|
await canvas.click({
|
||||||
|
position: {
|
||||||
|
x: 100,
|
||||||
|
y: 100
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Add Driving Tag again
|
||||||
|
await page.getByText('Annotations').click();
|
||||||
|
await page.getByRole('button', { name: /Add Tag/ }).click();
|
||||||
|
await page.getByPlaceholder('Type to select tag').click();
|
||||||
|
await page.getByText('Driving').click();
|
||||||
|
|
||||||
|
//Science and Driving Tags should be visible
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Driving')).toBeVisible();
|
||||||
|
|
||||||
|
// Delete Driving Tag again
|
||||||
|
await page.hover('[aria-label="Tag"]:has-text("Driving")');
|
||||||
|
await page.locator('[aria-label="Remove tag Driving"]').click();
|
||||||
|
|
||||||
|
//Science Tag should be visible and Driving Tag should be hidden
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Driving')).toBeHidden();
|
||||||
|
}
|
104
e2e/helper/stylingUtils.js
Normal file
104
e2e/helper/stylingUtils.js
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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 { expect } from '../pluginFixtures.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts a hex color value to its RGB equivalent.
|
||||||
|
*
|
||||||
|
* @param {string} hex - The hex color value. i.e. '#5b0f00'
|
||||||
|
* @returns {string} The RGB equivalent of the hex color.
|
||||||
|
*/
|
||||||
|
function hexToRGB(hex) {
|
||||||
|
const result = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(hex);
|
||||||
|
return result
|
||||||
|
? `rgb(${parseInt(result[1], 16)}, ${parseInt(result[2], 16)}, ${parseInt(result[3], 16)})`
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the background and text color of a given element.
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page - The Playwright page object.
|
||||||
|
* @param {string} borderColorHex - The hex value of the border color to set, or 'No Style'.
|
||||||
|
* @param {string} backgroundColorHex - The hex value of the background color to set, or 'No Style'.
|
||||||
|
* @param {string} textColorHex - The hex value of the text color to set, or 'No Style'.
|
||||||
|
* @param {import('@playwright/test').Locator} locator - The Playwright locator for the element whose style is to be set.
|
||||||
|
*/
|
||||||
|
async function setStyles(page, borderColorHex, backgroundColorHex, textColorHex, locator) {
|
||||||
|
await locator.click(); // Assuming the locator is clickable and opens the style setting UI
|
||||||
|
await page.getByLabel('Set border color').click();
|
||||||
|
await page.getByLabel(borderColorHex).click();
|
||||||
|
await page.getByLabel('Set background color').click();
|
||||||
|
await page.getByLabel(backgroundColorHex).click();
|
||||||
|
await page.getByLabel('Set text color').click();
|
||||||
|
await page.getByLabel(textColorHex).click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the styles of an element match the expected values.
|
||||||
|
*
|
||||||
|
* @param {string} expectedBorderColor - The expected border color in RGB format. Default is '#e6b8af' or 'rgb(230, 184, 175)'
|
||||||
|
* @param {string} expectedBackgroundColor - The expected background color in RGB format.
|
||||||
|
* @param {string} expectedTextColor - The expected text color in RGB format. Default is #aaaaaa or 'rgb(170, 170, 170)'
|
||||||
|
* @param {import('@playwright/test').Locator} locator - The Playwright locator for the element whose style is to be checked.
|
||||||
|
*/
|
||||||
|
async function checkStyles(
|
||||||
|
expectedBorderColor,
|
||||||
|
expectedBackgroundColor,
|
||||||
|
expectedTextColor,
|
||||||
|
locator
|
||||||
|
) {
|
||||||
|
const layoutStyles = await locator.evaluate((el) => {
|
||||||
|
return {
|
||||||
|
border: window.getComputedStyle(el).getPropertyValue('border-top-color'), //infer the left, right, and bottom
|
||||||
|
background: window.getComputedStyle(el).getPropertyValue('background-color'),
|
||||||
|
fontColor: window.getComputedStyle(el).getPropertyValue('color')
|
||||||
|
};
|
||||||
|
});
|
||||||
|
expect(layoutStyles.border).toContain(expectedBorderColor);
|
||||||
|
expect(layoutStyles.background).toContain(expectedBackgroundColor);
|
||||||
|
expect(layoutStyles.fontColor).toContain(expectedTextColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the font Styles of an element match the expected values.
|
||||||
|
*
|
||||||
|
* @param {string} expectedFontSize - The expected font size in '72px' format. Default is 'Default'
|
||||||
|
* @param {string} expectedFontWeight - The expected font Type. Format as '700' for bold. Default is 'Default'
|
||||||
|
* @param {string} expectedFontFamily - The expected font Type. Format as "\"Andale Mono\", sans-serif". Default is 'Default'
|
||||||
|
* @param {import('@playwright/test').Locator} locator - The Playwright locator for the element whose style is to be checked.
|
||||||
|
*/
|
||||||
|
async function checkFontStyles(expectedFontSize, expectedFontWeight, expectedFontFamily, locator) {
|
||||||
|
const layoutStyles = await locator.evaluate((el) => {
|
||||||
|
return {
|
||||||
|
fontSize: window.getComputedStyle(el).getPropertyValue('font-size'),
|
||||||
|
fontWeight: window.getComputedStyle(el).getPropertyValue('font-weight'),
|
||||||
|
fontFamily: window.getComputedStyle(el).getPropertyValue('font-family')
|
||||||
|
};
|
||||||
|
});
|
||||||
|
expect(layoutStyles.fontSize).toContain(expectedFontSize);
|
||||||
|
expect(layoutStyles.fontWeight).toContain(expectedFontWeight);
|
||||||
|
expect(layoutStyles.fontFamily).toContain(expectedFontFamily);
|
||||||
|
}
|
||||||
|
|
||||||
|
export { checkFontStyles, checkStyles, hexToRGB, setStyles };
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
/* eslint-disable no-undef */
|
|
||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
// eslint-disable-next-line no-unused-vars
|
// eslint-disable-next-line no-unused-vars
|
||||||
const { devices } = require('@playwright/test');
|
import { devices } from '@playwright/test';
|
||||||
const MAX_FAILURES = 5;
|
const MAX_FAILURES = 5;
|
||||||
const NUM_WORKERS = 2;
|
const NUM_WORKERS = 2;
|
||||||
|
|
||||||
@ -17,7 +16,7 @@ const config = {
|
|||||||
command: 'npm run start:coverage',
|
command: 'npm run start:coverage',
|
||||||
url: 'http://localhost:8080/#',
|
url: 'http://localhost:8080/#',
|
||||||
timeout: 200 * 1000,
|
timeout: 200 * 1000,
|
||||||
reuseExistingServer: false
|
reuseExistingServer: true //This was originally disabled to prevent differences in local debugging vs. CI. However, it significantly speeds up local debugging.
|
||||||
},
|
},
|
||||||
maxFailures: MAX_FAILURES, //Limits failures to 5 to reduce CI Waste
|
maxFailures: MAX_FAILURES, //Limits failures to 5 to reduce CI Waste
|
||||||
workers: NUM_WORKERS, //Limit to 2 for CircleCI Agent
|
workers: NUM_WORKERS, //Limit to 2 for CircleCI Agent
|
||||||
@ -76,9 +75,8 @@ const config = {
|
|||||||
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
['junit', { outputFile: '../test-results/results.xml' }],
|
['junit', { outputFile: '../test-results/results.xml' }]
|
||||||
['@deploysentinel/playwright']
|
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
/* eslint-disable no-undef */
|
|
||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
// eslint-disable-next-line no-unused-vars
|
// eslint-disable-next-line no-unused-vars
|
||||||
const { devices } = require('@playwright/test');
|
import { devices } from '@playwright/test';
|
||||||
|
|
||||||
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
||||||
const config = {
|
const config = {
|
||||||
@ -104,4 +103,4 @@ const config = {
|
|||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
/* eslint-disable no-undef */
|
|
||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
@ -40,4 +39,4 @@ const config = {
|
|||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
/* eslint-disable no-undef */
|
|
||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
@ -57,4 +56,4 @@ const config = {
|
|||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
/** @type {import('@playwright/test').PlaywrightTestConfig<{ theme: string }>} */
|
/** @type {import('@playwright/test').PlaywrightTestConfig<{ theme: string }>} */
|
||||||
const config = {
|
const config = {
|
||||||
retries: 0, // Visual tests should never retry due to snapshot comparison errors. Leaving as a shim
|
retries: 0, // Visual tests should never retry due to snapshot comparison errors. Leaving as a shim
|
||||||
testDir: 'tests/visual',
|
testDir: 'tests/visual-a11y',
|
||||||
testMatch: '**/*.visual.spec.js', // only run visual tests
|
testMatch: '**/*.visual.spec.js', // only run visual tests
|
||||||
timeout: 60 * 1000,
|
timeout: 60 * 1000,
|
||||||
workers: 1, //Lower stress on Circle CI Agent for Visual tests https://github.com/percy/cli/discussions/1067
|
workers: 1, //Lower stress on Circle CI Agent for Visual tests https://github.com/percy/cli/discussions/1067
|
||||||
@ -51,4 +51,4 @@ const config = {
|
|||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = config;
|
export default config;
|
46
e2e/playwright-watch.config.js
Normal file
46
e2e/playwright-watch.config.js
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
// playwright.config.js
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
||||||
|
const config = {
|
||||||
|
retries: 0, //Retries are not needed with watch mode
|
||||||
|
testDir: 'tests',
|
||||||
|
timeout: 60 * 1000,
|
||||||
|
webServer: {
|
||||||
|
command: 'npm run start', //Start in dev mode for hot reloading
|
||||||
|
url: 'http://localhost:8080/#',
|
||||||
|
timeout: 200 * 1000,
|
||||||
|
reuseExistingServer: true //This was originally disabled to prevent differences in local debugging vs. CI. However, it significantly speeds up local debugging.
|
||||||
|
},
|
||||||
|
workers: '75%', //Limit to 75% of the CPU to support running with dev server
|
||||||
|
use: {
|
||||||
|
baseURL: 'http://localhost:8080/',
|
||||||
|
headless: true,
|
||||||
|
ignoreHTTPSErrors: true,
|
||||||
|
screenshot: 'only-on-failure',
|
||||||
|
trace: 'on-first-retry',
|
||||||
|
video: 'off'
|
||||||
|
},
|
||||||
|
projects: [
|
||||||
|
{
|
||||||
|
name: 'chrome',
|
||||||
|
testMatch: '**/*.spec.js', // run all tests
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
reporter: [
|
||||||
|
['list'],
|
||||||
|
[
|
||||||
|
'html',
|
||||||
|
{
|
||||||
|
open: 'never',
|
||||||
|
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
||||||
|
}
|
||||||
|
],
|
||||||
|
['junit', { outputFile: '../test-results/results.xml' }]
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
export default config;
|
@ -1,6 +1,6 @@
|
|||||||
/* eslint-disable no-undef */
|
/* eslint-disable no-undef */
|
||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -26,9 +26,10 @@
|
|||||||
* and appActions. These fixtures should be generalized across all plugins.
|
* and appActions. These fixtures should be generalized across all plugins.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect, request } = require('./baseFixtures');
|
// import { createDomainObjectWithDefaults } from './appActions.js';
|
||||||
// const { createDomainObjectWithDefaults } = require('./appActions');
|
import { fileURLToPath } from 'url';
|
||||||
const path = require('path');
|
|
||||||
|
import { expect, request, test } from './baseFixtures.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @typedef {Object} ObjectCreateOptions
|
* @typedef {Object} ObjectCreateOptions
|
||||||
@ -117,7 +118,7 @@ const theme = 'espresso';
|
|||||||
*/
|
*/
|
||||||
const myItemsFolderName = 'My Items';
|
const myItemsFolderName = 'My Items';
|
||||||
|
|
||||||
exports.test = test.extend({
|
const extendedTest = test.extend({
|
||||||
// This should follow in the Project's configuration. Can be set to 'snow' in playwright config.js
|
// This should follow in the Project's configuration. Can be set to 'snow' in playwright config.js
|
||||||
theme: [theme, { option: true }],
|
theme: [theme, { option: true }],
|
||||||
// eslint-disable-next-line no-shadow
|
// eslint-disable-next-line no-shadow
|
||||||
@ -125,7 +126,9 @@ exports.test = test.extend({
|
|||||||
// eslint-disable-next-line playwright/no-conditional-in-test
|
// eslint-disable-next-line playwright/no-conditional-in-test
|
||||||
if (theme === 'snow') {
|
if (theme === 'snow') {
|
||||||
//inject snow theme
|
//inject snow theme
|
||||||
await page.addInitScript({ path: path.join(__dirname, './helper', './useSnowTheme.js') });
|
await page.addInitScript({
|
||||||
|
path: fileURLToPath(new URL('./helper/useSnowTheme.js', import.meta.url))
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Attach info about the currently running test and its project.
|
// Attach info about the currently running test and its project.
|
||||||
@ -142,19 +145,18 @@ exports.test = test.extend({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
exports.expect = expect;
|
export { expect, request, extendedTest as test };
|
||||||
exports.request = request;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Takes a readable stream and returns a string.
|
* Takes a readable stream and returns a string.
|
||||||
* @param {ReadableStream} readable - the readable stream
|
* @param {ReadableStream} readable - the readable stream
|
||||||
* @return {Promise<String>} the stringified stream
|
* @return {Promise<String>} the stringified stream
|
||||||
*/
|
*/
|
||||||
exports.streamToString = async function (readable) {
|
export async function streamToString(readable) {
|
||||||
let result = '';
|
let result = '';
|
||||||
for await (const chunk of readable) {
|
for await (const chunk of readable) {
|
||||||
result += chunk;
|
result += chunk;
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
};
|
}
|
||||||
|
22
e2e/test-data/display_layout_with_child_layouts.json
Normal file
22
e2e/test-data/display_layout_with_child_layouts.json
Normal file
File diff suppressed because one or more lines are too long
54
e2e/test-data/examplePlans/ExamplePlanWithOrderedLanes.json
Normal file
54
e2e/test-data/examplePlans/ExamplePlanWithOrderedLanes.json
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
{
|
||||||
|
"Groups": [
|
||||||
|
{
|
||||||
|
"name": "Group 1"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Group 2"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Group 2": [
|
||||||
|
{
|
||||||
|
"name": "Past event 3",
|
||||||
|
"start": 1660493208000,
|
||||||
|
"end": 1660503981000,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 4",
|
||||||
|
"start": 1660579608000,
|
||||||
|
"end": 1660624108000,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 5",
|
||||||
|
"start": 1660666008000,
|
||||||
|
"end": 1660681529000,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Group 1": [
|
||||||
|
{
|
||||||
|
"name": "Past event 1",
|
||||||
|
"start": 1660320408000,
|
||||||
|
"end": 1660343797000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 2",
|
||||||
|
"start": 1660406808000,
|
||||||
|
"end": 1660429160000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -6,7 +6,8 @@
|
|||||||
"end": 1660343797000,
|
"end": 1660343797000,
|
||||||
"type": "Group 1",
|
"type": "Group 1",
|
||||||
"color": "orange",
|
"color": "orange",
|
||||||
"textColor": "white"
|
"textColor": "white",
|
||||||
|
"id": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Past event 2",
|
"name": "Past event 2",
|
||||||
@ -14,7 +15,8 @@
|
|||||||
"end": 1660429160000,
|
"end": 1660429160000,
|
||||||
"type": "Group 1",
|
"type": "Group 1",
|
||||||
"color": "orange",
|
"color": "orange",
|
||||||
"textColor": "white"
|
"textColor": "white",
|
||||||
|
"id": 2
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Past event 3",
|
"name": "Past event 3",
|
||||||
@ -22,7 +24,8 @@
|
|||||||
"end": 1660503981000,
|
"end": 1660503981000,
|
||||||
"type": "Group 1",
|
"type": "Group 1",
|
||||||
"color": "orange",
|
"color": "orange",
|
||||||
"textColor": "white"
|
"textColor": "white",
|
||||||
|
"id": 3
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Past event 4",
|
"name": "Past event 4",
|
||||||
@ -30,7 +33,8 @@
|
|||||||
"end": 1660624108000,
|
"end": 1660624108000,
|
||||||
"type": "Group 1",
|
"type": "Group 1",
|
||||||
"color": "orange",
|
"color": "orange",
|
||||||
"textColor": "white"
|
"textColor": "white",
|
||||||
|
"id": 4
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Past event 5",
|
"name": "Past event 5",
|
||||||
@ -38,7 +42,8 @@
|
|||||||
"end": 1660681529000,
|
"end": 1660681529000,
|
||||||
"type": "Group 1",
|
"type": "Group 1",
|
||||||
"color": "orange",
|
"color": "orange",
|
||||||
"textColor": "white"
|
"textColor": "white",
|
||||||
|
"id": 5
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
42
e2e/test-data/examplePlans/ExamplePlan_Small3.json
Normal file
42
e2e/test-data/examplePlans/ExamplePlan_Small3.json
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"Group 1": [
|
||||||
|
{
|
||||||
|
"name": "Time until birthday",
|
||||||
|
"start": 1650320402000,
|
||||||
|
"end": 1660343797000,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white",
|
||||||
|
"id": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Time until supper",
|
||||||
|
"start": 1650320402000,
|
||||||
|
"end": 1650420410000,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "blue",
|
||||||
|
"textColor": "white",
|
||||||
|
"id": 2
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Group 2": [
|
||||||
|
{
|
||||||
|
"name": "Time since the last time I ate",
|
||||||
|
"start": 1650320102001,
|
||||||
|
"end": 1650320102001,
|
||||||
|
"type": "Group 2",
|
||||||
|
"color": "green",
|
||||||
|
"textColor": "white",
|
||||||
|
"id": 3
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Time since last accident",
|
||||||
|
"start": 1650320102002,
|
||||||
|
"end": 1650320102002,
|
||||||
|
"type": "Group 1",
|
||||||
|
"color": "yellow",
|
||||||
|
"textColor": "white",
|
||||||
|
"id": 4
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
22
e2e/test-data/flexible_layout_with_child_layouts.json
Normal file
22
e2e/test-data/flexible_layout_with_child_layouts.json
Normal file
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -6,11 +6,11 @@
|
|||||||
"localStorage": [
|
"localStorage": [
|
||||||
{
|
{
|
||||||
"name": "mct",
|
"name": "mct",
|
||||||
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602540,\"created\":1732413600760,\"persisted\":1732413602540},\"20e7d5fe-9cf8-4099-8957-9453a8954c67\":{\"identifier\":{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},\"name\":\"Overlay Plot with Telemetry Object\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"configuration\":{\"series\":[]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with Telemetry Object\\nchrome\",\"modified\":1732413603960,\"location\":\"mine\",\"created\":1732413601820,\"persisted\":1732413603960},\"2db521a9-996d-4d04-a171-93f4c5c220af\":{\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"identifier\":{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"},\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":0,\"infinityValues\":false,\"staleness\":false},\"modified\":1732413602540,\"location\":\"mine\",\"created\":1732413602540,\"persisted\":1732413602540}}"
|
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602460,\"created\":1732413600960,\"persisted\":1732413602460},\"e78ca721-fb5e-409b-bf6d-597c87cb716f\":{\"identifier\":{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},\"name\":\"Overlay Plot with Telemetry Object\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with Telemetry Object\\nchrome\",\"modified\":1732413603880,\"location\":\"mine\",\"created\":1732413601740,\"persisted\":1732413603880},\"c6100044-56be-44b3-acca-6b9fddfb3849\":{\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"identifier\":{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"},\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":0,\"infinityValues\":false,\"exceedFloat32\":false,\"staleness\":false},\"modified\":1732413602460,\"location\":\"mine\",\"created\":1732413602460,\"persisted\":1732413602460}}"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "mct-recent-objects",
|
"name": "mct-recent-objects",
|
||||||
"value": "[{\"objectPath\":[{\"identifier\":{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"},\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":0,\"infinityValues\":false,\"staleness\":false},\"modified\":1732413602540,\"location\":\"mine\",\"created\":1732413602540,\"persisted\":1732413602540},{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602540,\"created\":1732413600760,\"persisted\":1732413602540},{\"identifier\":{\"key\":\"ROOT\",\"namespace\":\"\"},\"name\":\"Open MCT\",\"type\":\"root\",\"composition\":[{\"key\":\"mine\",\"namespace\":\"\"}]}],\"navigationPath\":\"/browse/mine/2db521a9-996d-4d04-a171-93f4c5c220af\",\"domainObject\":{\"identifier\":{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"},\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":0,\"infinityValues\":false,\"staleness\":false},\"modified\":1732413602540,\"location\":\"mine\",\"created\":1732413602540,\"persisted\":1732413602540}},{\"objectPath\":[{\"identifier\":{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},\"name\":\"Overlay Plot with Telemetry Object\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with Telemetry Object\\nchrome\",\"modified\":1732413603960,\"location\":\"mine\",\"created\":1732413601820,\"persisted\":1732413603960},{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602540,\"created\":1732413600760,\"persisted\":1732413602540},{\"identifier\":{\"key\":\"ROOT\",\"namespace\":\"\"},\"name\":\"Open MCT\",\"type\":\"root\",\"composition\":[{\"key\":\"mine\",\"namespace\":\"\"}]}],\"navigationPath\":\"/browse/mine/20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"domainObject\":{\"identifier\":{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},\"name\":\"Overlay Plot with Telemetry Object\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with Telemetry Object\\nchrome\",\"modified\":1732413603960,\"location\":\"mine\",\"created\":1732413601820,\"persisted\":1732413603960}},{\"objectPath\":[{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602540,\"created\":1732413600760,\"persisted\":1732413602540},{\"identifier\":{\"key\":\"ROOT\",\"namespace\":\"\"},\"name\":\"Open MCT\",\"type\":\"root\",\"composition\":[{\"key\":\"mine\",\"namespace\":\"\"}]}],\"navigationPath\":\"/browse/mine\",\"domainObject\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"20e7d5fe-9cf8-4099-8957-9453a8954c67\",\"namespace\":\"\"},{\"key\":\"2db521a9-996d-4d04-a171-93f4c5c220af\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602540,\"created\":1732413600760,\"persisted\":1732413602540}}]"
|
"value": "[{\"objectPath\":[{\"identifier\":{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"},\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":0,\"infinityValues\":false,\"exceedFloat32\":false,\"staleness\":false},\"modified\":1732413602460,\"location\":\"mine\",\"created\":1732413602460,\"persisted\":1732413602460},{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602460,\"created\":1732413600960,\"persisted\":1732413602460},{\"identifier\":{\"key\":\"ROOT\",\"namespace\":\"\"},\"name\":\"Open MCT\",\"type\":\"root\",\"composition\":[{\"key\":\"mine\",\"namespace\":\"\"}]}],\"navigationPath\":\"/browse/mine/c6100044-56be-44b3-acca-6b9fddfb3849\",\"domainObject\":{\"identifier\":{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"},\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":0,\"infinityValues\":false,\"exceedFloat32\":false,\"staleness\":false},\"modified\":1732413602460,\"location\":\"mine\",\"created\":1732413602460,\"persisted\":1732413602460}},{\"objectPath\":[{\"identifier\":{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},\"name\":\"Overlay Plot with Telemetry Object\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with Telemetry Object\\nchrome\",\"modified\":1732413603880,\"location\":\"mine\",\"created\":1732413601740,\"persisted\":1732413603880},{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602460,\"created\":1732413600960,\"persisted\":1732413602460},{\"identifier\":{\"key\":\"ROOT\",\"namespace\":\"\"},\"name\":\"Open MCT\",\"type\":\"root\",\"composition\":[{\"key\":\"mine\",\"namespace\":\"\"}]}],\"navigationPath\":\"/browse/mine/e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"domainObject\":{\"identifier\":{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},\"name\":\"Overlay Plot with Telemetry Object\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with Telemetry Object\\nchrome\",\"modified\":1732413603880,\"location\":\"mine\",\"created\":1732413601740,\"persisted\":1732413603880}},{\"objectPath\":[{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602460,\"created\":1732413600960,\"persisted\":1732413602460},{\"identifier\":{\"key\":\"ROOT\",\"namespace\":\"\"},\"name\":\"Open MCT\",\"type\":\"root\",\"composition\":[{\"key\":\"mine\",\"namespace\":\"\"}]}],\"navigationPath\":\"/browse/mine\",\"domainObject\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"e78ca721-fb5e-409b-bf6d-597c87cb716f\",\"namespace\":\"\"},{\"key\":\"c6100044-56be-44b3-acca-6b9fddfb3849\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413602460,\"created\":1732413600960,\"persisted\":1732413602460}}]"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "mct-tree-expanded",
|
"name": "mct-tree-expanded",
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
"localStorage": [
|
"localStorage": [
|
||||||
{
|
{
|
||||||
"name": "mct",
|
"name": "mct",
|
||||||
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"98161570-a735-4a50-9c75-11b346ad3789\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413601340,\"created\":1732413600580,\"persisted\":1732413601340},\"98161570-a735-4a50-9c75-11b346ad3789\":{\"identifier\":{\"key\":\"98161570-a735-4a50-9c75-11b346ad3789\",\"namespace\":\"\"},\"name\":\"Overlay Plot with 5s Delay\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"477e60bb-4cba-4603-b4c9-2281ccf7e054\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"477e60bb-4cba-4603-b4c9-2281ccf7e054\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with 5s Delay\\nchrome\",\"modified\":1732413602660,\"location\":\"mine\",\"created\":1732413601340,\"persisted\":1732413602660},\"477e60bb-4cba-4603-b4c9-2281ccf7e054\":{\"identifier\":{\"key\":\"477e60bb-4cba-4603-b4c9-2281ccf7e054\",\"namespace\":\"\"},\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":5000,\"infinityValues\":false,\"staleness\":false},\"modified\":1732413602520,\"location\":\"98161570-a735-4a50-9c75-11b346ad3789\",\"created\":1732413602040,\"persisted\":1732413602520}}"
|
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"67ca2e0a-b37e-4eda-86a4-ccdbb228bbc0\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"modified\":1732413601720,\"created\":1732413600920,\"persisted\":1732413601720},\"67ca2e0a-b37e-4eda-86a4-ccdbb228bbc0\":{\"identifier\":{\"key\":\"67ca2e0a-b37e-4eda-86a4-ccdbb228bbc0\",\"namespace\":\"\"},\"name\":\"Overlay Plot with 5s Delay\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"8f524b49-ad06-47f9-98e0-087b31a2f3e0\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"8f524b49-ad06-47f9-98e0-087b31a2f3e0\",\"namespace\":\"\"}}]},\"notes\":\"framework/generateLocalStorageData.e2e.spec.js\\nGenerate Visual Test Data @localStorage @generatedata\\nGenerate Overlay Plot with 5s Delay\\nchrome\",\"modified\":1732413603020,\"location\":\"mine\",\"created\":1732413601720,\"persisted\":1732413603020},\"8f524b49-ad06-47f9-98e0-087b31a2f3e0\":{\"identifier\":{\"key\":\"8f524b49-ad06-47f9-98e0-087b31a2f3e0\",\"namespace\":\"\"},\"name\":\"VIPER Rover Heading\",\"type\":\"generator\",\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":5000,\"infinityValues\":false,\"exceedFloat32\":false,\"staleness\":false},\"modified\":1732413602920,\"location\":\"67ca2e0a-b37e-4eda-86a4-ccdbb228bbc0\",\"created\":1732413602420,\"persisted\":1732413602920}}"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "mct-tree-expanded",
|
"name": "mct-tree-expanded",
|
||||||
|
File diff suppressed because one or more lines are too long
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,12 +20,13 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures.js');
|
import {
|
||||||
const {
|
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
createNotification,
|
createNotification,
|
||||||
expandEntireTree
|
expandEntireTree,
|
||||||
} = require('../../appActions.js');
|
openObjectTreeContextMenu
|
||||||
|
} from '../../appActions.js';
|
||||||
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('AppActions', () => {
|
test.describe('AppActions', () => {
|
||||||
test('createDomainObjectsWithDefaults', async ({ page }) => {
|
test('createDomainObjectsWithDefaults', async ({ page }) => {
|
||||||
@ -155,7 +156,7 @@ test.describe('AppActions', () => {
|
|||||||
|
|
||||||
await page.goto('./#/browse/mine');
|
await page.goto('./#/browse/mine');
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click the object specified by 'type'
|
// Click the object specified by 'type'
|
||||||
await page.click(`li[role='menuitem']:text("Clock")`);
|
await page.click(`li[role='menuitem']:text("Clock")`);
|
||||||
@ -166,4 +167,13 @@ test.describe('AppActions', () => {
|
|||||||
const locatorTreeCollapsedItems = locatorTree.locator('role=treeitem[expanded=false]');
|
const locatorTreeCollapsedItems = locatorTree.locator('role=treeitem[expanded=false]');
|
||||||
expect(await locatorTreeCollapsedItems.count()).toBe(0);
|
expect(await locatorTreeCollapsedItems.count()).toBe(0);
|
||||||
});
|
});
|
||||||
|
test('openObjectTreeContextMenu', async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
const folder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder'
|
||||||
|
});
|
||||||
|
await openObjectTreeContextMenu(page, folder.url);
|
||||||
|
await expect(page.getByLabel('Menu')).toBeVisible();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -26,7 +26,7 @@ relates to how we've extended it (i.e. ./e2e/baseFixtures.js) and assumptions ma
|
|||||||
(`npm start` and ./e2e/webpack-dev-middleware.js)
|
(`npm start` and ./e2e/webpack-dev-middleware.js)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test } = require('../../baseFixtures.js');
|
import { test } from '../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('baseFixtures tests', () => {
|
test.describe('baseFixtures tests', () => {
|
||||||
//Skip this test for now https://github.com/nasa/openmct/issues/6785
|
//Skip this test for now https://github.com/nasa/openmct/issues/6785
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -45,8 +45,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
// Structure: Some standard Imports. Please update the required pathing.
|
// Structure: Some standard Imports. Please update the required pathing.
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
import { createDomainObjectWithDefaults } from '../../appActions.js';
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions');
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Structure:
|
* Structure:
|
||||||
@ -164,7 +164,7 @@ async function renameTimerFrom3DotMenu(page, timerUrl, newNameForTimer) {
|
|||||||
await page.goto(timerUrl);
|
await page.goto(timerUrl);
|
||||||
|
|
||||||
// Click on 3 Dot Menu
|
// Click on 3 Dot Menu
|
||||||
await page.locator('button[title="More options"]').click();
|
await page.locator('button[title="More actions"]').click();
|
||||||
|
|
||||||
// Click text=Edit Properties...
|
// Click text=Edit Properties...
|
||||||
await page.locator('text=Edit Properties...').click();
|
await page.locator('text=Edit Properties...').click();
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2022, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,7 +19,6 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
/* global __dirname */
|
|
||||||
/**
|
/**
|
||||||
* This test suite is dedicated to generating LocalStorage via Session Storage to be used
|
* This test suite is dedicated to generating LocalStorage via Session Storage to be used
|
||||||
* in some visual test suites like controlledClock.visual.spec.js. This suite should run to completion
|
* in some visual test suites like controlledClock.visual.spec.js. This suite should run to completion
|
||||||
@ -32,13 +31,11 @@
|
|||||||
* and is additionally verified in the validation test suites below.
|
* and is additionally verified in the validation test suites below.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures.js');
|
import { fileURLToPath } from 'url';
|
||||||
const {
|
|
||||||
createDomainObjectWithDefaults,
|
import { createDomainObjectWithDefaults, createExampleTelemetryObject } from '../../appActions.js';
|
||||||
createExampleTelemetryObject
|
import { MISSION_TIME } from '../../constants.js';
|
||||||
} = require('../../appActions.js');
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
const { MISSION_TIME } = require('../../constants.js');
|
|
||||||
const path = require('path');
|
|
||||||
|
|
||||||
const overlayPlotName = 'Overlay Plot with Telemetry Object';
|
const overlayPlotName = 'Overlay Plot with Telemetry Object';
|
||||||
|
|
||||||
@ -55,6 +52,70 @@ test.describe('Generate Visual Test Data @localStorage @generatedata', () => {
|
|||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('Generate display layout with 2 child display layouts', async ({ page, context }) => {
|
||||||
|
const parent = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: 'Parent Display Layout'
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: 'Child Layout 1',
|
||||||
|
parent: parent.uuid
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: 'Child Layout 2',
|
||||||
|
parent: parent.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(parent.url, { waitUntil: 'domcontentloaded' });
|
||||||
|
await page.getByLabel('Edit Object').click();
|
||||||
|
await page.getByLabel('Child Layout 2 Layout', { exact: true }).hover();
|
||||||
|
await page.getByLabel('Move Sub-object Frame').nth(1).click();
|
||||||
|
await page.getByLabel('X:').fill('30');
|
||||||
|
|
||||||
|
await page.getByLabel('Child Layout 1 Layout', { exact: true }).hover();
|
||||||
|
await page.getByLabel('Move Sub-object Frame').first().click();
|
||||||
|
await page.getByLabel('Y:').fill('30');
|
||||||
|
|
||||||
|
await page.getByLabel('Save').click();
|
||||||
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
|
//Save localStorage for future test execution
|
||||||
|
await context.storageState({
|
||||||
|
path: fileURLToPath(
|
||||||
|
new URL('../../../e2e/test-data/display_layout_with_child_layouts.json', import.meta.url)
|
||||||
|
)
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Generate flexible layout with 2 child display layouts', async ({ page, context }) => {
|
||||||
|
// Create Display Layout
|
||||||
|
const parent = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Flexible Layout',
|
||||||
|
name: 'Parent Flexible Layout'
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: 'Child Layout 1',
|
||||||
|
parent: parent.uuid
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: 'Child Layout 2',
|
||||||
|
parent: parent.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(parent.url, { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
//Save localStorage for future test execution
|
||||||
|
await context.storageState({
|
||||||
|
path: fileURLToPath(
|
||||||
|
new URL('../../../e2e/test-data/flexible_layout_with_child_layouts.json', import.meta.url)
|
||||||
|
)
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
// TODO: Visual test for the generated object here
|
// TODO: Visual test for the generated object here
|
||||||
// - Move to using appActions to create the overlay plot
|
// - Move to using appActions to create the overlay plot
|
||||||
// and embedded standard telemetry object
|
// and embedded standard telemetry object
|
||||||
@ -69,10 +130,10 @@ test.describe('Generate Visual Test Data @localStorage @generatedata', () => {
|
|||||||
const exampleTelemetry = await createExampleTelemetryObject(page);
|
const exampleTelemetry = await createExampleTelemetryObject(page);
|
||||||
|
|
||||||
// Make Link from Telemetry Object to Overlay Plot
|
// Make Link from Telemetry Object to Overlay Plot
|
||||||
await page.locator('button[title="More options"]').click();
|
await page.locator('button[title="More actions"]').click();
|
||||||
|
|
||||||
// Select 'Create Link' from dropdown
|
// Select 'Create Link' from dropdown
|
||||||
await page.getByRole('menuitem', { name: ' Create Link' }).click();
|
await page.getByRole('menuitem', { name: 'Create Link' }).click();
|
||||||
|
|
||||||
// Search and Select for overlay Plot within Create Modal
|
// Search and Select for overlay Plot within Create Modal
|
||||||
await page.getByRole('dialog').getByRole('searchbox', { name: 'Search Input' }).click();
|
await page.getByRole('dialog').getByRole('searchbox', { name: 'Search Input' }).click();
|
||||||
@ -128,7 +189,9 @@ test.describe('Generate Visual Test Data @localStorage @generatedata', () => {
|
|||||||
|
|
||||||
// Save localStorage for future test execution
|
// Save localStorage for future test execution
|
||||||
await context.storageState({
|
await context.storageState({
|
||||||
path: path.join(__dirname, '../../../e2e/test-data/overlay_plot_storage.json')
|
path: fileURLToPath(
|
||||||
|
new URL('../../../e2e/test-data/overlay_plot_storage.json', import.meta.url)
|
||||||
|
)
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
// TODO: Merge this with previous test. Edit object created in previous test.
|
// TODO: Merge this with previous test. Edit object created in previous test.
|
||||||
@ -142,8 +205,8 @@ test.describe('Generate Visual Test Data @localStorage @generatedata', () => {
|
|||||||
const swgWith5sDelay = await createExampleTelemetryObject(page, overlayPlot.uuid);
|
const swgWith5sDelay = await createExampleTelemetryObject(page, overlayPlot.uuid);
|
||||||
|
|
||||||
await page.goto(swgWith5sDelay.url);
|
await page.goto(swgWith5sDelay.url);
|
||||||
await page.getByTitle('More options').click();
|
await page.getByLabel('More actions').click();
|
||||||
await page.getByRole('menuitem', { name: ' Edit Properties...' }).click();
|
await page.getByLabel('Edit Properties...').click();
|
||||||
|
|
||||||
//Edit Example Telemetry Object to include 5s loading Delay
|
//Edit Example Telemetry Object to include 5s loading Delay
|
||||||
await page.locator('[aria-label="Loading Delay \\(ms\\)"]').fill('5000');
|
await page.locator('[aria-label="Loading Delay \\(ms\\)"]').fill('5000');
|
||||||
@ -162,17 +225,21 @@ test.describe('Generate Visual Test Data @localStorage @generatedata', () => {
|
|||||||
|
|
||||||
// Clear Recently Viewed
|
// Clear Recently Viewed
|
||||||
await page.getByRole('button', { name: 'Clear Recently Viewed' }).click();
|
await page.getByRole('button', { name: 'Clear Recently Viewed' }).click();
|
||||||
await page.getByRole('button', { name: 'OK' }).click();
|
await page.getByRole('button', { name: 'OK', exact: true }).click();
|
||||||
//Save localStorage for future test execution
|
//Save localStorage for future test execution
|
||||||
await context.storageState({
|
await context.storageState({
|
||||||
path: path.join(__dirname, '../../../e2e/test-data/overlay_plot_with_delay_storage.json')
|
path: fileURLToPath(
|
||||||
|
new URL('../../../e2e/test-data/overlay_plot_with_delay_storage.json', import.meta.url)
|
||||||
|
)
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test.describe('Validate Overlay Plot with Telemetry Object @localStorage @generatedata', () => {
|
test.describe('Validate Overlay Plot with Telemetry Object @localStorage @generatedata', () => {
|
||||||
test.use({
|
test.use({
|
||||||
storageState: path.join(__dirname, '../../../e2e/test-data/overlay_plot_storage.json')
|
storageState: fileURLToPath(
|
||||||
|
new URL('../../../e2e/test-data/overlay_plot_storage.json', import.meta.url)
|
||||||
|
)
|
||||||
});
|
});
|
||||||
test('Validate Overlay Plot with Telemetry Object', async ({ page }) => {
|
test('Validate Overlay Plot with Telemetry Object', async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
@ -214,9 +281,8 @@ test.describe('Validate Overlay Plot with Telemetry Object @localStorage @genera
|
|||||||
|
|
||||||
test.describe('Validate Overlay Plot with 5s Delay Telemetry Object @localStorage @generatedata', () => {
|
test.describe('Validate Overlay Plot with 5s Delay Telemetry Object @localStorage @generatedata', () => {
|
||||||
test.use({
|
test.use({
|
||||||
storageState: path.join(
|
storageState: fileURLToPath(
|
||||||
__dirname,
|
new URL('../../../e2e/test-data/overlay_plot_with_delay_storage.json', import.meta.url)
|
||||||
'../../../e2e/test-data/overlay_plot_with_delay_storage.json'
|
|
||||||
)
|
)
|
||||||
});
|
});
|
||||||
test('Validate Overlay Plot with Telemetry Object', async ({ page }) => {
|
test('Validate Overlay Plot with Telemetry Object', async ({ page }) => {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,7 +25,7 @@ This test suite is dedicated to testing our use of our custom fixtures to verify
|
|||||||
that they are working as expected.
|
that they are working as expected.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test } = require('../../pluginFixtures.js');
|
import { test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
// eslint-disable-next-line playwright/no-skipped-test
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
test.describe.skip('pluginFixtures tests', () => {
|
test.describe.skip('pluginFixtures tests', () => {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,42 +24,36 @@
|
|||||||
This test suite is dedicated to tests which verify branding related components.
|
This test suite is dedicated to tests which verify branding related components.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../baseFixtures.js');
|
import { expect, test } from '../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('Branding tests', () => {
|
test.describe('Branding tests', () => {
|
||||||
test('About Modal launches with basic branding properties', async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
// Go to baseURL
|
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
});
|
||||||
// Click About button
|
test('About Modal launches with basic branding properties', async ({ page }) => {
|
||||||
await page.click('.l-shell__app-logo');
|
await page.getByLabel('About Modal').click();
|
||||||
|
|
||||||
// Verify that the NASA Logo Appears
|
// Verify that the NASA Logo Appears
|
||||||
await expect(page.locator('.c-about__image')).toBeVisible();
|
await expect(page.getByAltText('Open MCT Splash Logo')).toBeVisible();
|
||||||
|
|
||||||
// Modify the Build information in 'about' Modal
|
// Modify the Build information in 'about' Modal
|
||||||
const versionInformationLocator = page.locator('ul.t-info.l-info.s-info').first();
|
await expect.soft(page.getByLabel('Version Number')).toContainText(/Version: \d/);
|
||||||
await expect(versionInformationLocator).toBeEnabled();
|
|
||||||
await expect.soft(versionInformationLocator).toContainText(/Version: \d/);
|
|
||||||
await expect
|
await expect
|
||||||
.soft(versionInformationLocator)
|
.soft(page.getByLabel('Build Date'))
|
||||||
.toContainText(/Build Date: ((?:Mon|Tue|Wed|Thu|Fri|Sat|Sun))/);
|
.toContainText(/Build Date: ((?:Mon|Tue|Wed|Thu|Fri|Sat|Sun))/);
|
||||||
await expect.soft(versionInformationLocator).toContainText(/Revision: \b[0-9a-f]{5,40}\b/);
|
await expect.soft(page.getByLabel('Revision')).toContainText(/Revision: \b[0-9a-f]{5,40}\b/);
|
||||||
await expect.soft(versionInformationLocator).toContainText(/Branch: ./);
|
await expect.soft(page.getByLabel('Branch')).toContainText(/Branch: ./);
|
||||||
});
|
});
|
||||||
test('Verify Links in About Modal @2p', async ({ page }) => {
|
test('Verify Links in About Modal @2p', async ({ page }) => {
|
||||||
// Go to baseURL
|
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
|
||||||
|
|
||||||
// Click About button
|
// Click About button
|
||||||
await page.click('.l-shell__app-logo');
|
await page.getByLabel('About Modal').click();
|
||||||
|
|
||||||
// Verify that clicking on the third party licenses information opens up another tab on licenses url
|
// Verify that clicking on the third party licenses information opens up another tab on licenses url
|
||||||
const [page2] = await Promise.all([
|
const [page2] = await Promise.all([
|
||||||
page.waitForEvent('popup'),
|
page.waitForEvent('popup'),
|
||||||
page.locator('text=click here for third party licensing information').click()
|
page.getByText('click here for third party licensing information').click()
|
||||||
]);
|
]);
|
||||||
await page2.waitForLoadState('networkidle'); //Avoids timing issues with juggler/firefox
|
await page2.waitForLoadState('domcontentloaded'); //Avoids timing issues with juggler/firefox
|
||||||
expect(page2.waitForURL('**/licenses**')).toBeTruthy();
|
expect(page2.waitForURL('**/licenses**')).toBeTruthy();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,8 +24,8 @@
|
|||||||
Verify that the "Clear Data" menu action performs as expected for various object types.
|
Verify that the "Clear Data" menu action performs as expected for various object types.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures.js');
|
import { createDomainObjectWithDefaults } from '../../appActions.js';
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions.js');
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
||||||
|
|
||||||
@ -43,17 +43,22 @@ test.describe('Clear Data Action', () => {
|
|||||||
await expect(page.locator(backgroundImageSelector)).toBeVisible();
|
await expect(page.locator(backgroundImageSelector)).toBeVisible();
|
||||||
});
|
});
|
||||||
test('works as expected with Example Imagery', async ({ page }) => {
|
test('works as expected with Example Imagery', async ({ page }) => {
|
||||||
await expect(await page.locator('.c-thumb__image').count()).toBeGreaterThan(0);
|
expect(await page.locator('.c-thumb__image').count()).toBeGreaterThan(0);
|
||||||
// Click the "Clear Data" menu action
|
// Click the "Clear Data" menu action
|
||||||
await page.getByTitle('More options').click();
|
await page.getByTitle('More actions').click();
|
||||||
const clearDataMenuItem = page.getByRole('menuitem', {
|
await expect(
|
||||||
name: 'Clear Data'
|
page.getByRole('menuitem', {
|
||||||
});
|
name: 'Clear Data for Object'
|
||||||
await expect(clearDataMenuItem).toBeEnabled();
|
})
|
||||||
await clearDataMenuItem.click();
|
).toBeEnabled();
|
||||||
|
await page
|
||||||
|
.getByRole('menuitem', {
|
||||||
|
name: 'Clear Data for Object'
|
||||||
|
})
|
||||||
|
.click();
|
||||||
|
|
||||||
// Verify that the background image is no longer visible
|
// Verify that the background image is no longer visible
|
||||||
await expect(page.locator(backgroundImageSelector)).toBeHidden();
|
await expect(page.locator(backgroundImageSelector)).toBeHidden();
|
||||||
await expect(await page.locator('.c-thumb__image').count()).toBe(0);
|
expect(await page.locator('.c-thumb__image').count()).toBe(0);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,7 +25,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('CouchDB Status Indicator with mocked responses @couchdb', () => {
|
test.describe('CouchDB Status Indicator with mocked responses @couchdb', () => {
|
||||||
test.use({ failOnConsoleError: false });
|
test.use({ failOnConsoleError: false });
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,8 +24,8 @@
|
|||||||
This test suite is dedicated to tests which verify the basic operations surrounding the example event generator.
|
This test suite is dedicated to tests which verify the basic operations surrounding the example event generator.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../pluginFixtures');
|
import { createDomainObjectWithDefaults } from '../../../appActions.js';
|
||||||
const { createDomainObjectWithDefaults } = require('../../../appActions');
|
import { expect, test } from '../../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('Example Event Generator CRUD Operations', () => {
|
test.describe('Example Event Generator CRUD Operations', () => {
|
||||||
test('Can create a Test Event Generator and it results in the table View', async ({ page }) => {
|
test('Can create a Test Event Generator and it results in the table View', async ({ page }) => {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,7 +24,7 @@
|
|||||||
This test suite is dedicated to tests which verify the basic operations surrounding conditionSets.
|
This test suite is dedicated to tests which verify the basic operations surrounding conditionSets.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../baseFixtures');
|
import { expect, test } from '../../../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('Sine Wave Generator', () => {
|
test.describe('Sine Wave Generator', () => {
|
||||||
test('Create new Sine Wave Generator Object and validate create Form Logic', async ({
|
test('Create new Sine Wave Generator Object and validate create Form Logic', async ({
|
||||||
@ -38,7 +38,7 @@ test.describe('Sine Wave Generator', () => {
|
|||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click Sine Wave Generator
|
// Click Sine Wave Generator
|
||||||
await page.click('text=Sine Wave Generator');
|
await page.click('text=Sine Wave Generator');
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,15 +19,16 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
/* global __dirname */
|
|
||||||
/*
|
/*
|
||||||
This test suite is dedicated to tests which verify form functionality in isolation
|
This test suite is dedicated to tests which verify form functionality in isolation
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
import { fileURLToPath } from 'url';
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions');
|
import { v4 as genUuid } from 'uuid';
|
||||||
const genUuid = require('uuid').v4;
|
|
||||||
const path = require('path');
|
import { createDomainObjectWithDefaults } from '../../appActions.js';
|
||||||
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
const TEST_FOLDER = 'test folder';
|
const TEST_FOLDER = 'test folder';
|
||||||
const jsonFilePath = 'e2e/test-data/ExampleLayouts.json';
|
const jsonFilePath = 'e2e/test-data/ExampleLayouts.json';
|
||||||
@ -40,8 +41,8 @@ test.describe('Form Validation Behavior', () => {
|
|||||||
//Go to baseURL
|
//Go to baseURL
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
await page.getByRole('menuitem', { name: ' Folder' }).click();
|
await page.getByRole('menuitem', { name: 'Folder' }).click();
|
||||||
|
|
||||||
// Fill in empty string into title and trigger validation with 'Tab'
|
// Fill in empty string into title and trigger validation with 'Tab'
|
||||||
await page.click('text=Properties Title Notes >> input[type="text"]');
|
await page.click('text=Properties Title Notes >> input[type="text"]');
|
||||||
@ -72,14 +73,14 @@ test.describe('Form Validation Behavior', () => {
|
|||||||
test.describe('Form File Input Behavior', () => {
|
test.describe('Form File Input Behavior', () => {
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.addInitScript({
|
await page.addInitScript({
|
||||||
path: path.join(__dirname, '../../helper', 'addInitFileInputObject.js')
|
path: fileURLToPath(new URL('../../helper/addInitFileInputObject.js', import.meta.url))
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Can select a JSON file type', async ({ page }) => {
|
test('Can select a JSON file type', async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
await page.getByRole('button', { name: ' Create ' }).click();
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
await page.getByRole('menuitem', { name: 'JSON File Input Object' }).click();
|
await page.getByRole('menuitem', { name: 'JSON File Input Object' }).click();
|
||||||
|
|
||||||
await page.setInputFiles('#fileElem', jsonFilePath);
|
await page.setInputFiles('#fileElem', jsonFilePath);
|
||||||
@ -93,7 +94,7 @@ test.describe('Form File Input Behavior', () => {
|
|||||||
test('Can select an image file type', async ({ page }) => {
|
test('Can select an image file type', async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
await page.getByRole('button', { name: ' Create ' }).click();
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
await page.getByRole('menuitem', { name: 'Image File Input Object' }).click();
|
await page.getByRole('menuitem', { name: 'Image File Input Object' }).click();
|
||||||
|
|
||||||
await page.setInputFiles('#fileElem', imageFilePath);
|
await page.setInputFiles('#fileElem', imageFilePath);
|
||||||
@ -109,7 +110,7 @@ test.describe('Persistence operations @addInit', () => {
|
|||||||
// add non persistable root item
|
// add non persistable root item
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.addInitScript({
|
await page.addInitScript({
|
||||||
path: path.join(__dirname, '../../helper', 'addNoneditableObject.js')
|
path: fileURLToPath(new URL('../../helper/addNoneditableObject.js', import.meta.url))
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -120,7 +121,7 @@ test.describe('Persistence operations @addInit', () => {
|
|||||||
});
|
});
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
await page.click('text=Condition Set');
|
await page.click('text=Condition Set');
|
||||||
|
|
||||||
@ -157,7 +158,7 @@ test.describe('Persistence operations @couchdb', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Open the edit form for the clock object
|
// Open the edit form for the clock object
|
||||||
await page.click('button[title="More options"]');
|
await page.click('button[title="More actions"]');
|
||||||
await page.click('li[title="Edit properties of this object."]');
|
await page.click('li[title="Edit properties of this object."]');
|
||||||
|
|
||||||
// Modify the display format from default 12hr -> 24hr and click 'Save'
|
// Modify the display format from default 12hr -> 24hr and click 'Save'
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,20 +19,20 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
/* global __dirname */
|
|
||||||
/*
|
/*
|
||||||
This test suite is dedicated to tests which verify persistability checks
|
This test suite is dedicated to tests which verify persistability checks
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../baseFixtures.js');
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
const path = require('path');
|
import { expect, test } from '../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('Persistence operations @addInit', () => {
|
test.describe('Persistence operations @addInit', () => {
|
||||||
// add non persistable root item
|
// add non persistable root item
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.addInitScript({
|
await page.addInitScript({
|
||||||
path: path.join(__dirname, '../../helper', 'addNoneditableObject.js')
|
path: fileURLToPath(new URL('../../helper/addNoneditableObject.js', import.meta.url))
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
127
e2e/tests/functional/missionStatus.e2e.spec.js
Normal file
127
e2e/tests/functional/missionStatus.e2e.spec.js
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
This test suite is dedicated to tests which verify persistability checks
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
|
import { expect, test } from '../../baseFixtures.js';
|
||||||
|
|
||||||
|
test.describe('Mission Status @addInit', () => {
|
||||||
|
const NO_GO = '0';
|
||||||
|
const GO = '1';
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// FIXME: determine if plugins will be added to index.html or need to be injected
|
||||||
|
await page.addInitScript({
|
||||||
|
path: fileURLToPath(new URL('../../helper/addInitExampleUser.js', import.meta.url))
|
||||||
|
});
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
await expect(page.getByText('Select Role')).toBeVisible();
|
||||||
|
// Description should be empty https://github.com/nasa/openmct/issues/6978
|
||||||
|
await expect(page.getByLabel('Dialog message')).toBeHidden();
|
||||||
|
// set role
|
||||||
|
await page.getByRole('button', { name: 'Select', exact: true }).click();
|
||||||
|
// dismiss role confirmation popup
|
||||||
|
await page.getByRole('button', { name: 'Dismiss' }).click();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Basic functionality', async ({ page }) => {
|
||||||
|
const imageryStatusSelect = page.getByRole('combobox', { name: 'Imagery' });
|
||||||
|
const commandingStatusSelect = page.getByRole('combobox', { name: 'Commanding' });
|
||||||
|
const drivingStatusSelect = page.getByRole('combobox', { name: 'Driving' });
|
||||||
|
const missionStatusPanel = page.getByRole('dialog', { name: 'User Control Panel' });
|
||||||
|
|
||||||
|
await test.step('Mission status panel shows/hides when toggled', async () => {
|
||||||
|
// Ensure that clicking the button toggles the dialog
|
||||||
|
await page.getByLabel('Toggle Mission Status Panel').click();
|
||||||
|
await expect(missionStatusPanel).toBeVisible();
|
||||||
|
await page.getByLabel('Toggle Mission Status Panel').click();
|
||||||
|
await expect(missionStatusPanel).toBeHidden();
|
||||||
|
await page.getByLabel('Toggle Mission Status Panel').click();
|
||||||
|
await expect(missionStatusPanel).toBeVisible();
|
||||||
|
|
||||||
|
// Ensure that clicking the close button closes the dialog
|
||||||
|
await page.getByLabel('Close Mission Status Panel').click();
|
||||||
|
await expect(missionStatusPanel).toBeHidden();
|
||||||
|
await page.getByLabel('Toggle Mission Status Panel').click();
|
||||||
|
await expect(missionStatusPanel).toBeVisible();
|
||||||
|
|
||||||
|
// Ensure clicking off the dialog also closes it
|
||||||
|
await page.getByLabel('My Items Grid View').click();
|
||||||
|
await expect(missionStatusPanel).toBeHidden();
|
||||||
|
await page.getByLabel('Toggle Mission Status Panel').click();
|
||||||
|
await expect(missionStatusPanel).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Mission action statuses have correct defaults and can be set', async () => {
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
|
||||||
|
await setMissionStatus(page, 'Imagery', GO);
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
|
||||||
|
await setMissionStatus(page, 'Commanding', GO);
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
|
||||||
|
await setMissionStatus(page, 'Driving', GO);
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(GO);
|
||||||
|
|
||||||
|
await setMissionStatus(page, 'Imagery', NO_GO);
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(GO);
|
||||||
|
|
||||||
|
await setMissionStatus(page, 'Commanding', NO_GO);
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(GO);
|
||||||
|
|
||||||
|
await setMissionStatus(page, 'Driving', NO_GO);
|
||||||
|
await expect(imageryStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(commandingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
await expect(drivingStatusSelect).toHaveValue(NO_GO);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {'Commanding'|'Imagery'|'Driving'} action
|
||||||
|
* @param {'0'|'1'} status
|
||||||
|
*/
|
||||||
|
async function setMissionStatus(page, action, status) {
|
||||||
|
await page.getByRole('combobox', { name: action }).selectOption(status);
|
||||||
|
await expect(
|
||||||
|
page.getByRole('alert').filter({ hasText: 'Successfully set mission status' })
|
||||||
|
).toBeVisible();
|
||||||
|
await page.getByLabel('Dismiss').click();
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,8 +24,8 @@
|
|||||||
This test suite is dedicated to tests which verify the basic operations surrounding moving & linking objects.
|
This test suite is dedicated to tests which verify the basic operations surrounding moving & linking objects.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
import { createDomainObjectWithDefaults } from '../../appActions.js';
|
||||||
const { createDomainObjectWithDefaults } = require('../../appActions');
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('Move & link item tests', () => {
|
test.describe('Move & link item tests', () => {
|
||||||
test('Create a basic object and verify that it can be moved to another folder', async ({
|
test('Create a basic object and verify that it can be moved to another folder', async ({
|
||||||
@ -149,7 +149,7 @@ test.describe('Move & link item tests', () => {
|
|||||||
|
|
||||||
// Finish editing and save Telemetry Table
|
// Finish editing and save Telemetry Table
|
||||||
await page.locator('.c-button--menu.c-button--major.icon-save').click();
|
await page.locator('.c-button--menu.c-button--major.icon-save').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Create New Folder Basic Domain Object
|
// Create New Folder Basic Domain Object
|
||||||
let folder = 'Test Folder';
|
let folder = 'Test Folder';
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,8 +24,8 @@
|
|||||||
This test suite is dedicated to tests which verify Open MCT's Notification functionality
|
This test suite is dedicated to tests which verify Open MCT's Notification functionality
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { createDomainObjectWithDefaults, createNotification } = require('../../appActions');
|
import { createDomainObjectWithDefaults, createNotification } from '../../appActions.js';
|
||||||
const { test, expect } = require('../../pluginFixtures');
|
import { expect, test } from '../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('Notifications List', () => {
|
test.describe('Notifications List', () => {
|
||||||
test.fixme('Notifications can be dismissed individually', async ({ page }) => {
|
test.fixme('Notifications can be dismissed individually', async ({ page }) => {
|
||||||
@ -91,28 +91,30 @@ test.describe('Notification Overlay', () => {
|
|||||||
// Create a new Display Layout object
|
// Create a new Display Layout object
|
||||||
await createDomainObjectWithDefaults(page, { type: 'Display Layout' });
|
await createDomainObjectWithDefaults(page, { type: 'Display Layout' });
|
||||||
|
|
||||||
|
// Dismiss notification banner
|
||||||
|
await page.getByRole('button', { name: 'Dismiss' }).click();
|
||||||
|
|
||||||
// Click on the button "Review 1 Notification"
|
// Click on the button "Review 1 Notification"
|
||||||
await page.click('button[aria-label="Review 1 Notification"]');
|
await page.getByRole('button', { name: 'Review 1 Notification' }).click();
|
||||||
|
|
||||||
// Verify that Notification List is open
|
// Verify that Notification List is open
|
||||||
expect(await page.locator('div[role="dialog"]').isVisible()).toBe(true);
|
await expect(page.getByRole('dialog', { name: 'Overlay' })).toBeVisible();
|
||||||
|
|
||||||
// Wait until there is no Notification Banner
|
// Wait until there is no Notification Banner
|
||||||
await page.waitForSelector('div[role="alert"]', { state: 'detached' });
|
await expect(page.getByRole('alert')).not.toBeAttached();
|
||||||
|
|
||||||
// Click on the "Close" button of the Notification List
|
// Click on the "Close" button of the Notification List
|
||||||
await page.click('button[aria-label="Close"]');
|
await page.getByRole('button', { name: 'Close' }).click();
|
||||||
|
|
||||||
// On the Display Layout object, click on the "Edit" button
|
// On the Display Layout object, click on the "Edit" button
|
||||||
await page.click('button[title="Edit"]');
|
await page.getByRole('button', { name: 'Edit Object' }).click();
|
||||||
|
|
||||||
// Click on the "Save" button
|
// Click on the "Save" button
|
||||||
await page.click('button[title="Save"]');
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|
||||||
// Click on the "Save and Finish Editing" option
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
await page.click('li[title="Save and Finish Editing"]');
|
|
||||||
|
|
||||||
// Verify that Notification List is NOT open
|
// Verify that Notification List is NOT open
|
||||||
expect(await page.locator('div[role="dialog"]').isVisible()).toBe(false);
|
await expect(page.getByRole('dialog', { name: 'Overlay' })).toBeHidden();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,15 +19,26 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
const { test, expect } = require('../../../pluginFixtures');
|
import fs from 'fs';
|
||||||
const { createPlanFromJSON, createDomainObjectWithDefaults } = require('../../../appActions');
|
|
||||||
const testPlan1 = require('../../../test-data/examplePlans/ExamplePlan_Small1.json');
|
import { getPreciseDuration } from '../../../../src/utils/duration.js';
|
||||||
const testPlan2 = require('../../../test-data/examplePlans/ExamplePlan_Small2.json');
|
import { createDomainObjectWithDefaults, createPlanFromJSON } from '../../../appActions.js';
|
||||||
const {
|
import {
|
||||||
assertPlanActivities,
|
assertPlanActivities,
|
||||||
setBoundsToSpanAllActivities
|
setBoundsToSpanAllActivities
|
||||||
} = require('../../../helper/planningUtils');
|
} from '../../../helper/planningUtils.js';
|
||||||
const { getPreciseDuration } = require('../../../../src/utils/duration');
|
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', () => {
|
test.describe('Gantt Chart', () => {
|
||||||
let ganttChart;
|
let ganttChart;
|
||||||
@ -58,7 +69,7 @@ test.describe('Gantt Chart', () => {
|
|||||||
.getByRole('dialog')
|
.getByRole('dialog')
|
||||||
.filter({ hasText: 'This action will replace the current Plan. Do you want to continue?' });
|
.filter({ hasText: 'This action will replace the current Plan. Do you want to continue?' });
|
||||||
await expect(replaceModal).toBeVisible();
|
await expect(replaceModal).toBeVisible();
|
||||||
await page.getByRole('button', { name: 'OK' }).click();
|
await page.getByRole('button', { name: 'Ok', exact: true }).click();
|
||||||
|
|
||||||
await assertPlanActivities(page, testPlan2, ganttChart.url);
|
await assertPlanActivities(page, testPlan2, ganttChart.url);
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,10 +19,27 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
const { test } = require('../../../pluginFixtures');
|
import fs from 'fs';
|
||||||
const { createPlanFromJSON } = require('../../../appActions');
|
|
||||||
const testPlan1 = require('../../../test-data/examplePlans/ExamplePlan_Small1.json');
|
import { createPlanFromJSON } from '../../../appActions.js';
|
||||||
const { assertPlanActivities } = require('../../../helper/planningUtils');
|
import {
|
||||||
|
addPlanGetInterceptor,
|
||||||
|
assertPlanActivities,
|
||||||
|
assertPlanOrderedSwimLanes
|
||||||
|
} 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 testPlanWithOrderedLanes = JSON.parse(
|
||||||
|
fs.readFileSync(
|
||||||
|
new URL('../../../test-data/examplePlans/ExamplePlanWithOrderedLanes.json', import.meta.url)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
test.describe('Plan', () => {
|
test.describe('Plan', () => {
|
||||||
let plan;
|
let plan;
|
||||||
@ -36,4 +53,57 @@ test.describe('Plan', () => {
|
|||||||
test('Displays all plan events', async ({ page }) => {
|
test('Displays all plan events', async ({ page }) => {
|
||||||
await assertPlanActivities(page, testPlan1, plan.url);
|
await assertPlanActivities(page, testPlan1, plan.url);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('Displays plans with ordered swim lanes configuration', async ({ page }) => {
|
||||||
|
// Add configuration for swim lanes
|
||||||
|
await addPlanGetInterceptor(page);
|
||||||
|
// Create the plan
|
||||||
|
const planWithSwimLanes = await createPlanFromJSON(page, {
|
||||||
|
json: testPlanWithOrderedLanes
|
||||||
|
});
|
||||||
|
await assertPlanOrderedSwimLanes(page, testPlanWithOrderedLanes, planWithSwimLanes.url);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows setting the state of an activity when selected.', async ({ page }) => {
|
||||||
|
const groups = Object.keys(testPlan1);
|
||||||
|
const firstGroupKey = groups[0];
|
||||||
|
const firstGroupItems = testPlan1[firstGroupKey];
|
||||||
|
const firstActivity = firstGroupItems[0];
|
||||||
|
const lastActivity = firstGroupItems[firstGroupItems.length - 1];
|
||||||
|
const startBound = firstActivity.start;
|
||||||
|
// Set the endBound to the end time of the current activity
|
||||||
|
let endBound = lastActivity.end;
|
||||||
|
// eslint-disable-next-line playwright/no-conditional-in-test
|
||||||
|
if (endBound === startBound) {
|
||||||
|
// Prevent oddities with setting start and end bound equal
|
||||||
|
// via URL params
|
||||||
|
endBound += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Switch to fixed time mode with all plan events within the bounds
|
||||||
|
await page.goto(
|
||||||
|
`${plan.url}?tc.mode=fixed&tc.startBound=${startBound}&tc.endBound=${endBound}&tc.timeSystem=utc&view=plan.view`
|
||||||
|
);
|
||||||
|
|
||||||
|
// select the first activity in the list
|
||||||
|
await page.getByText('Past event 1').click();
|
||||||
|
|
||||||
|
// Find the activity state section in the inspector
|
||||||
|
await page.getByRole('tab', { name: 'Activity' }).click();
|
||||||
|
|
||||||
|
// Check that activity state dropdown selection shows the `set status` option by default
|
||||||
|
await expect(page.getByLabel('Activity Status').locator("[aria-selected='true']")).toHaveText(
|
||||||
|
'Not started'
|
||||||
|
);
|
||||||
|
|
||||||
|
// Change the selection of the activity status
|
||||||
|
await page.getByRole('combobox').selectOption({ label: 'Aborted' });
|
||||||
|
// select a different activity and back to the previous one
|
||||||
|
await page.getByText('Past event 2').click();
|
||||||
|
await page.getByText('Past event 1').click();
|
||||||
|
// Check that activity state dropdown selection shows the previously selected option by default
|
||||||
|
await expect(page.getByLabel('Activity Status').locator("[aria-selected='true']")).toHaveText(
|
||||||
|
'Aborted'
|
||||||
|
);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,57 +19,34 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
import fs from 'fs';
|
||||||
|
|
||||||
const { test, expect } = require('../../../pluginFixtures');
|
import { createDomainObjectWithDefaults, createPlanFromJSON } from '../../../appActions.js';
|
||||||
const { createDomainObjectWithDefaults, createPlanFromJSON } = require('../../../appActions');
|
import { getEarliestStartTime } from '../../../helper/planningUtils';
|
||||||
|
import { expect, test } from '../../../pluginFixtures.js';
|
||||||
|
|
||||||
const testPlan = {
|
const examplePlanSmall3 = JSON.parse(
|
||||||
TEST_GROUP: [
|
fs.readFileSync(
|
||||||
{
|
new URL('../../../test-data/examplePlans/ExamplePlan_Small3.json', import.meta.url)
|
||||||
name: 'Past event 1',
|
)
|
||||||
start: 1660320408000,
|
);
|
||||||
end: 1660343797000,
|
const examplePlanSmall1 = JSON.parse(
|
||||||
type: 'TEST-GROUP',
|
fs.readFileSync(
|
||||||
color: 'orange',
|
new URL('../../../test-data/examplePlans/ExamplePlan_Small1.json', import.meta.url)
|
||||||
textColor: 'white'
|
)
|
||||||
},
|
);
|
||||||
{
|
// eslint-disable-next-line no-unused-vars
|
||||||
name: 'Past event 2',
|
const START_TIME_COLUMN = 0;
|
||||||
start: 1660406808000,
|
// eslint-disable-next-line no-unused-vars
|
||||||
end: 1660429160000,
|
const END_TIME_COLUMN = 1;
|
||||||
type: 'TEST-GROUP',
|
const TIME_TO_FROM_COLUMN = 2;
|
||||||
color: 'orange',
|
// eslint-disable-next-line no-unused-vars
|
||||||
textColor: 'white'
|
const ACTIVITY_COLUMN = 3;
|
||||||
},
|
const HEADER_ROW = 0;
|
||||||
{
|
const NUM_COLUMNS = 5;
|
||||||
name: 'Past event 3',
|
|
||||||
start: 1660493208000,
|
|
||||||
end: 1660503981000,
|
|
||||||
type: 'TEST-GROUP',
|
|
||||||
color: 'orange',
|
|
||||||
textColor: 'white'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'Past event 4',
|
|
||||||
start: 1660579608000,
|
|
||||||
end: 1660624108000,
|
|
||||||
type: 'TEST-GROUP',
|
|
||||||
color: 'orange',
|
|
||||||
textColor: 'white'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'Past event 5',
|
|
||||||
start: 1660666008000,
|
|
||||||
end: 1660681529000,
|
|
||||||
type: 'TEST-GROUP',
|
|
||||||
color: 'orange',
|
|
||||||
textColor: 'white'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
};
|
|
||||||
|
|
||||||
test.describe('Time List', () => {
|
test.describe('Time List', () => {
|
||||||
test('Create a Time List, add a single Plan to it and verify all the activities are displayed with no milliseconds', async ({
|
test("Create a Time List, add a single Plan to it, verify all the activities are displayed with no milliseconds and selecting an activity shows it's properties", async ({
|
||||||
page
|
page
|
||||||
}) => {
|
}) => {
|
||||||
// Goto baseURL
|
// Goto baseURL
|
||||||
@ -84,21 +61,18 @@ test.describe('Time List', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await test.step('Create a Plan and add it to the timelist', async () => {
|
await test.step('Create a Plan and add it to the timelist', async () => {
|
||||||
const createdPlan = await createPlanFromJSON(page, {
|
await createPlanFromJSON(page, {
|
||||||
name: 'Test Plan',
|
name: 'Test Plan',
|
||||||
json: testPlan
|
json: examplePlanSmall1,
|
||||||
|
parent: timelist.uuid
|
||||||
});
|
});
|
||||||
|
const groups = Object.keys(examplePlanSmall1);
|
||||||
await page.goto(timelist.url);
|
const firstGroupKey = groups[0];
|
||||||
// Expand the tree to show the plan
|
const firstGroupItems = examplePlanSmall1[firstGroupKey];
|
||||||
await page.click("button[title='Show selected item in tree']");
|
const firstActivity = firstGroupItems[0];
|
||||||
await page.dragAndDrop(`role=treeitem[name=/${createdPlan.name}/]`, '.c-object-view');
|
const lastActivity = firstGroupItems[firstGroupItems.length - 1];
|
||||||
await page.click("button[title='Save']");
|
const startBound = firstActivity.start;
|
||||||
await page.click("li[title='Save and Finish Editing']");
|
const endBound = lastActivity.end;
|
||||||
const startBound = testPlan.TEST_GROUP[0].start;
|
|
||||||
const endBound = testPlan.TEST_GROUP[testPlan.TEST_GROUP.length - 1].end;
|
|
||||||
|
|
||||||
await page.goto(timelist.url);
|
|
||||||
|
|
||||||
// Switch to fixed time mode with all plan events within the bounds
|
// Switch to fixed time mode with all plan events within the bounds
|
||||||
await page.goto(
|
await page.goto(
|
||||||
@ -106,13 +80,14 @@ test.describe('Time List', () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Verify all events are displayed
|
// Verify all events are displayed
|
||||||
const eventCount = await page.locator('.js-list-item').count();
|
const eventCount = await page.getByRole('row').count();
|
||||||
expect(eventCount).toEqual(testPlan.TEST_GROUP.length);
|
// subtracting one for the header
|
||||||
|
await expect(eventCount - 1).toEqual(firstGroupItems.length);
|
||||||
});
|
});
|
||||||
|
|
||||||
await test.step('Does not show milliseconds in times', async () => {
|
await test.step('Does not show milliseconds in times', async () => {
|
||||||
// Get the first activity
|
// Get an activity
|
||||||
const row = page.locator('.js-list-item').first();
|
const row = page.getByRole('row').nth(2);
|
||||||
// Verify that none fo the times have milliseconds displayed.
|
// Verify that none fo the times have milliseconds displayed.
|
||||||
// Example: 2024-11-17T16:00:00Z is correct and 2024-11-17T16:00:00.000Z is wrong
|
// Example: 2024-11-17T16:00:00Z is correct and 2024-11-17T16:00:00.000Z is wrong
|
||||||
|
|
||||||
@ -120,5 +95,243 @@ test.describe('Time List', () => {
|
|||||||
await expect(row.locator('.--end')).not.toContainText('.');
|
await expect(row.locator('.--end')).not.toContainText('.');
|
||||||
await expect(row.locator('.--duration')).not.toContainText('.');
|
await expect(row.locator('.--duration')).not.toContainText('.');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await test.step('Shows activity properties when a row is selected', async () => {
|
||||||
|
await page.getByRole('row').nth(2).click();
|
||||||
|
|
||||||
|
// Find the activity state section in the inspector
|
||||||
|
await page.getByRole('tab', { name: 'Activity' }).click();
|
||||||
|
// Check that activity state label is displayed in the inspector.
|
||||||
|
await expect(page.getByLabel('Activity Status').locator("[aria-selected='true']")).toHaveText(
|
||||||
|
'Not started'
|
||||||
|
);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("View a timelist in expanded view, verify all the activities are displayed and selecting an activity shows it's properties", async ({
|
||||||
|
page
|
||||||
|
}) => {
|
||||||
|
// Goto baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
const timelist = await test.step('Create a Time List', async () => {
|
||||||
|
const createdTimeList = await createDomainObjectWithDefaults(page, { type: 'Time List' });
|
||||||
|
const objectName = await page.locator('.l-browse-bar__object-name').innerText();
|
||||||
|
expect(objectName).toBe(createdTimeList.name);
|
||||||
|
|
||||||
|
return createdTimeList;
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Create a Plan and add it to the timelist', async () => {
|
||||||
|
await createPlanFromJSON(page, {
|
||||||
|
name: 'Test Plan',
|
||||||
|
json: examplePlanSmall1,
|
||||||
|
parent: timelist.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
// Ensure that all activities are shown in the expanded view
|
||||||
|
const groups = Object.keys(examplePlanSmall1);
|
||||||
|
const firstGroupKey = groups[0];
|
||||||
|
const firstGroupItems = examplePlanSmall1[firstGroupKey];
|
||||||
|
const firstActivity = firstGroupItems[0];
|
||||||
|
const lastActivity = firstGroupItems[firstGroupItems.length - 1];
|
||||||
|
const startBound = firstActivity.start;
|
||||||
|
const endBound = lastActivity.end;
|
||||||
|
|
||||||
|
// Switch to fixed time mode with all plan events within the bounds
|
||||||
|
await page.goto(
|
||||||
|
`${timelist.url}?tc.mode=fixed&tc.startBound=${startBound}&tc.endBound=${endBound}&tc.timeSystem=utc&view=timelist.view`
|
||||||
|
);
|
||||||
|
|
||||||
|
// Change the object to edit mode
|
||||||
|
await page.getByRole('button', { name: 'Edit Object' }).click();
|
||||||
|
|
||||||
|
// Find the display properties section in the inspector
|
||||||
|
await page.getByRole('tab', { name: 'View Properties' }).click();
|
||||||
|
// Switch to expanded view and save the setting
|
||||||
|
await page.getByLabel('Display Style').selectOption({ label: 'Expanded' });
|
||||||
|
|
||||||
|
// Click on the "Save" button
|
||||||
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
|
// Verify all events are displayed
|
||||||
|
const eventCount = await page.getByRole('row').count();
|
||||||
|
await expect(eventCount).toEqual(firstGroupItems.length);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Shows activity properties when a row is selected', async () => {
|
||||||
|
await page.getByRole('row').nth(2).click();
|
||||||
|
|
||||||
|
// Find the activity state section in the inspector
|
||||||
|
await page.getByRole('tab', { name: 'Activity' }).click();
|
||||||
|
// Check that activity state label is displayed in the inspector.
|
||||||
|
await expect(page.getByLabel('Activity Status').locator("[aria-selected='true']")).toHaveText(
|
||||||
|
'Not started'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The regular expression used to parse the countdown string.
|
||||||
|
* Some examples of valid Countdown strings:
|
||||||
|
* ```
|
||||||
|
* '35D 02:03:04'
|
||||||
|
* '-1D 01:02:03'
|
||||||
|
* '01:02:03'
|
||||||
|
* '-05:06:07'
|
||||||
|
* ```
|
||||||
|
*/
|
||||||
|
const COUNTDOWN_REGEXP = /(-)?(\d+D\s)?(\d{2}):(\d{2}):(\d{2})/;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} CountdownOrUpObject
|
||||||
|
* @property {string} sign - The sign of the countdown ('-' if the countdown is negative, '+' otherwise).
|
||||||
|
* @property {string} days - The number of days in the countdown (undefined if there are no days).
|
||||||
|
* @property {string} hours - The number of hours in the countdown.
|
||||||
|
* @property {string} minutes - The number of minutes in the countdown.
|
||||||
|
* @property {string} seconds - The number of seconds in the countdown.
|
||||||
|
* @property {string} toString - The countdown string.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Object representing the indices of the capture groups in a countdown regex match.
|
||||||
|
*
|
||||||
|
* @typedef {{ SIGN: number, DAYS: number, HOURS: number, MINUTES: number, SECONDS: number, REGEXP: RegExp }}
|
||||||
|
* @property {number} SIGN - The index for the sign capture group (1 if a '-' sign is present, otherwise undefined).
|
||||||
|
* @property {number} DAYS - The index for the days capture group (2 for the number of days, otherwise undefined).
|
||||||
|
* @property {number} HOURS - The index for the hours capture group (3 for the hour part of the time).
|
||||||
|
* @property {number} MINUTES - The index for the minutes capture group (4 for the minute part of the time).
|
||||||
|
* @property {number} SECONDS - The index for the seconds capture group (5 for the second part of the time).
|
||||||
|
*/
|
||||||
|
const COUNTDOWN = Object.freeze({
|
||||||
|
SIGN: 1,
|
||||||
|
DAYS: 2,
|
||||||
|
HOURS: 3,
|
||||||
|
MINUTES: 4,
|
||||||
|
SECONDS: 5
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Time List with controlled clock', () => {
|
||||||
|
test.use({
|
||||||
|
clockOptions: {
|
||||||
|
now: getEarliestStartTime(examplePlanSmall3),
|
||||||
|
shouldAdvanceTime: true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
});
|
||||||
|
test('Time List shows current events and counts down correctly in real-time mode', async ({
|
||||||
|
page
|
||||||
|
}) => {
|
||||||
|
await test.step('Create a Time List, add a Plan to it, and switch to real-time mode', async () => {
|
||||||
|
// Create Time List
|
||||||
|
const timelist = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Time List'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create a Plan with events that count down and up.
|
||||||
|
// Add it as a child to the Time List.
|
||||||
|
await createPlanFromJSON(page, {
|
||||||
|
json: examplePlanSmall3,
|
||||||
|
parent: timelist.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
// Navigate to the Time List in real-time mode
|
||||||
|
await page.goto(
|
||||||
|
`${timelist.url}?tc.mode=local&tc.startDelta=900000&tc.endDelta=1800000&tc.timeSystem=utc&view=grid`
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
const countUpCells = [
|
||||||
|
getCellByIndex(page, 1, TIME_TO_FROM_COLUMN),
|
||||||
|
getCellByIndex(page, 2, TIME_TO_FROM_COLUMN)
|
||||||
|
];
|
||||||
|
const countdownCells = [
|
||||||
|
getCellByIndex(page, 3, TIME_TO_FROM_COLUMN),
|
||||||
|
getCellByIndex(page, 4, TIME_TO_FROM_COLUMN)
|
||||||
|
];
|
||||||
|
|
||||||
|
// Verify that the countdown cells are counting down
|
||||||
|
for (let i = 0; i < countdownCells.length; i++) {
|
||||||
|
await test.step(`Countdown cell ${i + 1} counts down`, async () => {
|
||||||
|
const countdownCell = countdownCells[i];
|
||||||
|
// Get the initial countdown timestamp object
|
||||||
|
const beforeCountdown = await getAndAssertCountdownOrUpObject(page, i + 3);
|
||||||
|
// should not have a '-' sign
|
||||||
|
await expect(countdownCell).not.toHaveText('-');
|
||||||
|
// Wait until it changes
|
||||||
|
await expect(countdownCell).not.toHaveText(beforeCountdown.toString());
|
||||||
|
// Get the new countdown timestamp object
|
||||||
|
const afterCountdown = await getAndAssertCountdownOrUpObject(page, i + 3);
|
||||||
|
// Verify that the new countdown timestamp object is less than the old one
|
||||||
|
expect(Number(afterCountdown.seconds)).toBeLessThan(Number(beforeCountdown.seconds));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Verify that the count-up cells are counting up
|
||||||
|
for (let i = 0; i < countUpCells.length; i++) {
|
||||||
|
await test.step(`Count-up cell ${i + 1} counts up`, async () => {
|
||||||
|
const countUpCell = countUpCells[i];
|
||||||
|
// Get the initial count-up timestamp object
|
||||||
|
const beforeCountUp = await getAndAssertCountdownOrUpObject(page, i + 1);
|
||||||
|
// should not have a '+' sign
|
||||||
|
await expect(countUpCell).not.toHaveText('+');
|
||||||
|
// Wait until it changes
|
||||||
|
await expect(countUpCell).not.toHaveText(beforeCountUp.toString());
|
||||||
|
// Get the new count-up timestamp object
|
||||||
|
const afterCountUp = await getAndAssertCountdownOrUpObject(page, i + 1);
|
||||||
|
// Verify that the new count-up timestamp object is greater than the old one
|
||||||
|
expect(Number(afterCountUp.seconds)).toBeGreaterThan(Number(beforeCountUp.seconds));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the cell at the given row and column indices.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {number} rowIndex
|
||||||
|
* @param {number} columnIndex
|
||||||
|
* @returns {import('@playwright/test').Locator} cell
|
||||||
|
*/
|
||||||
|
function getCellByIndex(page, rowIndex, columnIndex) {
|
||||||
|
return page.getByRole('cell').nth(rowIndex * NUM_COLUMNS + columnIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the innerText of the cell at the given row and column indices.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {number} rowIndex
|
||||||
|
* @param {number} columnIndex
|
||||||
|
* @returns {Promise<string>} text
|
||||||
|
*/
|
||||||
|
async function getCellTextByIndex(page, rowIndex, columnIndex) {
|
||||||
|
const text = await getCellByIndex(page, rowIndex, columnIndex).innerText();
|
||||||
|
return text;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the text from the countdown (or countup) cell in the given row, assert that it matches the countdown/countup
|
||||||
|
* regex, and return an object representing the countdown.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {number} rowIndex the row index
|
||||||
|
* @returns {Promise<CountdownOrUpObject>} The countdown (or countup) object
|
||||||
|
*/
|
||||||
|
async function getAndAssertCountdownOrUpObject(page, rowIndex) {
|
||||||
|
const timeToFrom = await getCellTextByIndex(page, HEADER_ROW + rowIndex, TIME_TO_FROM_COLUMN);
|
||||||
|
|
||||||
|
expect(timeToFrom).toMatch(COUNTDOWN_REGEXP);
|
||||||
|
const match = timeToFrom.match(COUNTDOWN_REGEXP);
|
||||||
|
|
||||||
|
return {
|
||||||
|
sign: match[COUNTDOWN.SIGN],
|
||||||
|
days: match[COUNTDOWN.DAYS],
|
||||||
|
hours: match[COUNTDOWN.HOURS],
|
||||||
|
minutes: match[COUNTDOWN.MINUTES],
|
||||||
|
seconds: match[COUNTDOWN.SECONDS],
|
||||||
|
toString: () => timeToFrom
|
||||||
|
};
|
||||||
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,12 +20,12 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { test, expect } = require('../../../pluginFixtures');
|
import {
|
||||||
const {
|
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
createPlanFromJSON,
|
createPlanFromJSON,
|
||||||
setIndependentTimeConductorBounds
|
setIndependentTimeConductorBounds
|
||||||
} = require('../../../appActions');
|
} from '../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../pluginFixtures.js';
|
||||||
|
|
||||||
const testPlan = {
|
const testPlan = {
|
||||||
TEST_GROUP: [
|
TEST_GROUP: [
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,7 +24,7 @@
|
|||||||
This test suite is dedicated to tests which verify the basic operations surrounding Clock.
|
This test suite is dedicated to tests which verify the basic operations surrounding Clock.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../baseFixtures');
|
import { expect, test } from '../../../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('Clock Generator CRUD Operations', () => {
|
test.describe('Clock Generator CRUD Operations', () => {
|
||||||
test('Timezone dropdown will collapse when clicked outside or on dropdown icon again', async ({
|
test('Timezone dropdown will collapse when clicked outside or on dropdown icon again', async ({
|
||||||
@ -38,7 +38,7 @@ test.describe('Clock Generator CRUD Operations', () => {
|
|||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click Clock
|
// Click Clock
|
||||||
await page.getByRole('menuitem').first().click();
|
await page.getByRole('menuitem').first().click();
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -22,7 +22,7 @@
|
|||||||
|
|
||||||
// FIXME: Remove this eslint exception once tests are implemented
|
// FIXME: Remove this eslint exception once tests are implemented
|
||||||
// eslint-disable-next-line no-unused-vars
|
// eslint-disable-next-line no-unused-vars
|
||||||
const { test, expect } = require('../../../../baseFixtures');
|
import { expect, test } from '../../../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('Remote Clock', () => {
|
test.describe('Remote Clock', () => {
|
||||||
// eslint-disable-next-line require-await
|
// eslint-disable-next-line require-await
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,52 +19,62 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This test suite is dedicated to tests which verify the basic operations surrounding conditionSets. Note: this
|
This test suite is dedicated to tests which verify the basic operations surrounding conditionSets. Note: this
|
||||||
suite is sharing state between tests which is considered an anti-pattern. Implementing in this way to
|
suite is sharing state between tests which is considered an anti-pattern. Implementing in this way to
|
||||||
demonstrate some playwright for test developers. This pattern should not be re-used in other CRUD suites.
|
demonstrate some playwright for test developers. This pattern should not be re-used in other CRUD suites.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures.js');
|
import { fileURLToPath } from 'url';
|
||||||
const {
|
|
||||||
|
import {
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
createExampleTelemetryObject
|
createExampleTelemetryObject
|
||||||
} = require('../../../../appActions');
|
} from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
let conditionSetUrl;
|
let conditionSetUrl;
|
||||||
let getConditionSetIdentifierFromUrl;
|
|
||||||
|
|
||||||
test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
test.describe.serial('Condition Set CRUD Operations on @localStorage @2p', () => {
|
||||||
test.beforeAll(async ({ browser }) => {
|
test.beforeAll(async ({ browser }) => {
|
||||||
//TODO: This needs to be refactored
|
//TODO: This needs to be refactored
|
||||||
const context = await browser.newContext();
|
const context = await browser.newContext();
|
||||||
const page = await context.newPage();
|
const page = await context.newPage();
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
await page.locator('li[role="menuitem"]:has-text("Condition Set")').click();
|
await page.locator('li[role="menuitem"]:has-text("Condition Set")').click();
|
||||||
|
|
||||||
await Promise.all([page.waitForNavigation(), page.click('button:has-text("OK")')]);
|
await Promise.all([page.waitForNavigation(), page.click('button:has-text("OK")')]);
|
||||||
|
|
||||||
//Save localStorage for future test execution
|
//Save localStorage for future test execution
|
||||||
await context.storageState({ path: './e2e/test-data/recycled_local_storage.json' });
|
await context.storageState({
|
||||||
|
path: fileURLToPath(
|
||||||
|
new URL('../../../../test-data/recycled_local_storage.json', import.meta.url)
|
||||||
|
)
|
||||||
|
});
|
||||||
|
|
||||||
//Set object identifier from url
|
//Set object identifier from url
|
||||||
conditionSetUrl = page.url();
|
conditionSetUrl = page.url();
|
||||||
|
|
||||||
getConditionSetIdentifierFromUrl = conditionSetUrl.split('/').pop().split('?')[0];
|
|
||||||
console.debug(`getConditionSetIdentifierFromUrl: ${getConditionSetIdentifierFromUrl}`);
|
|
||||||
await page.close();
|
await page.close();
|
||||||
});
|
});
|
||||||
|
|
||||||
//Load localStorage for subsequent tests
|
//Load localStorage for subsequent tests
|
||||||
test.use({ storageState: './e2e/test-data/recycled_local_storage.json' });
|
test.use({
|
||||||
|
storageState: fileURLToPath(
|
||||||
|
new URL('../../../../test-data/recycled_local_storage.json', import.meta.url)
|
||||||
|
)
|
||||||
|
});
|
||||||
|
|
||||||
//Begin suite of tests again localStorage
|
//Begin suite of tests again localStorage
|
||||||
test('Condition set object properties persist in main view and inspector @localStorage', async ({
|
test.fixme(
|
||||||
page
|
'Condition set object properties persist in main view and inspector @localStorage',
|
||||||
}) => {
|
async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/7421'
|
||||||
|
});
|
||||||
//Navigate to baseURL with injected localStorage
|
//Navigate to baseURL with injected localStorage
|
||||||
await page.goto(conditionSetUrl, { waitUntil: 'networkidle' });
|
await page.goto(conditionSetUrl, { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
@ -85,7 +95,8 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
.toContainText('Unnamed Condition Set');
|
.toContainText('Unnamed Condition Set');
|
||||||
//Assertions on loaded Condition Set in Inspector
|
//Assertions on loaded Condition Set in Inspector
|
||||||
expect.soft(page.locator('_vue=item.name=Unnamed Condition Set')).toBeTruthy();
|
expect.soft(page.locator('_vue=item.name=Unnamed Condition Set')).toBeTruthy();
|
||||||
});
|
}
|
||||||
|
);
|
||||||
test('condition set object can be modified on @localStorage', async ({ page, openmctConfig }) => {
|
test('condition set object can be modified on @localStorage', async ({ page, openmctConfig }) => {
|
||||||
const { myItemsFolderName } = openmctConfig;
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
@ -117,7 +128,7 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
.nth(1)
|
.nth(1)
|
||||||
.click();
|
.click();
|
||||||
// Click Save and Finish Editing Option
|
// Click Save and Finish Editing Option
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
//Verify Main section reflects updated Name Property
|
//Verify Main section reflects updated Name Property
|
||||||
await expect
|
await expect
|
||||||
@ -187,7 +198,7 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
.first()
|
.first()
|
||||||
.click();
|
.click();
|
||||||
// Click hamburger button
|
// Click hamburger button
|
||||||
await page.locator('[title="More options"]').click();
|
await page.locator('[title="More actions"]').click();
|
||||||
|
|
||||||
// Click 'Remove' and press OK
|
// Click 'Remove' and press OK
|
||||||
await page.locator('li[role="menuitem"]:has-text("Remove")').click();
|
await page.locator('li[role="menuitem"]:has-text("Remove")').click();
|
||||||
@ -225,7 +236,7 @@ test.describe('Basic Condition Set Use', () => {
|
|||||||
await page.goto(conditionSet.url);
|
await page.goto(conditionSet.url);
|
||||||
|
|
||||||
// Change the object to edit mode
|
// Change the object to edit mode
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Click Add Condition button
|
// Click Add Condition button
|
||||||
await page.locator('#addCondition').click();
|
await page.locator('#addCondition').click();
|
||||||
@ -253,7 +264,7 @@ test.describe('Basic Condition Set Use', () => {
|
|||||||
await page.goto(conditionSet.url);
|
await page.goto(conditionSet.url);
|
||||||
|
|
||||||
// Change the object to edit mode
|
// Change the object to edit mode
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
page.click('button[title="Show selected item in tree"]');
|
page.click('button[title="Show selected item in tree"]');
|
||||||
@ -290,7 +301,7 @@ test.describe('Basic Condition Set Use', () => {
|
|||||||
await page.getByTitle('Show selected item in tree').click();
|
await page.getByTitle('Show selected item in tree').click();
|
||||||
await page.goto(conditionSet.url);
|
await page.goto(conditionSet.url);
|
||||||
// Change the object to edit mode
|
// Change the object to edit mode
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Create two conditions
|
// Create two conditions
|
||||||
await page.locator('#addCondition').click();
|
await page.locator('#addCondition').click();
|
||||||
@ -357,7 +368,7 @@ test.describe('Basic Condition Set Use', () => {
|
|||||||
|
|
||||||
// Edit SWG to add 8 second loading delay to simulate the case
|
// Edit SWG to add 8 second loading delay to simulate the case
|
||||||
// where telemetry is not available.
|
// where telemetry is not available.
|
||||||
await page.getByTitle('More options').click();
|
await page.getByTitle('More actions').click();
|
||||||
await page.getByRole('menuitem', { name: 'Edit Properties...' }).click();
|
await page.getByRole('menuitem', { name: 'Edit Properties...' }).click();
|
||||||
await page.getByRole('spinbutton', { name: 'Loading Delay (ms)' }).fill('8000');
|
await page.getByRole('spinbutton', { name: 'Loading Delay (ms)' }).fill('8000');
|
||||||
await page.getByLabel('Save').click();
|
await page.getByLabel('Save').click();
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,15 +19,97 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
import { fileURLToPath } from 'url';
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
import {
|
||||||
const {
|
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
setStartOffset,
|
|
||||||
setFixedTimeMode,
|
setFixedTimeMode,
|
||||||
|
setIndependentTimeConductorBounds,
|
||||||
setRealTimeMode,
|
setRealTimeMode,
|
||||||
setIndependentTimeConductorBounds
|
setStartOffset
|
||||||
} = require('../../../../appActions');
|
} from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
|
const LOCALSTORAGE_PATH = fileURLToPath(
|
||||||
|
new URL('../../../../test-data/display_layout_with_child_layouts.json', import.meta.url)
|
||||||
|
);
|
||||||
|
const TINY_IMAGE_BASE64 =
|
||||||
|
'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAgAAAAIAQMAAAD+wSzIAAAABlBMVEX///+/v7+jQ3Y5AAAADklEQVQI12P4AIX8EAgALgAD/aNpbtEAAAAASUVORK5CYII';
|
||||||
|
|
||||||
|
test.describe('Display Layout Toolbar Actions @localStorage', () => {
|
||||||
|
const PARENT_DISPLAY_LAYOUT_NAME = 'Parent Display Layout';
|
||||||
|
const CHILD_DISPLAY_LAYOUT_NAME1 = 'Child Layout 1';
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
await page
|
||||||
|
.locator('a')
|
||||||
|
.filter({ hasText: 'Parent Display Layout Display Layout' })
|
||||||
|
.first()
|
||||||
|
.click();
|
||||||
|
await page.getByLabel('Edit Object').click();
|
||||||
|
});
|
||||||
|
test.use({
|
||||||
|
storageState: LOCALSTORAGE_PATH
|
||||||
|
});
|
||||||
|
|
||||||
|
test('can add/remove Text element to a single layout', async ({ page }) => {
|
||||||
|
const layoutObject = 'Text';
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the parent's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, PARENT_DISPLAY_LAYOUT_NAME);
|
||||||
|
});
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the child's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, CHILD_DISPLAY_LAYOUT_NAME1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('can add/remove Image to a single layout', async ({ page }) => {
|
||||||
|
const layoutObject = 'Image';
|
||||||
|
await test.step("Add and remove image element from the parent's layout", async () => {
|
||||||
|
expect(await page.getByLabel(`Move ${layoutObject} Frame`).count()).toBe(0);
|
||||||
|
await addLayoutObject(page, PARENT_DISPLAY_LAYOUT_NAME, layoutObject);
|
||||||
|
expect(await page.getByLabel(`Move ${layoutObject} Frame`).count()).toBe(1);
|
||||||
|
await removeLayoutObject(page, layoutObject);
|
||||||
|
expect(await page.getByLabel(`Move ${layoutObject} Frame`).count()).toBe(0);
|
||||||
|
});
|
||||||
|
await test.step("Add and remove image from the child's layout", async () => {
|
||||||
|
await addLayoutObject(page, CHILD_DISPLAY_LAYOUT_NAME1, layoutObject);
|
||||||
|
expect(await page.getByLabel(`Move ${layoutObject} Frame`).count()).toBe(1);
|
||||||
|
await removeLayoutObject(page, layoutObject);
|
||||||
|
expect(await page.getByLabel(`Move ${layoutObject} Frame`).count()).toBe(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test(`can add/remove Box to a single layout`, async ({ page }) => {
|
||||||
|
const layoutObject = 'Box';
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the parent's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, PARENT_DISPLAY_LAYOUT_NAME);
|
||||||
|
});
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the child's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, CHILD_DISPLAY_LAYOUT_NAME1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test(`can add/remove Line to a single layout`, async ({ page }) => {
|
||||||
|
const layoutObject = 'Line';
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the parent's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, PARENT_DISPLAY_LAYOUT_NAME);
|
||||||
|
});
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the child's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, CHILD_DISPLAY_LAYOUT_NAME1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test(`can add/remove Ellipse to a single layout`, async ({ page }) => {
|
||||||
|
const layoutObject = 'Ellipse';
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the parent's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, PARENT_DISPLAY_LAYOUT_NAME);
|
||||||
|
});
|
||||||
|
await test.step(`Add and remove ${layoutObject} from the child's layout`, async () => {
|
||||||
|
await addAndRemoveDrawingObjectAndAssert(page, layoutObject, CHILD_DISPLAY_LAYOUT_NAME1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test.fixme('Can switch view types of a single SWG in a layout', async ({ page }) => {});
|
||||||
|
test.fixme('Can merge multiple plots in a layout', async ({ page }) => {});
|
||||||
|
test.fixme('Can adjust stack order of a single object in a layout', async ({ page }) => {});
|
||||||
|
test.fixme('Can duplicate a single object in a layout', async ({ page }) => {});
|
||||||
|
});
|
||||||
|
|
||||||
test.describe('Display Layout', () => {
|
test.describe('Display Layout', () => {
|
||||||
/** @type {import('../../../../appActions').CreatedObjectInfo} */
|
/** @type {import('../../../../appActions').CreatedObjectInfo} */
|
||||||
@ -41,6 +123,7 @@ test.describe('Display Layout', () => {
|
|||||||
type: 'Sine Wave Generator'
|
type: 'Sine Wave Generator'
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test('alpha-numeric widget telemetry value exactly matches latest telemetry value received in real time', async ({
|
test('alpha-numeric widget telemetry value exactly matches latest telemetry value received in real time', async ({
|
||||||
page
|
page
|
||||||
}) => {
|
}) => {
|
||||||
@ -50,7 +133,7 @@ test.describe('Display Layout', () => {
|
|||||||
name: 'Test Display Layout'
|
name: 'Test Display Layout'
|
||||||
});
|
});
|
||||||
// Edit Display Layout
|
// Edit Display Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -64,7 +147,7 @@ test.describe('Display Layout', () => {
|
|||||||
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
||||||
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Subscribe to the Sine Wave Generator data
|
// Subscribe to the Sine Wave Generator data
|
||||||
// On getting data, check if the value found in the Display Layout is the most recent value
|
// On getting data, check if the value found in the Display Layout is the most recent value
|
||||||
@ -78,6 +161,13 @@ test.describe('Display Layout', () => {
|
|||||||
const trimmedDisplayValue = displayLayoutValue.trim();
|
const trimmedDisplayValue = displayLayoutValue.trim();
|
||||||
|
|
||||||
expect(trimmedDisplayValue).toBe(formattedTelemetryValue);
|
expect(trimmedDisplayValue).toBe(formattedTelemetryValue);
|
||||||
|
|
||||||
|
// ensure we can right click on the alpha-numeric widget and view historical data
|
||||||
|
await page.getByLabel('Sine', { exact: true }).click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
await page.getByLabel('View Historical Data').click();
|
||||||
|
await expect(page.getByLabel('Plot Container Style Target')).toBeVisible();
|
||||||
});
|
});
|
||||||
test('alpha-numeric widget telemetry value exactly matches latest telemetry value received in fixed time', async ({
|
test('alpha-numeric widget telemetry value exactly matches latest telemetry value received in fixed time', async ({
|
||||||
page
|
page
|
||||||
@ -88,7 +178,7 @@ test.describe('Display Layout', () => {
|
|||||||
name: 'Test Display Layout'
|
name: 'Test Display Layout'
|
||||||
});
|
});
|
||||||
// Edit Display Layout
|
// Edit Display Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -102,7 +192,7 @@ test.describe('Display Layout', () => {
|
|||||||
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
||||||
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Subscribe to the Sine Wave Generator data
|
// Subscribe to the Sine Wave Generator data
|
||||||
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
||||||
@ -130,7 +220,7 @@ test.describe('Display Layout', () => {
|
|||||||
name: 'Test Display Layout'
|
name: 'Test Display Layout'
|
||||||
});
|
});
|
||||||
// Edit Display Layout
|
// Edit Display Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -144,7 +234,7 @@ test.describe('Display Layout', () => {
|
|||||||
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
||||||
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(1);
|
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(1);
|
||||||
|
|
||||||
@ -172,7 +262,7 @@ test.describe('Display Layout', () => {
|
|||||||
type: 'Display Layout'
|
type: 'Display Layout'
|
||||||
});
|
});
|
||||||
// Edit Display Layout
|
// Edit Display Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -186,7 +276,7 @@ test.describe('Display Layout', () => {
|
|||||||
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
const layoutGridHolder = page.locator('.l-layout__grid-holder');
|
||||||
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
await sineWaveGeneratorTreeItem.dragTo(layoutGridHolder);
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(1);
|
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(1);
|
||||||
|
|
||||||
@ -218,7 +308,7 @@ test.describe('Display Layout', () => {
|
|||||||
type: 'Display Layout'
|
type: 'Display Layout'
|
||||||
});
|
});
|
||||||
// Edit Display Layout
|
// Edit Display Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -242,7 +332,7 @@ test.describe('Display Layout', () => {
|
|||||||
await page.locator('div[title="Resize object width"] > input').fill('70');
|
await page.locator('div[title="Resize object width"] > input').fill('70');
|
||||||
|
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
const startDate = '2021-12-30 01:01:00.000Z';
|
const startDate = '2021-12-30 01:01:00.000Z';
|
||||||
const endDate = '2021-12-30 01:11:00.000Z';
|
const endDate = '2021-12-30 01:11:00.000Z';
|
||||||
@ -274,7 +364,7 @@ test.describe('Display Layout', () => {
|
|||||||
name: 'Test Display Layout'
|
name: 'Test Display Layout'
|
||||||
});
|
});
|
||||||
// Edit Display Layout
|
// Edit Display Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -304,7 +394,7 @@ test.describe('Display Layout', () => {
|
|||||||
await page.getByText('Overlay Plot').click();
|
await page.getByText('Overlay Plot').click();
|
||||||
|
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Time to inspect some network traffic
|
// Time to inspect some network traffic
|
||||||
let networkRequests = [];
|
let networkRequests = [];
|
||||||
@ -339,6 +429,59 @@ test.describe('Display Layout', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
async function addAndRemoveDrawingObjectAndAssert(page, layoutObject, DISPLAY_LAYOUT_NAME) {
|
||||||
|
expect(await page.getByLabel(layoutObject, { exact: true }).count()).toBe(0);
|
||||||
|
await addLayoutObject(page, DISPLAY_LAYOUT_NAME, layoutObject);
|
||||||
|
expect(
|
||||||
|
await page
|
||||||
|
.getByLabel(layoutObject, {
|
||||||
|
exact: true
|
||||||
|
})
|
||||||
|
.count()
|
||||||
|
).toBe(1);
|
||||||
|
await removeLayoutObject(page, layoutObject);
|
||||||
|
expect(await page.getByLabel(layoutObject, { exact: true }).count()).toBe(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove the first matching layout object from the layout
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {'Box' | 'Ellipse' | 'Line' | 'Text' | 'Image'} layoutObject
|
||||||
|
*/
|
||||||
|
async function removeLayoutObject(page, layoutObject) {
|
||||||
|
await page
|
||||||
|
.getByLabel(`Move ${layoutObject} Frame`, { exact: true })
|
||||||
|
.or(page.getByLabel(layoutObject, { exact: true }))
|
||||||
|
.first()
|
||||||
|
// eslint-disable-next-line playwright/no-force-option
|
||||||
|
.click({ force: true });
|
||||||
|
await page.getByTitle('Delete the selected object').click();
|
||||||
|
await page.getByRole('button', { name: 'OK', exact: true }).click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add a layout object to the specified layout
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} layoutName
|
||||||
|
* @param {'Box' | 'Ellipse' | 'Line' | 'Text' | 'Image'} layoutObject
|
||||||
|
*/
|
||||||
|
async function addLayoutObject(page, layoutName, layoutObject) {
|
||||||
|
await page.getByLabel(`${layoutName} Layout`, { exact: true }).click();
|
||||||
|
await page.getByText('Add Drawing Object').click();
|
||||||
|
await page
|
||||||
|
.getByRole('menuitem', {
|
||||||
|
name: layoutObject
|
||||||
|
})
|
||||||
|
.click();
|
||||||
|
if (layoutObject === 'Text') {
|
||||||
|
await page.getByRole('textbox', { name: 'Text' }).fill('Hello, Universe!');
|
||||||
|
await page.getByText('OK').click();
|
||||||
|
} else if (layoutObject === 'Image') {
|
||||||
|
await page.getByLabel('Image URL').fill(TINY_IMAGE_BASE64);
|
||||||
|
await page.getByText('OK').click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Util for subscribing to a telemetry object by object identifier
|
* Util for subscribing to a telemetry object by object identifier
|
||||||
* Limitations: Currently only works to return telemetry once to the node scope
|
* Limitations: Currently only works to return telemetry once to the node scope
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,8 +20,8 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
import * as utils from '../../../../helper/faultUtils.js';
|
||||||
const utils = require('../../../../helper/faultUtils');
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('The Fault Management Plugin using example faults', () => {
|
test.describe('The Fault Management Plugin using example faults', () => {
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,11 +20,17 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
import { fileURLToPath } from 'url';
|
||||||
const {
|
|
||||||
|
import {
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
setIndependentTimeConductorBounds
|
setIndependentTimeConductorBounds
|
||||||
} = require('../../../../appActions');
|
} from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
|
const LOCALSTORAGE_PATH = fileURLToPath(
|
||||||
|
new URL('../../../../test-data/flexible_layout_with_child_layouts.json', import.meta.url)
|
||||||
|
);
|
||||||
|
|
||||||
test.describe('Flexible Layout', () => {
|
test.describe('Flexible Layout', () => {
|
||||||
let sineWaveObject;
|
let sineWaveObject;
|
||||||
@ -67,7 +73,7 @@ test.describe('Flexible Layout', () => {
|
|||||||
}) => {
|
}) => {
|
||||||
await page.goto(flexibleLayout.url);
|
await page.goto(flexibleLayout.url);
|
||||||
// Edit Flexible Layout
|
// Edit Flexible Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').first().click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').first().click();
|
||||||
@ -81,7 +87,7 @@ test.describe('Flexible Layout', () => {
|
|||||||
await expect(dragWrapper).toHaveAttribute('draggable', 'true');
|
await expect(dragWrapper).toHaveAttribute('draggable', 'true');
|
||||||
// Save Flexible Layout
|
// Save Flexible Layout
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
// Check that panes are not draggable while Flexible Layout is in Browse mode
|
// Check that panes are not draggable while Flexible Layout is in Browse mode
|
||||||
dragWrapper = page.locator('.c-fl-container__frames-holder .c-fl-frame__drag-wrapper').first();
|
dragWrapper = page.locator('.c-fl-container__frames-holder .c-fl-frame__drag-wrapper').first();
|
||||||
await expect(dragWrapper).toHaveAttribute('draggable', 'false');
|
await expect(dragWrapper).toHaveAttribute('draggable', 'false');
|
||||||
@ -160,14 +166,14 @@ test.describe('Flexible Layout', () => {
|
|||||||
}) => {
|
}) => {
|
||||||
await page.goto(flexibleLayout.url);
|
await page.goto(flexibleLayout.url);
|
||||||
// Edit Flexible Layout
|
// Edit Flexible Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').first().click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').first().click();
|
||||||
// Add the Sine Wave Generator to the Flexible Layout and save changes
|
// Add the Sine Wave Generator to the Flexible Layout and save changes
|
||||||
await sineWaveGeneratorTreeItem.dragTo(page.locator('.c-fl__container.is-empty').first());
|
await sineWaveGeneratorTreeItem.dragTo(page.locator('.c-fl__container.is-empty').first());
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
expect.soft(await page.locator('.c-fl-container__frame').count()).toEqual(1);
|
expect.soft(await page.locator('.c-fl-container__frame').count()).toEqual(1);
|
||||||
|
|
||||||
@ -191,14 +197,14 @@ test.describe('Flexible Layout', () => {
|
|||||||
});
|
});
|
||||||
await page.goto(flexibleLayout.url);
|
await page.goto(flexibleLayout.url);
|
||||||
// Edit Flexible Layout
|
// Edit Flexible Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
// Add the Sine Wave Generator to the Flexible Layout and save changes
|
// Add the Sine Wave Generator to the Flexible Layout and save changes
|
||||||
await sineWaveGeneratorTreeItem.dragTo(page.locator('.c-fl__container.is-empty').first());
|
await sineWaveGeneratorTreeItem.dragTo(page.locator('.c-fl__container.is-empty').first());
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
expect.soft(await page.locator('.c-fl-container__frame').count()).toEqual(1);
|
expect.soft(await page.locator('.c-fl-container__frame').count()).toEqual(1);
|
||||||
|
|
||||||
@ -228,7 +234,7 @@ test.describe('Flexible Layout', () => {
|
|||||||
|
|
||||||
await page.goto(flexibleLayout.url);
|
await page.goto(flexibleLayout.url);
|
||||||
// Edit Flexible Layout
|
// Edit Flexible Layout
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
@ -239,7 +245,7 @@ test.describe('Flexible Layout', () => {
|
|||||||
await exampleImageryTreeItem.dragTo(page.locator('.c-fl__container.is-empty').first());
|
await exampleImageryTreeItem.dragTo(page.locator('.c-fl__container.is-empty').first());
|
||||||
|
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// flip on independent time conductor
|
// flip on independent time conductor
|
||||||
await setIndependentTimeConductorBounds(
|
await setIndependentTimeConductorBounds(
|
||||||
@ -257,3 +263,56 @@ test.describe('Flexible Layout', () => {
|
|||||||
await expect(page.getByText('2021-12-30 01:11:00.000Z')).toBeHidden();
|
await expect(page.getByText('2021-12-30 01:11:00.000Z')).toBeHidden();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test.describe('Flexible Layout Toolbar Actions @localStorage', () => {
|
||||||
|
test.use({
|
||||||
|
storageState: LOCALSTORAGE_PATH
|
||||||
|
});
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
await page
|
||||||
|
.locator('a')
|
||||||
|
.filter({ hasText: 'Parent Flexible Layout Flexible Layout' })
|
||||||
|
.first()
|
||||||
|
.click();
|
||||||
|
await page.getByLabel('Edit Object').click();
|
||||||
|
});
|
||||||
|
test('Add/Remove Container', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/7234'
|
||||||
|
});
|
||||||
|
|
||||||
|
const containerHandles = page.getByRole('columnheader', { name: 'Handle' });
|
||||||
|
expect(await containerHandles.count()).toEqual(2);
|
||||||
|
await page.getByRole('columnheader', { name: 'Container Handle 1' }).click();
|
||||||
|
await page.getByTitle('Add Container').click();
|
||||||
|
expect(await containerHandles.count()).toEqual(3);
|
||||||
|
await page.getByTitle('Remove Container').click();
|
||||||
|
await expect(page.getByRole('dialog', { name: 'Overlay' })).toHaveText(
|
||||||
|
'This action will permanently delete this container from this Flexible Layout. Do you want to continue?'
|
||||||
|
);
|
||||||
|
await page.getByRole('button', { name: 'OK', exact: true }).click();
|
||||||
|
expect(await containerHandles.count()).toEqual(2);
|
||||||
|
});
|
||||||
|
test('Remove Frame', async ({ page }) => {
|
||||||
|
expect(await page.getByRole('group', { name: 'Frame' }).count()).toEqual(2);
|
||||||
|
await page.getByRole('group', { name: 'Child Layout 1' }).click();
|
||||||
|
await page.getByTitle('Remove Frame').click();
|
||||||
|
await expect(page.getByRole('dialog', { name: 'Overlay' })).toHaveText(
|
||||||
|
'This action will remove this frame from this Flexible Layout. Do you want to continue?'
|
||||||
|
);
|
||||||
|
await page.getByRole('button', { name: 'OK', exact: true }).click();
|
||||||
|
expect(await page.getByRole('group', { name: 'Frame' }).count()).toEqual(1);
|
||||||
|
});
|
||||||
|
test('Columns/Rows Layout Toggle', async ({ page }) => {
|
||||||
|
await page.getByRole('columnheader', { name: 'Container Handle 1' }).click();
|
||||||
|
const flexRows = page.getByLabel('Flexible Layout Row');
|
||||||
|
expect(await flexRows.count()).toEqual(0);
|
||||||
|
await page.getByTitle('Columns layout').click();
|
||||||
|
expect(await flexRows.count()).toEqual(1);
|
||||||
|
await page.getByTitle('Rows layout').click();
|
||||||
|
expect(await flexRows.count()).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,9 +24,13 @@
|
|||||||
* This test suite is dedicated to testing the Gauge component.
|
* This test suite is dedicated to testing the Gauge component.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
import { v4 as uuid } from 'uuid';
|
||||||
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
|
||||||
const uuid = require('uuid').v4;
|
import {
|
||||||
|
createDomainObjectWithDefaults,
|
||||||
|
createExampleTelemetryObject
|
||||||
|
} from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('Gauge', () => {
|
test.describe('Gauge', () => {
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
@ -37,8 +41,6 @@ test.describe('Gauge', () => {
|
|||||||
test('Can add and remove telemetry sources @unstable', async ({ page }) => {
|
test('Can add and remove telemetry sources @unstable', async ({ page }) => {
|
||||||
// Create the gauge with defaults
|
// Create the gauge with defaults
|
||||||
const gauge = await createDomainObjectWithDefaults(page, { type: 'Gauge' });
|
const gauge = await createDomainObjectWithDefaults(page, { type: 'Gauge' });
|
||||||
const editButtonLocator = page.locator('button[title="Edit"]');
|
|
||||||
const saveButtonLocator = page.locator('button[title="Save"]');
|
|
||||||
|
|
||||||
// Create a sine wave generator within the gauge
|
// Create a sine wave generator within the gauge
|
||||||
const swg1 = await createDomainObjectWithDefaults(page, {
|
const swg1 = await createDomainObjectWithDefaults(page, {
|
||||||
@ -50,10 +52,10 @@ test.describe('Gauge', () => {
|
|||||||
// Navigate to the gauge and verify that
|
// Navigate to the gauge and verify that
|
||||||
// the SWG appears in the elements pool
|
// the SWG appears in the elements pool
|
||||||
await page.goto(gauge.url);
|
await page.goto(gauge.url);
|
||||||
await editButtonLocator.click();
|
await page.getByLabel('Edit Object').click();
|
||||||
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeVisible();
|
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeVisible();
|
||||||
await saveButtonLocator.click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
await page.locator('li[title="Save and Finish Editing"]').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Create another sine wave generator within the gauge
|
// Create another sine wave generator within the gauge
|
||||||
const swg2 = await createDomainObjectWithDefaults(page, {
|
const swg2 = await createDomainObjectWithDefaults(page, {
|
||||||
@ -75,10 +77,10 @@ test.describe('Gauge', () => {
|
|||||||
// Navigate to the gauge and verify that the new SWG
|
// Navigate to the gauge and verify that the new SWG
|
||||||
// appears in the elements pool and the old one is gone
|
// appears in the elements pool and the old one is gone
|
||||||
await page.goto(gauge.url);
|
await page.goto(gauge.url);
|
||||||
await editButtonLocator.click();
|
await page.getByLabel('Edit Object').click();
|
||||||
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeHidden();
|
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg1.name}`)).toBeHidden();
|
||||||
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg2.name}`)).toBeVisible();
|
await expect.soft(page.locator(`#inspector-elements-tree >> text=${swg2.name}`)).toBeVisible();
|
||||||
await saveButtonLocator.click();
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|
||||||
// Right click on the new SWG in the elements pool and delete it
|
// Right click on the new SWG in the elements pool and delete it
|
||||||
await page.locator(`#inspector-elements-tree >> text=${swg2.name}`).click({
|
await page.locator(`#inspector-elements-tree >> text=${swg2.name}`).click({
|
||||||
@ -105,7 +107,7 @@ test.describe('Gauge', () => {
|
|||||||
description: 'https://github.com/nasa/openmct/issues/5356'
|
description: 'https://github.com/nasa/openmct/issues/5356'
|
||||||
});
|
});
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click the object specified by 'type'
|
// Click the object specified by 'type'
|
||||||
await page.click(`li[role='menuitem']:text("Gauge")`);
|
await page.click(`li[role='menuitem']:text("Gauge")`);
|
||||||
@ -124,7 +126,7 @@ test.describe('Gauge', () => {
|
|||||||
|
|
||||||
// Create the gauge with defaults
|
// Create the gauge with defaults
|
||||||
await createDomainObjectWithDefaults(page, { type: 'Gauge' });
|
await createDomainObjectWithDefaults(page, { type: 'Gauge' });
|
||||||
await page.click('button[title="More options"]');
|
await page.click('button[title="More actions"]');
|
||||||
await page.click('li[role="menuitem"]:has-text("Edit Properties")');
|
await page.click('li[role="menuitem"]:has-text("Edit Properties")');
|
||||||
// FIXME: We need better selectors for these custom form controls
|
// FIXME: We need better selectors for these custom form controls
|
||||||
const displayCurrentValueSwitch = page.locator('.c-toggle-switch__slider >> nth=0');
|
const displayCurrentValueSwitch = page.locator('.c-toggle-switch__slider >> nth=0');
|
||||||
@ -133,4 +135,54 @@ test.describe('Gauge', () => {
|
|||||||
|
|
||||||
// TODO: Verify changes in the UI
|
// TODO: Verify changes in the UI
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test.fixme('Gauge does not display NaN when data not available', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/7421'
|
||||||
|
});
|
||||||
|
// Create a Gauge
|
||||||
|
const gauge = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Gauge'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create a Sine Wave Generator in the Gauge with a loading delay
|
||||||
|
const swgWith5sDelay = await createExampleTelemetryObject(page, gauge.uuid);
|
||||||
|
|
||||||
|
await page.goto(swgWith5sDelay.url);
|
||||||
|
await page.getByTitle('More actions').click();
|
||||||
|
await page.getByRole('menuitem', { name: /Edit Properties.../ }).click();
|
||||||
|
|
||||||
|
//Edit Example Telemetry Object to include 5s loading Delay
|
||||||
|
await page.locator('[aria-label="Loading Delay \\(ms\\)"]').fill('5000');
|
||||||
|
|
||||||
|
await page.getByRole('button', { name: 'Save' }).click();
|
||||||
|
|
||||||
|
// Wait until the URL is updated
|
||||||
|
await page.waitForURL(`**/${gauge.uuid}/*`);
|
||||||
|
|
||||||
|
// Nav to the Gauge
|
||||||
|
await page.goto(gauge.url);
|
||||||
|
const gaugeNoDataText = await page.locator('.js-dial-current-value tspan').textContent();
|
||||||
|
expect(gaugeNoDataText).toBe('--');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Gauge enforces composition policy', async ({ page }) => {
|
||||||
|
// Create a Gauge
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Gauge',
|
||||||
|
name: 'Unnamed Gauge'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Try to create a Folder into the Gauge. Should be disallowed.
|
||||||
|
await page.getByRole('button', { name: /Create/ }).click();
|
||||||
|
await page.getByRole('menuitem', { name: /Folder/ }).click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.getByLabel('Cancel').click();
|
||||||
|
|
||||||
|
// Try to create a Display Layout into the Gauge. Should be disallowed.
|
||||||
|
await page.getByRole('button', { name: /Create/ }).click();
|
||||||
|
await page.getByRole('menuitem', { name: /Display Layout/ }).click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -25,9 +25,9 @@ This test suite is dedicated to tests which verify the basic operations surround
|
|||||||
but only assume that example imagery is present.
|
but only assume that example imagery is present.
|
||||||
*/
|
*/
|
||||||
/* globals process */
|
/* globals process */
|
||||||
const { waitForAnimations } = require('../../../../baseFixtures');
|
import { createDomainObjectWithDefaults, setRealTimeMode } from '../../../../appActions.js';
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
import { waitForAnimations } from '../../../../baseFixtures.js';
|
||||||
const { createDomainObjectWithDefaults, setRealTimeMode } = require('../../../../appActions');
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
||||||
const panHotkey = process.platform === 'linux' ? ['Shift', 'Alt'] : ['Alt'];
|
const panHotkey = process.platform === 'linux' ? ['Shift', 'Alt'] : ['Alt'];
|
||||||
const tagHotkey = ['Shift', 'Alt'];
|
const tagHotkey = ['Shift', 'Alt'];
|
||||||
@ -61,6 +61,31 @@ test.describe('Example Imagery Object', () => {
|
|||||||
await expect(page.locator('.c-hud')).toBeHidden();
|
await expect(page.locator('.c-hud')).toBeHidden();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('Can right click on image and open it in a new tab @2p', async ({ page, context }) => {
|
||||||
|
// try to right click on image
|
||||||
|
const backgroundImage = await page.locator(backgroundImageSelector);
|
||||||
|
await backgroundImage.click({
|
||||||
|
button: 'right',
|
||||||
|
// eslint-disable-next-line playwright/no-force-option
|
||||||
|
force: true
|
||||||
|
});
|
||||||
|
// expect context menu to appear
|
||||||
|
await expect(page.getByText('Save Image As')).toBeVisible();
|
||||||
|
await expect(page.getByText('Open Image in New Tab')).toBeVisible();
|
||||||
|
|
||||||
|
// click on open image in new tab
|
||||||
|
const pagePromise = context.waitForEvent('page');
|
||||||
|
await page.getByText('Open Image in New Tab').click();
|
||||||
|
// expect new tab to be in browser
|
||||||
|
const newPage = await pagePromise;
|
||||||
|
await newPage.waitForLoadState();
|
||||||
|
// expect new tab url to have jpg in it
|
||||||
|
await expect(newPage.url()).toContain('.jpg');
|
||||||
|
});
|
||||||
|
|
||||||
|
// this requires CORS to be enabled in some fashion
|
||||||
|
test.fixme('Can right click on image and save it as a file', async ({ page }) => {});
|
||||||
|
|
||||||
test('Can adjust image brightness/contrast by dragging the sliders', async ({
|
test('Can adjust image brightness/contrast by dragging the sliders', async ({
|
||||||
page,
|
page,
|
||||||
browserName
|
browserName
|
||||||
@ -247,6 +272,14 @@ test.describe('Example Imagery Object', () => {
|
|||||||
await page.mouse.click(canvasCenterX - 50, canvasCenterY - 50);
|
await page.mouse.click(canvasCenterX - 50, canvasCenterY - 50);
|
||||||
await expect(page.getByText('Driving')).toBeVisible();
|
await expect(page.getByText('Driving')).toBeVisible();
|
||||||
await expect(page.getByText('Science')).toBeVisible();
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
|
||||||
|
// add another tag and expect it to appear without changing selection
|
||||||
|
await page.getByRole('button', { name: /Add Tag/ }).click();
|
||||||
|
await page.getByPlaceholder('Type to select tag').click();
|
||||||
|
await page.getByText('Drilling').click();
|
||||||
|
await expect(page.getByText('Driving')).toBeVisible();
|
||||||
|
await expect(page.getByText('Science')).toBeVisible();
|
||||||
|
await expect(page.getByText('Drilling')).toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Can use + - buttons to zoom on the image @unstable', async ({ page }) => {
|
test('Can use + - buttons to zoom on the image @unstable', async ({ page }) => {
|
||||||
@ -364,7 +397,7 @@ test.describe('Example Imagery in Display Layout', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Edit mode
|
// Edit mode
|
||||||
await page.click('button[title="Edit"]');
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Click on example imagery to expose toolbar
|
// Click on example imagery to expose toolbar
|
||||||
await page.locator('.c-so-view__header').click();
|
await page.locator('.c-so-view__header').click();
|
||||||
@ -383,7 +416,7 @@ test.describe('Example Imagery in Display Layout', () => {
|
|||||||
test('Resizing the layout changes thumbnail visibility and size', async ({ page }) => {
|
test('Resizing the layout changes thumbnail visibility and size', async ({ page }) => {
|
||||||
const thumbsWrapperLocator = page.locator('.c-imagery__thumbs-wrapper');
|
const thumbsWrapperLocator = page.locator('.c-imagery__thumbs-wrapper');
|
||||||
// Edit mode
|
// Edit mode
|
||||||
await page.click('button[title="Edit"]');
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Click on example imagery to expose toolbar
|
// Click on example imagery to expose toolbar
|
||||||
await page.locator('.c-so-view__header').click();
|
await page.locator('.c-so-view__header').click();
|
||||||
@ -452,6 +485,33 @@ test.describe('Example Imagery in Display Layout', () => {
|
|||||||
|
|
||||||
test.describe('Example Imagery in Flexible layout', () => {
|
test.describe('Example Imagery in Flexible layout', () => {
|
||||||
let flexibleLayout;
|
let flexibleLayout;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
flexibleLayout = await createDomainObjectWithDefaults(page, { type: 'Flexible Layout' });
|
||||||
|
|
||||||
|
// Create Example Imagery inside the Flexible Layout
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Example Imagery',
|
||||||
|
parent: flexibleLayout.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
// Navigate back to Flexible Layout
|
||||||
|
await page.goto(flexibleLayout.url);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can double-click on the image to view large image', async ({ page }) => {
|
||||||
|
// Double-click on the image to open large view
|
||||||
|
const imageElement = await page.getByRole('button', { name: 'Image Wrapper' });
|
||||||
|
await imageElement.dblclick();
|
||||||
|
|
||||||
|
// Check if the large view is visible
|
||||||
|
await page.getByRole('button', { name: 'Background Image', state: 'visible' });
|
||||||
|
|
||||||
|
// Close the large view
|
||||||
|
await page.getByLabel('Close').click();
|
||||||
|
});
|
||||||
|
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
@ -460,7 +520,7 @@ test.describe('Example Imagery in Flexible layout', () => {
|
|||||||
|
|
||||||
/* Create Sine Wave Generator with minimum Image Load Delay */
|
/* Create Sine Wave Generator with minimum Image Load Delay */
|
||||||
// Click the Create button
|
// Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click text=Example Imagery
|
// Click text=Example Imagery
|
||||||
await page.click('li[role="menuitem"]:has-text("Example Imagery")');
|
await page.click('li[role="menuitem"]:has-text("Example Imagery")');
|
||||||
@ -504,7 +564,7 @@ test.describe('Example Imagery in Tabs View', () => {
|
|||||||
|
|
||||||
/* Create Sine Wave Generator with minimum Image Load Delay */
|
/* Create Sine Wave Generator with minimum Image Load Delay */
|
||||||
// Click the Create button
|
// Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click text=Example Imagery
|
// Click text=Example Imagery
|
||||||
await page.click('li[role="menuitem"]:has-text("Example Imagery")');
|
await page.click('li[role="menuitem"]:has-text("Example Imagery")');
|
||||||
@ -547,6 +607,7 @@ test.describe('Example Imagery in Time Strip', () => {
|
|||||||
// Navigate to timestrip
|
// Navigate to timestrip
|
||||||
await page.goto(timeStripObject.url);
|
await page.goto(timeStripObject.url);
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Clicking a thumbnail loads the image in large view', async ({ page, browserName }) => {
|
test('Clicking a thumbnail loads the image in large view', async ({ page, browserName }) => {
|
||||||
test.info().annotations.push({
|
test.info().annotations.push({
|
||||||
type: 'issue',
|
type: 'issue',
|
||||||
@ -951,7 +1012,7 @@ async function resetImageryPanAndZoom(page) {
|
|||||||
*/
|
*/
|
||||||
async function createImageryView(page) {
|
async function createImageryView(page) {
|
||||||
// Click the Create button
|
// Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.getByRole('button', { name: 'Create' }).click();
|
||||||
|
|
||||||
// Click text=Example Imagery
|
// Click text=Example Imagery
|
||||||
await page.click('li[role="menuitem"]:has-text("Example Imagery")');
|
await page.click('li[role="menuitem"]:has-text("Example Imagery")');
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -24,36 +24,182 @@
|
|||||||
This test suite is dedicated to tests which verify the basic operations surrounding exportAsJSON.
|
This test suite is dedicated to tests which verify the basic operations surrounding exportAsJSON.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// FIXME: Remove this eslint exception once tests are implemented
|
import fs from 'fs/promises';
|
||||||
// eslint-disable-next-line no-unused-vars
|
|
||||||
const { test, expect } = require('../../../../baseFixtures');
|
import {
|
||||||
|
createDomainObjectWithDefaults,
|
||||||
|
openObjectTreeContextMenu
|
||||||
|
} from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../baseFixtures.js';
|
||||||
|
import { navigateToFaultManagementWithExample } from '../../../../helper/faultUtils.js';
|
||||||
|
|
||||||
test.describe('ExportAsJSON', () => {
|
test.describe('ExportAsJSON', () => {
|
||||||
test.fixme(
|
let folder;
|
||||||
'Create a basic object and verify that it can be exported as JSON from Tree',
|
test.beforeEach(async ({ page }) => {
|
||||||
async ({ page }) => {
|
// Go to baseURL
|
||||||
//Create domain object
|
await page.goto('./');
|
||||||
//Save Domain Object
|
// Perform actions to create the domain object
|
||||||
//Verify that the newly created domain object can be exported as JSON from the Tree
|
folder = await createDomainObjectWithDefaults(page, {
|
||||||
}
|
type: 'Folder',
|
||||||
);
|
name: 'e2e folder'
|
||||||
test.fixme(
|
|
||||||
'Create a basic object and verify that it can be exported as JSON from 3 dot menu',
|
|
||||||
async ({ page }) => {
|
|
||||||
//Create domain object
|
|
||||||
//Save Domain Object
|
|
||||||
//Verify that the newly created domain object can be exported as JSON from the 3 dot menu
|
|
||||||
}
|
|
||||||
);
|
|
||||||
test.fixme('Verify that a nested Object can be exported as JSON', async ({ page }) => {
|
|
||||||
// Create 2 objects with hierarchy
|
|
||||||
// Export as JSON
|
|
||||||
// Verify Hierarchy
|
|
||||||
});
|
});
|
||||||
test.fixme(
|
});
|
||||||
'Verify that the ExportAsJSON dropdown does not appear for the item X',
|
test('Create a basic object and verify that it can be exported as JSON from Tree', async ({
|
||||||
async ({ page }) => {
|
page
|
||||||
// Other than non-persistable objects
|
}) => {
|
||||||
}
|
// Navigate to the page
|
||||||
|
await page.goto(folder.url);
|
||||||
|
|
||||||
|
// Open context menu and initiate download
|
||||||
|
await openObjectTreeContextMenu(page, folder.url);
|
||||||
|
const [download] = await Promise.all([
|
||||||
|
page.waitForEvent('download'), // Waits for the download event
|
||||||
|
page.getByLabel('Export as JSON').click() // Triggers the download
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Wait for the download process to complete
|
||||||
|
const path = await download.path();
|
||||||
|
|
||||||
|
// Read the contents of the downloaded file using readFile from fs/promises
|
||||||
|
const fileContents = await fs.readFile(path, 'utf8');
|
||||||
|
const jsonData = JSON.parse(fileContents);
|
||||||
|
|
||||||
|
// Use the function to retrieve the key
|
||||||
|
const key = getFirstKeyFromOpenMctJson(jsonData);
|
||||||
|
|
||||||
|
// Verify the contents of the JSON file
|
||||||
|
expect(jsonData.openmct[key]).toHaveProperty('name', 'e2e folder');
|
||||||
|
expect(jsonData.openmct[key]).toHaveProperty('type', 'folder');
|
||||||
|
});
|
||||||
|
test('Create a basic object and verify that it can be exported as JSON from 3 dot menu', async ({
|
||||||
|
page
|
||||||
|
}) => {
|
||||||
|
// Navigate to the page
|
||||||
|
await page.goto(folder.url);
|
||||||
|
//3 dot menu
|
||||||
|
await page.getByLabel('More actions').click();
|
||||||
|
// Open context menu and initiate download
|
||||||
|
const [download] = await Promise.all([
|
||||||
|
page.waitForEvent('download'), // Waits for the download event
|
||||||
|
page.getByLabel('Export as JSON').click() // Triggers the download
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Read the contents of the downloaded file using readFile from fs/promises
|
||||||
|
const fileContents = await fs.readFile(await download.path(), 'utf8');
|
||||||
|
const jsonData = JSON.parse(fileContents);
|
||||||
|
|
||||||
|
// Use the function to retrieve the key
|
||||||
|
const key = getFirstKeyFromOpenMctJson(jsonData);
|
||||||
|
|
||||||
|
// Verify the contents of the JSON file
|
||||||
|
expect(jsonData.openmct[key]).toHaveProperty('name', 'e2e folder');
|
||||||
|
expect(jsonData.openmct[key]).toHaveProperty('type', 'folder');
|
||||||
|
});
|
||||||
|
test('Verify that a nested Object can be exported as JSON', async ({ page }) => {
|
||||||
|
const timer = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Timer',
|
||||||
|
name: 'timer',
|
||||||
|
parent: folder.uuid
|
||||||
|
});
|
||||||
|
// Navigate to the page
|
||||||
|
await page.goto(timer.url);
|
||||||
|
|
||||||
|
//do this against parent folder.url, NOT timer.url child
|
||||||
|
await openObjectTreeContextMenu(page, folder.url);
|
||||||
|
// Open context menu and initiate download
|
||||||
|
const [download] = await Promise.all([
|
||||||
|
page.waitForEvent('download'), // Waits for the download event
|
||||||
|
page.getByLabel('Export as JSON').click() // Triggers the download
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Read the contents of the downloaded file
|
||||||
|
const fileContents = await fs.readFile(await download.path(), 'utf8');
|
||||||
|
const jsonData = JSON.parse(fileContents);
|
||||||
|
|
||||||
|
// Retrieve the keys for folder and timer
|
||||||
|
const folderKey = getFirstKeyFromOpenMctJson(jsonData);
|
||||||
|
const timerKey = jsonData.openmct[folderKey].composition[0].key;
|
||||||
|
|
||||||
|
// Verify the folder properties
|
||||||
|
expect(jsonData.openmct[folderKey]).toHaveProperty('name', 'e2e folder');
|
||||||
|
expect(jsonData.openmct[folderKey]).toHaveProperty('type', 'folder');
|
||||||
|
|
||||||
|
// Verify the timer properties
|
||||||
|
expect(jsonData.openmct[timerKey]).toHaveProperty('name', 'timer');
|
||||||
|
expect(jsonData.openmct[timerKey]).toHaveProperty('type', 'timer');
|
||||||
|
|
||||||
|
// Verify the composition of the folder includes the timer
|
||||||
|
expect(jsonData.openmct[folderKey].composition).toEqual(
|
||||||
|
expect.arrayContaining([expect.objectContaining({ key: timerKey })])
|
||||||
);
|
);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
test.describe('ExportAsJSON Disabled Actions', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
//Use a Fault Management Object which is not composable
|
||||||
|
await navigateToFaultManagementWithExample(page);
|
||||||
|
});
|
||||||
|
test('Verify that the ExportAsJSON dropdown does not appear for the item X', async ({ page }) => {
|
||||||
|
await page.getByLabel('More actions').click();
|
||||||
|
await expect(await page.getByLabel('Export as JSON')).toHaveCount(0);
|
||||||
|
|
||||||
|
await page.getByRole('treeitem', { name: 'Fault Management' }).click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
await expect(await page.getByLabel('Export as JSON')).toHaveCount(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test.describe('ExportAsJSON ProgressBar @couchdb', () => {
|
||||||
|
let folder;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
// Perform actions to create the domain object
|
||||||
|
folder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder'
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Timer',
|
||||||
|
parent: folder.uuid
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Timer',
|
||||||
|
parent: folder.uuid
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('Verify that the ExportAsJSON action creates a progressbar', async ({ page }) => {
|
||||||
|
// Navigate to the page
|
||||||
|
await page.goto(folder.url);
|
||||||
|
|
||||||
|
//Export My Items to create a large export
|
||||||
|
await page.getByRole('treeitem', { name: 'My Items' }).click({ button: 'right' });
|
||||||
|
// Open context menu and initiate download
|
||||||
|
await Promise.all([
|
||||||
|
page.getByRole('progressbar'), // This is just a check for the progress bar
|
||||||
|
page.getByText(
|
||||||
|
'Do not navigate away from this page or close this browser tab while this message'
|
||||||
|
), // This is the text associated with the download
|
||||||
|
page.waitForEvent('download'), // Waits for the download event
|
||||||
|
page.getByLabel('Export as JSON').click() // Triggers the download
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the first key from the 'openmct' property of the provided JSON object.
|
||||||
|
*
|
||||||
|
* @param {Object} jsonData - The JSON object containing the 'openmct' property.
|
||||||
|
* @returns {string} The first key found in the 'openmct' object.
|
||||||
|
* @throws {Error} If no keys are found in the 'openmct' object.
|
||||||
|
*/
|
||||||
|
function getFirstKeyFromOpenMctJson(jsonData) {
|
||||||
|
if (!jsonData.openmct) {
|
||||||
|
throw new Error("The provided JSON object does not have an 'openmct' property.");
|
||||||
|
}
|
||||||
|
|
||||||
|
const keys = Object.keys(jsonData.openmct);
|
||||||
|
if (keys.length === 0) {
|
||||||
|
throw new Error('No keys found in the openmct object');
|
||||||
|
}
|
||||||
|
|
||||||
|
return keys[0];
|
||||||
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -26,7 +26,7 @@ This test suite is dedicated to tests which verify the basic operations surround
|
|||||||
|
|
||||||
// FIXME: Remove this eslint exception once tests are implemented
|
// FIXME: Remove this eslint exception once tests are implemented
|
||||||
// eslint-disable-next-line no-unused-vars
|
// eslint-disable-next-line no-unused-vars
|
||||||
const { test, expect } = require('../../../../baseFixtures');
|
import { expect, test } from '../../../../baseFixtures.js';
|
||||||
|
|
||||||
test.describe('ExportAsJSON', () => {
|
test.describe('ExportAsJSON', () => {
|
||||||
test.fixme('Verify that domain object can be importAsJSON from Tree', async ({ page }) => {
|
test.fixme('Verify that domain object can be importAsJSON from Tree', async ({ page }) => {
|
||||||
|
@ -0,0 +1,82 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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 { fileURLToPath } from 'url';
|
||||||
|
|
||||||
|
import { createDomainObjectWithDefaults } from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
|
test.describe('Testing numeric data with inspector data visualization (i.e., data pivoting)', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({
|
||||||
|
path: fileURLToPath(
|
||||||
|
new URL('../../../../helper/addInitDataVisualization.js', import.meta.url)
|
||||||
|
)
|
||||||
|
});
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can click on telemetry and see data in inspector @2p', async ({ page, context }) => {
|
||||||
|
const exampleDataVisualizationSource = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Example Data Visualization Source'
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: 'First Sine Wave Generator',
|
||||||
|
parent: exampleDataVisualizationSource.uuid
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: 'Second Sine Wave Generator',
|
||||||
|
parent: exampleDataVisualizationSource.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(exampleDataVisualizationSource.url);
|
||||||
|
|
||||||
|
await page.getByRole('tab', { name: 'Data Visualization' }).click();
|
||||||
|
await page.getByRole('cell', { name: /First Sine Wave Generator/ }).click();
|
||||||
|
await expect(page.getByText('Numeric Data')).toBeVisible();
|
||||||
|
await expect(
|
||||||
|
page.locator('span.plot-series-name', { hasText: 'First Sine Wave Generator Hz' })
|
||||||
|
).toBeVisible();
|
||||||
|
await expect(page.locator('.js-series-data-loaded')).toBeVisible();
|
||||||
|
|
||||||
|
await page.getByRole('cell', { name: /Second Sine Wave Generator/ }).click();
|
||||||
|
await expect(page.getByText('Numeric Data')).toBeVisible();
|
||||||
|
await expect(
|
||||||
|
page.locator('span.plot-series-name', { hasText: 'Second Sine Wave Generator Hz' })
|
||||||
|
).toBeVisible();
|
||||||
|
await expect(page.locator('.js-series-data-loaded')).toBeVisible();
|
||||||
|
|
||||||
|
// test new tab
|
||||||
|
await page.getByLabel('Inspector Views').getByLabel('More actions').click();
|
||||||
|
const pagePromise = context.waitForEvent('page');
|
||||||
|
await page.getByRole('menuitem', { name: /Open In New Tab/ }).click();
|
||||||
|
|
||||||
|
// ensure our new tab's title is correct
|
||||||
|
const newPage = await pagePromise;
|
||||||
|
await newPage.waitForLoadState();
|
||||||
|
// expect new tab title to contain 'Second Sine Wave Generator'
|
||||||
|
await expect(newPage).toHaveTitle('Second Sine Wave Generator');
|
||||||
|
});
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -20,26 +20,29 @@
|
|||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
const { test, expect } = require('../../../../pluginFixtures');
|
import {
|
||||||
const {
|
|
||||||
createDomainObjectWithDefaults,
|
createDomainObjectWithDefaults,
|
||||||
setStartOffset,
|
openObjectTreeContextMenu,
|
||||||
setFixedTimeMode,
|
setFixedTimeMode,
|
||||||
setRealTimeMode
|
setRealTimeMode,
|
||||||
} = require('../../../../appActions');
|
setStartOffset
|
||||||
|
} from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
test.describe('Testing LAD table configuration', () => {
|
test.describe('Testing LAD table configuration', () => {
|
||||||
|
let ladTable;
|
||||||
|
let sineWaveObject;
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
// Create LAD table
|
// Create LAD table
|
||||||
const ladTable = await createDomainObjectWithDefaults(page, {
|
ladTable = await createDomainObjectWithDefaults(page, {
|
||||||
type: 'LAD Table',
|
type: 'LAD Table',
|
||||||
name: 'Test LAD Table'
|
name: 'Test LAD Table'
|
||||||
});
|
});
|
||||||
|
|
||||||
// Create Sine Wave Generator
|
// Create Sine Wave Generator
|
||||||
await createDomainObjectWithDefaults(page, {
|
sineWaveObject = await createDomainObjectWithDefaults(page, {
|
||||||
type: 'Sine Wave Generator',
|
type: 'Sine Wave Generator',
|
||||||
name: 'Test Sine Wave Generator',
|
name: 'Test Sine Wave Generator',
|
||||||
parent: ladTable.uuid
|
parent: ladTable.uuid
|
||||||
@ -49,77 +52,168 @@ test.describe('Testing LAD table configuration', () => {
|
|||||||
});
|
});
|
||||||
test('in edit mode, LAD Tables provide ability to hide columns', async ({ page }) => {
|
test('in edit mode, LAD Tables provide ability to hide columns', async ({ page }) => {
|
||||||
// Edit LAD table
|
// Edit LAD table
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// // Expand the 'My Items' folder in the left tree
|
|
||||||
// await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
|
||||||
// // Add the Sine Wave Generator to the LAD table and save changes
|
|
||||||
// await page.dragAndDrop('role=treeitem[name=/Test Sine Wave Generator/]', '.c-lad-table-wrapper');
|
|
||||||
// select configuration tab in inspector
|
|
||||||
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
||||||
|
|
||||||
// make sure headers are visible initially
|
// make sure headers are visible initially
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// hide timestamp column
|
// hide timestamp column
|
||||||
await page.getByLabel('Timestamp').uncheck();
|
await page.getByLabel('Timestamp', { exact: true }).uncheck();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// hide units & type column
|
// hide units & type column
|
||||||
await page.getByLabel('Units').uncheck();
|
await page.getByLabel('Units').uncheck();
|
||||||
await page.getByLabel('Type').uncheck();
|
await page.getByLabel('Type', { exact: true }).uncheck();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
|
// hide WATCH column
|
||||||
|
await page.getByLabel('WATCH').uncheck();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// save and reload and verify they columns are still hidden
|
// save and reload and verify they columns are still hidden
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
await page.reload();
|
await page.reload();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// Edit LAD table
|
// Edit LAD table
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
||||||
|
|
||||||
// show timestamp column
|
// show timestamp column
|
||||||
await page.getByLabel('Timestamp').check();
|
await page.getByLabel('Timestamp', { exact: true }).check();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// save and reload and make sure only timestamp is still visible
|
// save and reload and make sure timestamp is still visible
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
await page.reload();
|
await page.reload();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeHidden();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeHidden();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// Edit LAD table
|
// Edit LAD table
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
||||||
|
|
||||||
// show units and type columns
|
// show units, type, and WATCH columns
|
||||||
await page.getByLabel('Units').check();
|
await page.getByLabel('Units').check();
|
||||||
await page.getByLabel('Type').check();
|
await page.getByLabel('Type', { exact: true }).check();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
await page.getByLabel('WATCH').check();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
// save and reload and make sure all columns are still visible
|
// save and reload and make sure all columns are still visible
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
await page.reload();
|
await page.reload();
|
||||||
await expect(page.getByRole('cell', { name: 'Timestamp' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
await expect(page.getByRole('cell', { name: 'Type' })).toBeVisible();
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('When adding something without Units, do not show Units column', async ({ page }) => {
|
||||||
|
// Create Sine Wave Generator
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Event Message Generator',
|
||||||
|
parent: ladTable.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(ladTable.url);
|
||||||
|
|
||||||
|
// Edit LAD table
|
||||||
|
await page.getByLabel('Edit Object').click();
|
||||||
|
await page.getByRole('tab', { name: 'LAD Table Configuration' }).click();
|
||||||
|
|
||||||
|
// make sure Sine Wave headers are visible initially too
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeVisible();
|
||||||
|
|
||||||
|
// save and reload and verify they columns are still hidden
|
||||||
|
await page.getByLabel('Save').click();
|
||||||
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
|
// Remove Sine Wave Generator
|
||||||
|
openObjectTreeContextMenu(page, sineWaveObject.url);
|
||||||
|
await page.getByRole('menuitem', { name: /Remove/ }).click();
|
||||||
|
await page.getByRole('button', { name: 'OK', exact: true }).click();
|
||||||
|
|
||||||
|
// Ensure Units & Limit columns are gone
|
||||||
|
// as Event Generator don't have them
|
||||||
|
await page.goto(ladTable.url);
|
||||||
|
await expect(page.getByRole('cell', { name: 'Timestamp', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Type', exact: true })).toBeVisible();
|
||||||
|
await expect(page.getByRole('cell', { name: 'Units' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WATCH' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'WARNING' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'DISTRESS' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'CRITICAL' })).toBeHidden();
|
||||||
|
await expect(page.getByRole('cell', { name: 'SEVERE' })).toBeHidden();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("LAD Tables don't allow selection of rows but does show context click menus", async ({
|
test("LAD Tables don't allow selection of rows but does show context click menus", async ({
|
||||||
@ -164,14 +258,14 @@ test.describe('Testing LAD table @unstable', () => {
|
|||||||
name: 'Test LAD Table'
|
name: 'Test LAD Table'
|
||||||
});
|
});
|
||||||
// Edit LAD table
|
// Edit LAD table
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
// Add the Sine Wave Generator to the LAD table and save changes
|
// Add the Sine Wave Generator to the LAD table and save changes
|
||||||
await page.dragAndDrop('text=Test Sine Wave Generator', '.c-lad-table-wrapper');
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.c-lad-table-wrapper');
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Subscribe to the Sine Wave Generator data
|
// Subscribe to the Sine Wave Generator data
|
||||||
// On getting data, check if the value found in the LAD table is the most recent value
|
// On getting data, check if the value found in the LAD table is the most recent value
|
||||||
@ -192,14 +286,14 @@ test.describe('Testing LAD table @unstable', () => {
|
|||||||
name: 'Test LAD Table'
|
name: 'Test LAD Table'
|
||||||
});
|
});
|
||||||
// Edit LAD table
|
// Edit LAD table
|
||||||
await page.locator('[title="Edit"]').click();
|
await page.getByLabel('Edit Object').click();
|
||||||
|
|
||||||
// Expand the 'My Items' folder in the left tree
|
// Expand the 'My Items' folder in the left tree
|
||||||
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').click();
|
||||||
// Add the Sine Wave Generator to the LAD table and save changes
|
// Add the Sine Wave Generator to the LAD table and save changes
|
||||||
await page.dragAndDrop('text=Test Sine Wave Generator', '.c-lad-table-wrapper');
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.c-lad-table-wrapper');
|
||||||
await page.locator('button[title="Save"]').click();
|
await page.locator('button[title="Save"]').click();
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
await page.getByRole('listitem', { name: 'Save and Finish Editing' }).click();
|
||||||
|
|
||||||
// Subscribe to the Sine Wave Generator data
|
// Subscribe to the Sine Wave Generator data
|
||||||
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
||||||
|
69
e2e/tests/functional/plugins/lad/ladSet.e2e.spec.js
Normal file
69
e2e/tests/functional/plugins/lad/ladSet.e2e.spec.js
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* 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 { createDomainObjectWithDefaults } from '../../../../appActions.js';
|
||||||
|
import { expect, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
|
test.describe('LAD Table Sets', () => {
|
||||||
|
test('Ensure we have numbers in cells', async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
const ladTableSet = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'LAD Table Set'
|
||||||
|
});
|
||||||
|
|
||||||
|
const firstLadTable = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'LAD Table',
|
||||||
|
parent: ladTableSet.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
parent: firstLadTable.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
const secondLadTable = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'LAD Table',
|
||||||
|
parent: ladTableSet.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
parent: secondLadTable.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(ladTableSet.url);
|
||||||
|
|
||||||
|
// Wait for the initial value to show after mount
|
||||||
|
await expect(page.getByLabel('lad value').first()).not.toContainText('---');
|
||||||
|
|
||||||
|
const valueFromFirstSineWave = await page.getByLabel('lad value').first().innerText();
|
||||||
|
const firstSineWaveNumber = parseFloat(valueFromFirstSineWave);
|
||||||
|
// ensure we have a float value in the cell and it's finite
|
||||||
|
expect(Number.isFinite(firstSineWaveNumber)).toBeTruthy();
|
||||||
|
|
||||||
|
const valueFromSecondSineWave = await page.getByLabel('lad value').last().innerText();
|
||||||
|
const secondSineWaveNumber = parseFloat(valueFromSecondSineWave);
|
||||||
|
// ensure we have a float value in the cell and it's finite
|
||||||
|
expect(Number.isFinite(secondSineWaveNumber)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
@ -1,5 +1,5 @@
|
|||||||
/*****************************************************************************
|
/*****************************************************************************
|
||||||
* Open MCT, Copyright (c) 2014-2023, United States Government
|
* Open MCT, Copyright (c) 2014-2024, United States Government
|
||||||
* as represented by the Administrator of the National Aeronautics and Space
|
* as represented by the Administrator of the National Aeronautics and Space
|
||||||
* Administration. All rights reserved.
|
* Administration. All rights reserved.
|
||||||
*
|
*
|
||||||
@ -19,25 +19,39 @@
|
|||||||
* this source code distribution or the Licensing information page available
|
* this source code distribution or the Licensing information page available
|
||||||
* at runtime from the About dialog for additional information.
|
* at runtime from the About dialog for additional information.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
/* global __dirname */
|
|
||||||
/*
|
/*
|
||||||
This test suite is dedicated to tests which verify the basic operations surrounding Notebooks.
|
This test suite is dedicated to tests which verify the basic operations surrounding Notebooks.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect, streamToString } = require('../../../../pluginFixtures');
|
import { fileURLToPath } from 'url';
|
||||||
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
|
||||||
const nbUtils = require('../../../../helper/notebookUtils');
|
import { createDomainObjectWithDefaults } from '../../../../appActions.js';
|
||||||
const path = require('path');
|
import * as nbUtils from '../../../../helper/notebookUtils.js';
|
||||||
|
import { expect, streamToString, test } from '../../../../pluginFixtures.js';
|
||||||
|
|
||||||
const NOTEBOOK_NAME = 'Notebook';
|
const NOTEBOOK_NAME = 'Notebook';
|
||||||
|
|
||||||
test.describe('Notebook CRUD Operations', () => {
|
test.describe('Notebook CRUD Operations', () => {
|
||||||
test.fixme('Can create a Notebook Object', async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
|
//Navigate to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
});
|
||||||
|
test('Can create a Notebook Object', async ({ page }) => {
|
||||||
//Create domain object
|
//Create domain object
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: NOTEBOOK_NAME
|
||||||
|
});
|
||||||
//Newly created notebook should have one Section and one page, 'Unnamed Section'/'Unnamed Page'
|
//Newly created notebook should have one Section and one page, 'Unnamed Section'/'Unnamed Page'
|
||||||
|
const notebookSectionNames = page.locator('.c-notebook__sections .c-list__item__name');
|
||||||
|
const notebookPageNames = page.locator('.c-notebook__pages .c-list__item__name');
|
||||||
|
await expect(notebookSectionNames).toBeHidden();
|
||||||
|
await expect(notebookPageNames).toBeHidden();
|
||||||
|
await expect(notebookSectionNames).toHaveText('Unnamed Section');
|
||||||
|
await expect(notebookPageNames).toHaveText('Unnamed Page');
|
||||||
});
|
});
|
||||||
test.fixme('Can update a Notebook Object', async ({ page }) => {});
|
test.fixme('Can update a Notebook Object', async ({ page }) => {});
|
||||||
test.fixme('Can view a perviously created Notebook Object', async ({ page }) => {});
|
test.fixme('Can view a previously created Notebook Object', async ({ page }) => {});
|
||||||
test.fixme('Can Delete a Notebook Object', async ({ page }) => {
|
test.fixme('Can Delete a Notebook Object', async ({ page }) => {
|
||||||
// Other than non-persistable objects
|
// Other than non-persistable objects
|
||||||
});
|
});
|
||||||
@ -233,7 +247,7 @@ test.describe('Notebook export tests', () => {
|
|||||||
test('can export notebook as text', async ({ page }) => {
|
test('can export notebook as text', async ({ page }) => {
|
||||||
await nbUtils.enterTextEntry(page, `Foo bar entry`);
|
await nbUtils.enterTextEntry(page, `Foo bar entry`);
|
||||||
// Click on 3 Dot Menu
|
// Click on 3 Dot Menu
|
||||||
await page.locator('button[title="More options"]').click();
|
await page.locator('button[title="More actions"]').click();
|
||||||
const downloadPromise = page.waitForEvent('download');
|
const downloadPromise = page.waitForEvent('download');
|
||||||
|
|
||||||
await page.getByRole('menuitem', { name: /Export Notebook as Text/ }).click();
|
await page.getByRole('menuitem', { name: /Export Notebook as Text/ }).click();
|
||||||
@ -265,7 +279,7 @@ test.describe('Notebook entry tests', () => {
|
|||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
await page.addInitScript({
|
await page.addInitScript({
|
||||||
path: path.join(__dirname, '../../../../helper/', 'addInitNotebookWithUrls.js')
|
path: fileURLToPath(new URL('../../../../helper/addInitNotebookWithUrls.js', import.meta.url))
|
||||||
});
|
});
|
||||||
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
await page.goto('./', { waitUntil: 'domcontentloaded' });
|
||||||
|
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user