mirror of
https://github.com/nasa/openmct.git
synced 2025-01-24 21:37:15 +00:00
Merge branch 'master' into test-heuristics
This commit is contained in:
commit
43857159be
@ -2,9 +2,11 @@ version: 2.1
|
|||||||
executors:
|
executors:
|
||||||
pw-focal-development:
|
pw-focal-development:
|
||||||
docker:
|
docker:
|
||||||
- image: mcr.microsoft.com/playwright:v1.21.1-focal
|
- image: mcr.microsoft.com/playwright:v1.25.2-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_LOGLEVEL: 'debug' # Enable DEBUG level logging for Percy (Issue: https://github.com/nasa/openmct/issues/5742)
|
||||||
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!"
|
||||||
@ -12,7 +14,7 @@ parameters:
|
|||||||
type: boolean
|
type: boolean
|
||||||
commands:
|
commands:
|
||||||
build_and_install:
|
build_and_install:
|
||||||
description: "All steps used to build and install. Will not work on node10"
|
description: "All steps used to build and install. Will use cache if found"
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
node-version:
|
||||||
type: string
|
type: string
|
||||||
@ -23,7 +25,7 @@ commands:
|
|||||||
- node/install:
|
- node/install:
|
||||||
install-npm: true
|
install-npm: true
|
||||||
node-version: << parameters.node-version >>
|
node-version: << parameters.node-version >>
|
||||||
- run: npm install
|
- run: npm install --prefer-offline --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:
|
||||||
@ -58,10 +60,14 @@ commands:
|
|||||||
ls -latR >> /tmp/artifacts/dir.txt
|
ls -latR >> /tmp/artifacts/dir.txt
|
||||||
- store_artifacts:
|
- store_artifacts:
|
||||||
path: /tmp/artifacts/
|
path: /tmp/artifacts/
|
||||||
upload_code_covio:
|
generate_e2e_code_cov_report:
|
||||||
description: "Command to upload code coverage reports to codecov.io"
|
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:
|
||||||
|
suite:
|
||||||
|
type: string
|
||||||
steps:
|
steps:
|
||||||
- run: curl -Os https://uploader.codecov.io/latest/linux/codecov;chmod +x codecov;./codecov
|
- run: npm run cov:e2e:report || true
|
||||||
|
- run: npm run cov:e2e:<<parameters.suite>>:publish
|
||||||
orbs:
|
orbs:
|
||||||
node: circleci/node@4.9.0
|
node: circleci/node@4.9.0
|
||||||
browser-tools: circleci/browser-tools@1.3.0
|
browser-tools: circleci/browser-tools@1.3.0
|
||||||
@ -90,106 +96,120 @@ jobs:
|
|||||||
parameters:
|
parameters:
|
||||||
node-version:
|
node-version:
|
||||||
type: string
|
type: string
|
||||||
browser:
|
|
||||||
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: <<parameters.node-version>>
|
||||||
- when:
|
|
||||||
condition:
|
|
||||||
equal: [ "FirefoxESR", <<parameters.browser>> ]
|
|
||||||
steps:
|
|
||||||
- browser-tools/install-firefox:
|
|
||||||
version: "91.7.1esr" #https://archive.mozilla.org/pub/firefox/releases/
|
|
||||||
- when:
|
|
||||||
condition:
|
|
||||||
equal: [ "FirefoxHeadless", <<parameters.browser>> ]
|
|
||||||
steps:
|
|
||||||
- browser-tools/install-firefox
|
|
||||||
- when:
|
|
||||||
condition:
|
|
||||||
equal: [ "ChromeHeadless", <<parameters.browser>> ]
|
|
||||||
steps:
|
|
||||||
- browser-tools/install-chrome:
|
- browser-tools/install-chrome:
|
||||||
replace-existing: false
|
replace-existing: false
|
||||||
- run: npm run test -- --browsers=<<parameters.browser>>
|
- run: npm run test
|
||||||
|
- run: npm run cov:unit:publish
|
||||||
- save_cache_cmd:
|
- save_cache_cmd:
|
||||||
node-version: <<parameters.node-version>>
|
node-version: <<parameters.node-version>>
|
||||||
- store_test_results:
|
- store_test_results:
|
||||||
path: dist/reports/tests/
|
path: dist/reports/tests/
|
||||||
- store_artifacts:
|
- store_artifacts:
|
||||||
path: dist/reports/
|
path: coverage
|
||||||
- generate_and_store_version_and_filesystem_artifacts
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
e2e-test:
|
e2e-test:
|
||||||
parameters:
|
parameters:
|
||||||
node-version:
|
node-version:
|
||||||
type: string
|
type: string
|
||||||
suite:
|
suite: #stable or full
|
||||||
|
type: string
|
||||||
|
executor: pw-focal-development
|
||||||
|
parallelism: 4
|
||||||
|
steps:
|
||||||
|
- build_and_install:
|
||||||
|
node-version: <<parameters.node-version>>
|
||||||
|
- when: #Only install chrome-beta when running the 'full' suite to save $$$
|
||||||
|
condition:
|
||||||
|
equal: [ "full", <<parameters.suite>> ]
|
||||||
|
steps:
|
||||||
|
- run: npx playwright install chrome-beta
|
||||||
|
- run: SHARD="$((${CIRCLE_NODE_INDEX}+1))"; npm run test:e2e:<<parameters.suite>> -- --shard=${SHARD}/${CIRCLE_NODE_TOTAL}
|
||||||
|
- generate_e2e_code_cov_report:
|
||||||
|
suite: <<parameters.suite>>
|
||||||
|
- store_test_results:
|
||||||
|
path: test-results/results.xml
|
||||||
|
- store_artifacts:
|
||||||
|
path: test-results
|
||||||
|
- store_artifacts:
|
||||||
|
path: coverage
|
||||||
|
- store_artifacts:
|
||||||
|
path: html-test-results
|
||||||
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
|
perf-test:
|
||||||
|
parameters:
|
||||||
|
node-version:
|
||||||
type: string
|
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: <<parameters.node-version>>
|
||||||
- run: npx playwright install
|
- run: npm run test:perf
|
||||||
- run: npm run test:e2e:<<parameters.suite>>
|
|
||||||
- store_test_results:
|
- store_test_results:
|
||||||
path: test-results/results.xml
|
path: test-results/results.xml
|
||||||
- store_artifacts:
|
- store_artifacts:
|
||||||
path: test-results
|
path: test-results
|
||||||
|
- store_artifacts:
|
||||||
|
path: html-test-results
|
||||||
|
- generate_and_store_version_and_filesystem_artifacts
|
||||||
|
visual-test:
|
||||||
|
parameters:
|
||||||
|
node-version:
|
||||||
|
type: string
|
||||||
|
executor: pw-focal-development
|
||||||
|
steps:
|
||||||
|
- build_and_install:
|
||||||
|
node-version: <<parameters.node-version>>
|
||||||
|
- run: npm run test:e2e:visual
|
||||||
|
- store_test_results:
|
||||||
|
path: test-results/results.xml
|
||||||
|
- store_artifacts:
|
||||||
|
path: test-results
|
||||||
|
- store_artifacts:
|
||||||
|
path: html-test-results
|
||||||
- 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: node14-lint
|
||||||
node-version: lts/gallium
|
|
||||||
- unit-test:
|
|
||||||
name: node14-chrome
|
|
||||||
node-version: lts/fermium
|
node-version: lts/fermium
|
||||||
browser: ChromeHeadless
|
|
||||||
post-steps:
|
|
||||||
- upload_code_covio
|
|
||||||
- unit-test:
|
|
||||||
name: node16-chrome
|
|
||||||
node-version: lts/gallium
|
|
||||||
browser: ChromeHeadless
|
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node18-chrome
|
name: node18-chrome
|
||||||
node-version: "18"
|
node-version: "18"
|
||||||
browser: ChromeHeadless
|
|
||||||
- e2e-test:
|
- e2e-test:
|
||||||
name: e2e-ci
|
name: e2e-stable
|
||||||
node-version: lts/gallium
|
node-version: lts/gallium
|
||||||
suite: ci
|
suite: stable
|
||||||
|
- perf-test:
|
||||||
|
node-version: lts/gallium
|
||||||
|
- visual-test:
|
||||||
|
node-version: lts/gallium
|
||||||
|
|
||||||
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:
|
|
||||||
name: node16-firefoxESR-nightly
|
|
||||||
node-version: lts/gallium
|
|
||||||
browser: FirefoxESR
|
|
||||||
- unit-test:
|
|
||||||
name: node14-firefox-nightly
|
|
||||||
node-version: lts/fermium
|
|
||||||
browser: FirefoxHeadless
|
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node14-chrome-nightly
|
name: node14-chrome-nightly
|
||||||
node-version: lts/fermium
|
node-version: lts/fermium
|
||||||
browser: ChromeHeadless
|
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node16-chrome-nightly
|
name: node16-chrome-nightly
|
||||||
node-version: lts/gallium
|
node-version: lts/gallium
|
||||||
browser: ChromeHeadless
|
|
||||||
- unit-test:
|
- unit-test:
|
||||||
name: node18-chrome
|
name: node18-chrome
|
||||||
node-version: "18"
|
node-version: "18"
|
||||||
browser: ChromeHeadless
|
|
||||||
- npm-audit:
|
- npm-audit:
|
||||||
node-version: lts/gallium
|
node-version: lts/gallium
|
||||||
- e2e-test:
|
- e2e-test:
|
||||||
name: e2e-full-nightly
|
name: e2e-full-nightly
|
||||||
node-version: lts/gallium
|
node-version: lts/gallium
|
||||||
suite: full
|
suite: full
|
||||||
|
- perf-test:
|
||||||
|
node-version: lts/gallium
|
||||||
|
- visual-test:
|
||||||
|
node-version: lts/gallium
|
||||||
triggers:
|
triggers:
|
||||||
- schedule:
|
- schedule:
|
||||||
cron: "0 0 * * *"
|
cron: "0 0 * * *"
|
||||||
|
@ -29,6 +29,7 @@ module.exports = {
|
|||||||
"you-dont-need-lodash-underscore/omit": "off",
|
"you-dont-need-lodash-underscore/omit": "off",
|
||||||
"you-dont-need-lodash-underscore/throttle": "off",
|
"you-dont-need-lodash-underscore/throttle": "off",
|
||||||
"you-dont-need-lodash-underscore/flatten": "off",
|
"you-dont-need-lodash-underscore/flatten": "off",
|
||||||
|
"you-dont-need-lodash-underscore/get": "off",
|
||||||
"no-bitwise": "error",
|
"no-bitwise": "error",
|
||||||
"curly": "error",
|
"curly": "error",
|
||||||
"eqeqeq": "error",
|
"eqeqeq": "error",
|
||||||
|
4
.github/ISSUE_TEMPLATE/bug_report.md
vendored
4
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@ -27,7 +27,7 @@ assignees: ''
|
|||||||
|
|
||||||
#### Environment
|
#### Environment
|
||||||
<!--- If encountered on local machine, execute the following:
|
<!--- If encountered on local machine, execute the following:
|
||||||
<!--- npx envinfo --system --browsers --npmPackages --binaries --languages --markdown -->
|
<!--- npx envinfo --system --browsers --npmPackages --binaries --markdown -->
|
||||||
* Open MCT Version: <!--- date of build, version, or SHA -->
|
* Open MCT Version: <!--- date of build, version, or SHA -->
|
||||||
* Deployment Type: <!--- npm dev? VIPER Dev? openmct-yamcs? -->
|
* Deployment Type: <!--- npm dev? VIPER Dev? openmct-yamcs? -->
|
||||||
* OS:
|
* OS:
|
||||||
@ -40,6 +40,8 @@ assignees: ''
|
|||||||
- [ ] Is there a workaround available?
|
- [ ] Is there a workaround available?
|
||||||
- [ ] Does this impact a critical component?
|
- [ ] Does this impact a critical component?
|
||||||
- [ ] Is this just a visual bug with no functional impact?
|
- [ ] Is this just a visual bug with no functional impact?
|
||||||
|
- [ ] Does this block the execution of e2e tests?
|
||||||
|
- [ ] Does this have an impact on Performance?
|
||||||
|
|
||||||
#### Additional Information
|
#### Additional Information
|
||||||
<!--- Include any screenshots, gifs, or logs which will expedite triage -->
|
<!--- Include any screenshots, gifs, or logs which will expedite triage -->
|
||||||
|
4
.github/PULL_REQUEST_TEMPLATE.md
vendored
4
.github/PULL_REQUEST_TEMPLATE.md
vendored
@ -13,10 +13,10 @@ Closes <!--- Insert Issue Number(s) this PR addresses. Start by typing # will op
|
|||||||
### Author Checklist
|
### Author Checklist
|
||||||
|
|
||||||
* [ ] Changes address original issue?
|
* [ ] Changes address original issue?
|
||||||
* [ ] Unit tests included and/or updated with changes?
|
* [ ] Tests included and/or updated with changes?
|
||||||
* [ ] Command line build passes?
|
* [ ] Command line build passes?
|
||||||
* [ ] Has this been smoke tested?
|
* [ ] Has this been smoke tested?
|
||||||
* [ ] Testing instructions included in associated issue?
|
* [ ] Testing instructions included in associated issue OR is this a dependency/testcase change?
|
||||||
|
|
||||||
### Reviewer Checklist
|
### Reviewer Checklist
|
||||||
|
|
||||||
|
11
.github/dependabot.yml
vendored
11
.github/dependabot.yml
vendored
@ -7,12 +7,19 @@ updates:
|
|||||||
interval: "daily"
|
interval: "daily"
|
||||||
open-pull-requests-limit: 10
|
open-pull-requests-limit: 10
|
||||||
labels:
|
labels:
|
||||||
|
- "pr:e2e"
|
||||||
- "type:maintenance"
|
- "type:maintenance"
|
||||||
- "dependencies"
|
- "dependencies"
|
||||||
- "pr:e2e"
|
|
||||||
- "pr:daveit"
|
- "pr:daveit"
|
||||||
- "pr:visual"
|
|
||||||
- "pr:platform"
|
- "pr:platform"
|
||||||
|
ignore:
|
||||||
|
#We have to source the container which is not detected by Dependabot
|
||||||
|
- dependency-name: "@playwright/test"
|
||||||
|
#Lots of noise in these type patch releases.
|
||||||
|
- dependency-name: "@babel/eslint-parser"
|
||||||
|
update-types: ["version-update:semver-patch"]
|
||||||
|
- dependency-name: "eslint-plugin-vue"
|
||||||
|
update-types: ["version-update:semver-patch"]
|
||||||
|
|
||||||
- package-ecosystem: "github-actions"
|
- package-ecosystem: "github-actions"
|
||||||
directory: "/"
|
directory: "/"
|
||||||
|
38
.github/workflows/e2e-couchdb.yml
vendored
Normal file
38
.github/workflows/e2e-couchdb.yml
vendored
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
name: "e2e-couchdb"
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
pull_request:
|
||||||
|
types:
|
||||||
|
- labeled
|
||||||
|
- opened
|
||||||
|
env:
|
||||||
|
OPENMCT_DATABASE_NAME: openmct
|
||||||
|
COUCH_ADMIN_USER: admin
|
||||||
|
COUCH_ADMIN_PASSWORD: password
|
||||||
|
COUCH_BASE_LOCAL: http://localhost:5984
|
||||||
|
COUCH_NODE_NAME: nonode@nohost
|
||||||
|
jobs:
|
||||||
|
e2e-couchdb:
|
||||||
|
if: ${{ github.event.label.name == 'pr:e2e:couchdb' }}
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- run : docker-compose -f src/plugins/persistence/couch/couchdb-compose.yaml up --detach
|
||||||
|
- run : sleep 3 # wait until CouchDB has started (TODO: there must be a better way)
|
||||||
|
- run : bash src/plugins/persistence/couch/setup-couchdb.sh
|
||||||
|
- uses: actions/setup-node@v3
|
||||||
|
with:
|
||||||
|
node-version: '16'
|
||||||
|
- run: npx playwright@1.25.2 install
|
||||||
|
- run: npm install
|
||||||
|
- run: sh src/plugins/persistence/couch/replace-localstorage-with-couchdb-indexhtml.sh
|
||||||
|
- run: npm run test:e2e:couchdb
|
||||||
|
- run: ls -latr
|
||||||
|
- name: Archive test results
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
path: test-results
|
||||||
|
- name: Archive html test results
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
|
with:
|
||||||
|
path: html-test-results
|
3
.github/workflows/e2e-pr.yml
vendored
3
.github/workflows/e2e-pr.yml
vendored
@ -30,7 +30,8 @@ jobs:
|
|||||||
- uses: actions/setup-node@v3
|
- uses: actions/setup-node@v3
|
||||||
with:
|
with:
|
||||||
node-version: '16'
|
node-version: '16'
|
||||||
- run: npx playwright@1.21.1 install
|
- run: npx playwright@1.25.2 install
|
||||||
|
- run: npx playwright install chrome-beta
|
||||||
- run: npm install
|
- run: npm install
|
||||||
- run: npm run test:e2e:full
|
- run: npm run test:e2e:full
|
||||||
- name: Archive test results
|
- name: Archive test results
|
||||||
|
25
.github/workflows/e2e-visual.yml
vendored
25
.github/workflows/e2e-visual.yml
vendored
@ -1,25 +0,0 @@
|
|||||||
name: "e2e-visual"
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
pull_request:
|
|
||||||
types:
|
|
||||||
- labeled
|
|
||||||
- opened
|
|
||||||
schedule:
|
|
||||||
- cron: '28 21 * * 1-5'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
e2e-visual:
|
|
||||||
if: ${{ github.event.label.name == 'pr:visual' }} || ${{ github.event.workflow_dispatch }} || ${{ github.event.schedule }}
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- uses: actions/setup-node@v3
|
|
||||||
with:
|
|
||||||
node-version: '16'
|
|
||||||
- run: npx playwright@1.21.1 install
|
|
||||||
- run: npm install
|
|
||||||
- name: Run the e2e visual tests
|
|
||||||
run: npm run test:e2e:visual
|
|
||||||
env:
|
|
||||||
PERCY_TOKEN: ${{ secrets.PERCY_TOKEN }}
|
|
2
.github/workflows/prcop-config.json
vendored
2
.github/workflows/prcop-config.json
vendored
@ -3,7 +3,7 @@
|
|||||||
{
|
{
|
||||||
"name": "descriptionRegexp",
|
"name": "descriptionRegexp",
|
||||||
"config": {
|
"config": {
|
||||||
"regexp": "x] Testing instructions",
|
"regexp": "[x|X]] Testing instructions",
|
||||||
"errorMessage": ":police_officer: PR Description does not confirm that associated issue(s) contain Testing instructions"
|
"errorMessage": ":police_officer: PR Description does not confirm that associated issue(s) contain Testing instructions"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
24
.gitignore
vendored
24
.gitignore
vendored
@ -15,8 +15,6 @@
|
|||||||
*.idea
|
*.idea
|
||||||
*.iml
|
*.iml
|
||||||
|
|
||||||
# External dependencies
|
|
||||||
|
|
||||||
# Build output
|
# Build output
|
||||||
target
|
target
|
||||||
dist
|
dist
|
||||||
@ -24,30 +22,28 @@ dist
|
|||||||
# Mac OS X Finder
|
# Mac OS X Finder
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
|
||||||
# Closed source libraries
|
|
||||||
closed-lib
|
|
||||||
|
|
||||||
# Node, Bower dependencies
|
# Node, Bower dependencies
|
||||||
node_modules
|
node_modules
|
||||||
bower_components
|
bower_components
|
||||||
|
|
||||||
# Protractor logs
|
|
||||||
protractor/logs
|
|
||||||
|
|
||||||
# npm-debug log
|
# npm-debug log
|
||||||
npm-debug.log
|
npm-debug.log
|
||||||
|
|
||||||
# karma reports
|
# karma reports
|
||||||
report.*.json
|
report.*.json
|
||||||
|
|
||||||
# Lighthouse reports
|
|
||||||
.lighthouseci
|
|
||||||
|
|
||||||
# e2e test artifacts
|
# e2e test artifacts
|
||||||
test-results
|
test-results
|
||||||
allure-results
|
html-test-results
|
||||||
|
|
||||||
package-lock.json
|
# couchdb scripting artifacts
|
||||||
|
src/plugins/persistence/couch/.env.local
|
||||||
|
index.html.bak
|
||||||
|
|
||||||
#codecov artifacts
|
# codecov artifacts
|
||||||
|
.nyc_output
|
||||||
|
coverage
|
||||||
codecov
|
codecov
|
||||||
|
|
||||||
|
# :(
|
||||||
|
package-lock.json
|
||||||
|
4
API.md
4
API.md
@ -390,7 +390,7 @@ A telemetry object is a domain object with a telemetry property. To take an exa
|
|||||||
{
|
{
|
||||||
"key": "value",
|
"key": "value",
|
||||||
"name": "Value",
|
"name": "Value",
|
||||||
"units": "kilograms",
|
"unit": "kilograms",
|
||||||
"format": "float",
|
"format": "float",
|
||||||
"min": 0,
|
"min": 0,
|
||||||
"max": 100,
|
"max": 100,
|
||||||
@ -425,7 +425,7 @@ attribute | type | flags | notes
|
|||||||
`name` | string | optional | a human readable label for this field. If omitted, defaults to `key`.
|
`name` | string | optional | a human readable label for this field. If omitted, defaults to `key`.
|
||||||
`source` | string | optional | identifies the property of a datum where this value is stored. If omitted, defaults to `key`.
|
`source` | string | optional | identifies the property of a datum where this value is stored. If omitted, defaults to `key`.
|
||||||
`format` | string | optional | a specific format identifier, mapping to a formatter. If omitted, uses a default formatter. For enumerations, use `enum`. For timestamps, use `utc` if you are using utc dates, otherwise use a key mapping to your custom date format.
|
`format` | string | optional | a specific format identifier, mapping to a formatter. If omitted, uses a default formatter. For enumerations, use `enum`. For timestamps, use `utc` if you are using utc dates, otherwise use a key mapping to your custom date format.
|
||||||
`units` | string | optional | the units of this value, e.g. `km`, `seconds`, `parsecs`
|
`unit` | string | optional | the unit of this value, e.g. `km`, `seconds`, `parsecs`
|
||||||
`min` | number | optional | the minimum possible value of this measurement. Will be used by plots, gauges, etc to automatically set a min value.
|
`min` | number | optional | the minimum possible value of this measurement. Will be used by plots, gauges, etc to automatically set a min value.
|
||||||
`max` | number | optional | the maximum possible value of this measurement. Will be used by plots, gauges, etc to automatically set a max value.
|
`max` | number | optional | the maximum possible value of this measurement. Will be used by plots, gauges, etc to automatically set a max value.
|
||||||
`enumerations` | array | optional | for objects where `format` is `"enum"`, this array tracks all possible enumerations of the value. Each entry in this array is an object, with a `value` property that is the numerical value of the enumeration, and a `string` property that is the text value of the enumeration. ex: `{"value": 0, "string": "OFF"}`. If you use an enumerations array, `min` and `max` will be set automatically for you.
|
`enumerations` | array | optional | for objects where `format` is `"enum"`, this array tracks all possible enumerations of the value. Each entry in this array is an object, with a `value` property that is the numerical value of the enumeration, and a `string` property that is the text value of the enumeration. ex: `{"value": 0, "string": "OFF"}`. If you use an enumerations array, `min` and `max` will be set automatically for you.
|
||||||
|
@ -173,7 +173,7 @@ The following guidelines are provided for anyone contributing source code to the
|
|||||||
1. Avoid deep nesting (especially of functions), except where necessary
|
1. Avoid deep nesting (especially of functions), except where necessary
|
||||||
(e.g. due to closure scope).
|
(e.g. due to closure scope).
|
||||||
1. End with a single new-line character.
|
1. End with a single new-line character.
|
||||||
1. Always use ES6 `Class`es and inheritence rather than the pre-ES6 prototypal
|
1. Always use ES6 `Class`es and inheritance rather than the pre-ES6 prototypal
|
||||||
pattern.
|
pattern.
|
||||||
1. Within a given function's scope, do not mix declarations and imperative
|
1. Within a given function's scope, do not mix declarations and imperative
|
||||||
code, and present these in the following order:
|
code, and present these in the following order:
|
||||||
|
68
README.md
68
README.md
@ -11,22 +11,6 @@ Once you've created something amazing with Open MCT, showcase your work in our G
|
|||||||
Try Open MCT now with our [live demo](https://openmct-demo.herokuapp.com/).
|
Try Open MCT now with our [live demo](https://openmct-demo.herokuapp.com/).
|
||||||
![Demo](https://nasa.github.io/openmct/static/res/images/Open-MCT.Browse.Layout.Mars-Weather-1.jpg)
|
![Demo](https://nasa.github.io/openmct/static/res/images/Open-MCT.Browse.Layout.Mars-Weather-1.jpg)
|
||||||
|
|
||||||
## Open MCT v2.0.0
|
|
||||||
Support for our legacy bundle-based API, and the libraries that it was built on (like Angular 1.x), have now been removed entirely from this repository.
|
|
||||||
|
|
||||||
For now if you have an Open MCT application that makes use of the legacy API, [a plugin](https://github.com/nasa/openmct-legacy-plugin) is provided that bootstraps the legacy bundling mechanism and API. This plugin will not be maintained over the long term however, and the legacy support plugin will not be tested for compatibility with future versions of Open MCT. It is provided for convenience only.
|
|
||||||
|
|
||||||
### How do I know if I am using legacy API?
|
|
||||||
You might still be using legacy API if your source code
|
|
||||||
|
|
||||||
* Contains files named bundle.js, or bundle.json,
|
|
||||||
* Makes calls to `openmct.$injector()`, or `openmct.$angular`,
|
|
||||||
* Makes calls to `openmct.legacyRegistry`, `openmct.legacyExtension`, or `openmct.legacyBundle`.
|
|
||||||
|
|
||||||
|
|
||||||
### 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.
|
|
||||||
|
|
||||||
## Building and Running Open MCT Locally
|
## Building and Running Open MCT Locally
|
||||||
|
|
||||||
Building and running Open MCT in your local dev environment is very easy. Be sure you have [Git](https://git-scm.com/downloads) and [Node.js](https://nodejs.org/) installed, then follow the directions below. Need additional information? Check out the [Getting Started](https://nasa.github.io/openmct/getting-started/) page on our website.
|
Building and running Open MCT in your local dev environment is very easy. Be sure you have [Git](https://git-scm.com/downloads) and [Node.js](https://nodejs.org/) installed, then follow the directions below. Need additional information? Check out the [Getting Started](https://nasa.github.io/openmct/getting-started/) page on our website.
|
||||||
@ -84,7 +68,10 @@ For information on writing plugins, please see [our API documentation](https://g
|
|||||||
|
|
||||||
## Tests
|
## Tests
|
||||||
|
|
||||||
Tests are written for [Jasmine 3](https://jasmine.github.io/api/3.1/global)
|
Our automated test coverage comes in the form of unit, e2e, visual, performance, and security tests.
|
||||||
|
|
||||||
|
### Unit Tests
|
||||||
|
Unit Tests are written for [Jasmine](https://jasmine.github.io/api/edge/global)
|
||||||
and run by [Karma](http://karma-runner.github.io). To run:
|
and run by [Karma](http://karma-runner.github.io). To run:
|
||||||
|
|
||||||
`npm test`
|
`npm test`
|
||||||
@ -93,16 +80,33 @@ The test suite is configured to load any scripts ending with `Spec.js` found
|
|||||||
in the `src` hierarchy. Full configuration details are found in
|
in the `src` hierarchy. Full configuration details are found in
|
||||||
`karma.conf.js`. By convention, unit test scripts should be located
|
`karma.conf.js`. By convention, unit test scripts should be located
|
||||||
alongside the units that they test; for example, `src/foo/Bar.js` would be
|
alongside the units that they test; for example, `src/foo/Bar.js` would be
|
||||||
tested by `src/foo/BarSpec.js`. (For legacy reasons, some existing tests may
|
tested by `src/foo/BarSpec.js`.
|
||||||
be located in separate `test` folders near the units they test, but the
|
|
||||||
naming convention is otherwise the same.)
|
|
||||||
|
|
||||||
### Test Reporting
|
### e2e, Visual, and Performance tests
|
||||||
|
The e2e, Visual, and Performance tests are written for playwright and run by playwright's new test runner [@playwright/test](https://playwright.dev/).
|
||||||
|
|
||||||
When `npm test` is run, test results will be written as HTML to
|
To run the e2e tests which are part of every commit:
|
||||||
`dist/reports/tests/`. Code coverage information is written to `dist/reports/coverage`.
|
|
||||||
|
|
||||||
Code Coverage Reports are available from [codecov.io](https://app.codecov.io/gh/nasa/openmct/)
|
`npm run test:e2e:stable`
|
||||||
|
|
||||||
|
To run the visual test suite:
|
||||||
|
|
||||||
|
`npm run test:e2e:visual`
|
||||||
|
|
||||||
|
To run the performance tests:
|
||||||
|
|
||||||
|
`npm run test:perf`
|
||||||
|
|
||||||
|
The test suite is configured to all tests localed in `e2e/tests/` ending in `*.e2e.spec.js`. For more about the e2e test suite, please see the [README](./e2e/README.md)
|
||||||
|
|
||||||
|
### Security Tests
|
||||||
|
Each commit is analyzed for known security vulnerabilities using [CodeQL](https://codeql.github.com/docs/codeql-language-guides/codeql-library-for-javascript/) and our overall security report is available on [LGTM](https://lgtm.com/projects/g/nasa/openmct/)
|
||||||
|
|
||||||
|
### Test Reporting and Code Coverage
|
||||||
|
|
||||||
|
Each test suite generates a report in CircleCI. For a complete overview of testing functionality, please see our [Circle CI Test Insights Dashboard](https://app.circleci.com/insights/github/nasa/openmct/workflows/the-nightly/overview?branch=master&reporting-window=last-30-days)
|
||||||
|
|
||||||
|
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/)
|
||||||
|
|
||||||
# Glossary
|
# Glossary
|
||||||
|
|
||||||
@ -143,3 +147,19 @@ documentation, may presume an understanding of these terms.
|
|||||||
user makes another such choice.)
|
user makes another such choice.)
|
||||||
* _namespace_: A name used to identify a persistence store. A running open MCT
|
* _namespace_: A name used to identify a persistence store. A running open MCT
|
||||||
application could potentially use multiple persistence stores, with the
|
application could potentially use multiple persistence stores, with the
|
||||||
|
|
||||||
|
## Open MCT v2.0.0
|
||||||
|
Support for our legacy bundle-based API, and the libraries that it was built on (like Angular 1.x), have now been removed entirely from this repository.
|
||||||
|
|
||||||
|
For now if you have an Open MCT application that makes use of the legacy API, [a plugin](https://github.com/nasa/openmct-legacy-plugin) is provided that bootstraps the legacy bundling mechanism and API. This plugin will not be maintained over the long term however, and the legacy support plugin will not be tested for compatibility with future versions of Open MCT. It is provided for convenience only.
|
||||||
|
|
||||||
|
### How do I know if I am using legacy API?
|
||||||
|
You might still be using legacy API if your source code
|
||||||
|
|
||||||
|
* Contains files named bundle.js, or bundle.json,
|
||||||
|
* Makes calls to `openmct.$injector()`, or `openmct.$angular`,
|
||||||
|
* Makes calls to `openmct.legacyRegistry`, `openmct.legacyExtension`, or `openmct.legacyBundle`.
|
||||||
|
|
||||||
|
|
||||||
|
### 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.
|
||||||
|
24
app.js
24
app.js
@ -12,6 +12,7 @@ const express = require('express');
|
|||||||
const app = express();
|
const app = express();
|
||||||
const fs = require('fs');
|
const fs = require('fs');
|
||||||
const request = require('request');
|
const request = require('request');
|
||||||
|
const __DEV__ = !process.env.NODE_ENV || process.env.NODE_ENV === 'development';
|
||||||
|
|
||||||
// Defaults
|
// Defaults
|
||||||
options.port = options.port || options.p || 8080;
|
options.port = options.port || options.p || 8080;
|
||||||
@ -49,14 +50,18 @@ class WatchRunPlugin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const webpack = require('webpack');
|
const webpack = require('webpack');
|
||||||
const webpackConfig = require('./webpack.dev.js');
|
let webpackConfig;
|
||||||
webpackConfig.plugins.push(new webpack.HotModuleReplacementPlugin());
|
if (__DEV__) {
|
||||||
webpackConfig.plugins.push(new WatchRunPlugin());
|
webpackConfig = require('./webpack.dev');
|
||||||
|
webpackConfig.plugins.push(new webpack.HotModuleReplacementPlugin());
|
||||||
webpackConfig.entry.openmct = [
|
webpackConfig.entry.openmct = [
|
||||||
'webpack-hot-middleware/client?reload=true',
|
'webpack-hot-middleware/client?reload=true',
|
||||||
webpackConfig.entry.openmct
|
webpackConfig.entry.openmct
|
||||||
];
|
];
|
||||||
|
webpackConfig.plugins.push(new WatchRunPlugin());
|
||||||
|
} else {
|
||||||
|
webpackConfig = require('./webpack.coverage');
|
||||||
|
}
|
||||||
|
|
||||||
const compiler = webpack(webpackConfig);
|
const compiler = webpack(webpackConfig);
|
||||||
|
|
||||||
@ -68,10 +73,12 @@ app.use(require('webpack-dev-middleware')(
|
|||||||
}
|
}
|
||||||
));
|
));
|
||||||
|
|
||||||
app.use(require('webpack-hot-middleware')(
|
if (__DEV__) {
|
||||||
|
app.use(require('webpack-hot-middleware')(
|
||||||
compiler,
|
compiler,
|
||||||
{}
|
{}
|
||||||
));
|
));
|
||||||
|
}
|
||||||
|
|
||||||
// Expose index.html for development users.
|
// Expose index.html for development users.
|
||||||
app.get('/', function (req, res) {
|
app.get('/', function (req, res) {
|
||||||
@ -82,3 +89,4 @@ app.get('/', function (req, res) {
|
|||||||
app.listen(options.port, options.host, function () {
|
app.listen(options.port, options.host, function () {
|
||||||
console.log('Open MCT application running at %s:%s', options.host, options.port);
|
console.log('Open MCT application running at %s:%s', options.host, options.port);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
// This is a Babel config that webpack.coverage.js uses in order to instrument
|
|
||||||
// code with coverage instrumentation.
|
|
||||||
const babelConfig = {
|
|
||||||
plugins: [['babel-plugin-istanbul', {
|
|
||||||
extension: ['.js', '.vue']
|
|
||||||
}]]
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = babelConfig;
|
|
17
codecov.yml
17
codecov.yml
@ -13,17 +13,16 @@ coverage:
|
|||||||
round: down
|
round: down
|
||||||
range: "66...100"
|
range: "66...100"
|
||||||
|
|
||||||
ignore:
|
flags:
|
||||||
|
unit:
|
||||||
parsers:
|
carryforward: true
|
||||||
gcov:
|
e2e-ci:
|
||||||
branch_detection:
|
carryforward: true
|
||||||
conditional: true
|
e2e-full:
|
||||||
loop: true
|
carryforward: true
|
||||||
method: false
|
|
||||||
macro: false
|
|
||||||
|
|
||||||
comment:
|
comment:
|
||||||
layout: "reach,diff,flags,files,footer"
|
layout: "reach,diff,flags,files,footer"
|
||||||
behavior: default
|
behavior: default
|
||||||
require_changes: false
|
require_changes: false
|
||||||
|
show_carryforward_flags: true
|
@ -1,4 +1,15 @@
|
|||||||
/* eslint-disable no-undef */
|
/* eslint-disable no-undef */
|
||||||
module.exports = {
|
module.exports = {
|
||||||
"extends": ["plugin:playwright/playwright-test"]
|
"extends": ["plugin:playwright/playwright-test"],
|
||||||
|
"rules": {
|
||||||
|
"playwright/max-nested-describe": ["error", { "max": 1 }]
|
||||||
|
},
|
||||||
|
"overrides": [
|
||||||
|
{
|
||||||
|
"files": ["tests/visual/*.spec.js"],
|
||||||
|
"rules": {
|
||||||
|
"playwright/no-wait-for-timeout": "off"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
};
|
};
|
||||||
|
@ -2,4 +2,5 @@ version: 2
|
|||||||
snapshot:
|
snapshot:
|
||||||
widths: [1024, 2000]
|
widths: [1024, 2000]
|
||||||
min-height: 1440 # px
|
min-height: 1440 # px
|
||||||
|
discovery:
|
||||||
|
concurrency: 2 # https://github.com/percy/cli/discussions/1067
|
||||||
|
380
e2e/README.md
Normal file
380
e2e/README.md
Normal file
@ -0,0 +1,380 @@
|
|||||||
|
# e2e testing
|
||||||
|
|
||||||
|
This document captures information specific to the e2e testing of Open MCT. For general information about testing, please see [the Open MCT README](https://github.com/nasa/openmct/blob/master/README.md#tests).
|
||||||
|
|
||||||
|
## Table of Contents
|
||||||
|
|
||||||
|
This document is designed to capture on the What, Why, and How's of writing and running e2e tests in Open MCT. Please use the built-in Github Table of Contents functionality at the top left of this page or the markup.
|
||||||
|
|
||||||
|
1. [Getting Started](#getting-started)
|
||||||
|
2. [Types of Testing](#types-of-e2e-testing)
|
||||||
|
3. [Architecture](#architecture)
|
||||||
|
|
||||||
|
## Getting Started
|
||||||
|
|
||||||
|
While our team does our best to lower the barrier to entry to working with our e2e framework and Open MCT, there is a bit of work required to get from 0 to 1 test contributed.
|
||||||
|
|
||||||
|
### Getting started with Playwright
|
||||||
|
|
||||||
|
If this is your first time ever using the Playwright framework, we recommend going through the [Getting Started Guide](https://playwright.dev/docs/next/intro) which can be completed in about 15 minutes. This will give you a concise tour of Playwright's functionality and an understanding of the official Playwright documentation which we leverage in Open MCT.
|
||||||
|
|
||||||
|
### Getting started with Open MCT's implementation of Playwright
|
||||||
|
|
||||||
|
Once you've got an understanding of Playwright, you'll need a baseline understanding of Open MCT:
|
||||||
|
|
||||||
|
1. Follow the steps [Building and Running Open MCT Locally](../README.md#building-and-running-open-mct-locally)
|
||||||
|
2. Once you're serving Open MCT locally, create a 'Display Layout' object. Save it.
|
||||||
|
3. Create a 'Plot' Object (e.g.: 'Stacked Plot')
|
||||||
|
4. Create an Example Telemetry Object (e.g.: 'Sine Wave Generator')
|
||||||
|
5. Expand the Tree and note the hierarchy of objects which were created.
|
||||||
|
6. Navigate to the Demo Display Layout Object to edit and modify the embedded plot.
|
||||||
|
7. Modify the embedded plot with Telemetry Data.
|
||||||
|
|
||||||
|
What you've created is a display which mimics the display that a mission control operator might use to understand and model telemetry data.
|
||||||
|
|
||||||
|
Recreate the steps above with Playwright's codegen tool:
|
||||||
|
|
||||||
|
1. `npm run start` in a terminal window to serve Open MCT locally
|
||||||
|
2. `npx @playwright/test install` to install playwright and dependencies
|
||||||
|
3. Open another terminal window and start the Playwright codegen application `npx playwright codegen`
|
||||||
|
4. Navigate the browser to `http://localhost:8080`
|
||||||
|
5. Click the Create button and notice how your actions in the browser are being recorded in the Playwright Inspector
|
||||||
|
6. Continue through the steps 2-6 above
|
||||||
|
|
||||||
|
What you've created is an automated test which mimics the creation of a mission control display.
|
||||||
|
|
||||||
|
Next, you should walk through our implementation of Playwright in Open MCT:
|
||||||
|
|
||||||
|
1. Close any terminals which are serving up a local instance of Open MCT
|
||||||
|
2. Run our 'Getting Started' test in debug mode with `npm run test:e2e:local -- exampleTemplate --debug`
|
||||||
|
3. Step through each test step in the Playwright Inspector to see how we leverage Playwright's capabilities to test Open MCT
|
||||||
|
|
||||||
|
## 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:
|
||||||
|
|
||||||
|
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_.
|
||||||
|
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.**
|
||||||
|
|
||||||
|
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".
|
||||||
|
|
||||||
|
We do not want to interleave visual and functional testing inside the same suite because visual test verification of correctness must happen with a 3rd party service. This service is not available when executing these tests in other contexts (i.e. VIPER).
|
||||||
|
|
||||||
|
### Functional Testing
|
||||||
|
|
||||||
|
The bulk of our e2e coverage lies in "functional" test coverage which verifies that Open MCT is functionally correct as well as defining _how we expect it to behave_. This enables us to test the application exactly as a user would, while prescribing exactly how a user can interact with the application via a web browser.
|
||||||
|
|
||||||
|
### Visual Testing
|
||||||
|
|
||||||
|
Visual Testing is an essential part of our e2e strategy as it ensures that the application _appears_ correctly to a user while it compliments the functional e2e suite. It would be impractical to make thousands of assertions functional assertions on the look and feel of the application. Visual testing is interested in getting the DOM into a specified state and then comparing that it has not changed against a baseline.
|
||||||
|
|
||||||
|
For a better understanding of the visual issues which affect Open MCT, please see our bug tracker with the `label:visual` filter applied [here](https://github.com/nasa/openmct/issues?q=label%3Abug%3Avisual+)
|
||||||
|
To read about how to write a good visual test, please see [How to write a great Visual Test](#how-to-write-a-great-visual-test).
|
||||||
|
|
||||||
|
`npm run test:e2e:visual` will run all of the visual tests against a local instance of Open MCT. If no `PERCY_TOKEN` API key is found in the terminal or command line environment variables, no visual comparisons will be made.
|
||||||
|
|
||||||
|
#### Percy.io
|
||||||
|
|
||||||
|
To make this possible, we're leveraging a 3rd party service, [Percy](https://percy.io/). This service maintains a copy of all changes, users, scm-metadata, and baselines to verify that the application looks and feels the same _unless approved by a Open MCT developer_. To request a Percy API token, please reach out to the Open MCT Dev team on GitHub. For more information, please see the official [Percy documentation](https://docs.percy.io/docs/visual-testing-basics)
|
||||||
|
|
||||||
|
### (Advanced) Snapshot Testing
|
||||||
|
|
||||||
|
Snapshot testing is very similar to visual testing but allows us to be more precise in detecting change without relying on a 3rd party service. Unfortuantely, this precision requires advanced test setup and teardown and so we're using this pattern as a last resort.
|
||||||
|
|
||||||
|
To give an example, if a _single_ visual test assertion for an Overlay plot is run through multiple DOM rendering engines at various viewports to see how the Plot looks. If that same test were run as a snapshot test, it could only be executed against a single browser, on a single platform (ubuntu docker container).
|
||||||
|
|
||||||
|
Read more about [Playwright Snapshots](https://playwright.dev/docs/test-snapshots)
|
||||||
|
|
||||||
|
#### Open MCT's implementation
|
||||||
|
|
||||||
|
- Our Snapshot tests receive a `@snapshot` tag.
|
||||||
|
- Snapshots need to be executed within the official Playwright container to ensure we're using the exact rendering platform in CI and locally.
|
||||||
|
|
||||||
|
```sh
|
||||||
|
docker run --rm --network host -v $(pwd):/work/ -w /work/ -it mcr.microsoft.com/playwright:[GET THIS VERSION FROM OUR CIRCLECI CONFIG FILE]-focal /bin/bash
|
||||||
|
npm install
|
||||||
|
npx playwright test --config=e2e/playwright-ci.config.js --project=chrome --grep @snapshot
|
||||||
|
```
|
||||||
|
|
||||||
|
### (WIP) Updating Snapshots
|
||||||
|
|
||||||
|
When the `@snapshot` tests fail, they will need to be evaluated to see if the failure is an acceptable change or
|
||||||
|
|
||||||
|
## Performance Testing
|
||||||
|
|
||||||
|
The open source performance tests function mostly as a contract for the locator logic, functionality, and assumptions will work in our downstream, closed source test suites.
|
||||||
|
|
||||||
|
They're found under `./e2e/tests/performance` and are to be executed with the following npm script:
|
||||||
|
|
||||||
|
`npm run test:perf`
|
||||||
|
|
||||||
|
These tests are expected to become blocking and gating with assertions as we extend the capabilities of Playwright.
|
||||||
|
|
||||||
|
## Test Architecture and CI
|
||||||
|
|
||||||
|
### Architecture (TODO)
|
||||||
|
|
||||||
|
### File Structure
|
||||||
|
|
||||||
|
Our file structure follows the type of type of testing being excercised at the e2e layer and files containing test suites which matcher application behavior or our `src` and `example` layout. This area is not well refined as we figure out what works best for closed source and downstream projects. This may change altogether if we move `e2e` to it's own npm package.
|
||||||
|
|
||||||
|
- `./helper` - contains helper functions or scripts which are leveraged directly within the testsuites. i.e. non-default plugin scripts injected into DOM
|
||||||
|
- `./test-data` - contains test data which is leveraged or generated in the functional, performance, or visual test suites. i.e. localStorage data
|
||||||
|
- `./tests/functional` - the bulk of the tests are contained within this folder to verify the functionality of open mct
|
||||||
|
- `./tests/functional/example/` - tests which specifically verify the example plugins
|
||||||
|
- `./tests/functional/plugins/` - tests which loosely test each plugin. This folder is the most likely to change. Note: some @snapshot tests are still contained within this structure
|
||||||
|
- `./tests/framework/` - tests which verify that our testframework functionality and assumptions will continue to work based on further refactoring or playwright version changes
|
||||||
|
- `./tests/performance/` - performance tests
|
||||||
|
- `./tests/visual/` - Visual tests
|
||||||
|
- `./appActions.js` - Contains common fixtures which can be leveraged by testcase authors to quickly move through the application when writing new tests.
|
||||||
|
- `./baseFixture.js` - Contains base fixtures which only extend default `@playwright/test` functionality. The goal is to remove these fixtures as native Playwright APIs improve.
|
||||||
|
|
||||||
|
Our functional tests end in `*.e2e.spec.js`, visual tests in `*.visual.spec.js` and performance tests in `*.perf.spec.js`.
|
||||||
|
|
||||||
|
### Configuration
|
||||||
|
|
||||||
|
Where possible, we try to run Open MCT without modification or configuration change so that the Open MCT doesn't fail exclusively in "test mode" or in "production mode".
|
||||||
|
|
||||||
|
Open MCT is leveraging the [config file](https://playwright.dev/docs/test-configuration) pattern to describe the capabilities of Open MCT e2e _where_ it's run
|
||||||
|
|
||||||
|
- `./playwright-ci.config.js` - Used when running in CI or to debug CI issues locally
|
||||||
|
- `./playwright-local.config.js` - Used when running locally
|
||||||
|
- `./playwright-performance.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
|
||||||
|
|
||||||
|
#### Test Tags
|
||||||
|
|
||||||
|
Test tags are a great way of organizing tests outside of a file structure. To learn more see the official documentation [here](https://playwright.dev/docs/test-annotations#tag-tests).
|
||||||
|
|
||||||
|
Current list of test tags:
|
||||||
|
|
||||||
|
- `@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).
|
||||||
|
- `@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 app.js.
|
||||||
|
- `@localStorage` - Captures or generates session storage to manipulate browser state. Useful for excluding in tests which require a persistent backend (i.e. CouchDB).
|
||||||
|
- `@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.
|
||||||
|
- `@2p` - Indicates that multiple users are involved, or multiple tabs/pages are used. Useful for testing multi-user interactivity.
|
||||||
|
|
||||||
|
### Continuous Integration
|
||||||
|
|
||||||
|
The cheapest time to catch a bug is pre-merge. Unfortuantely, this is the most expensive time to run all of the tests since each merge event can consist of hundreds of commits. For this reason, we're selective in _what we run_ as much as _when we run it_.
|
||||||
|
|
||||||
|
We leverage CircleCI to run tests against each commit and inject the Test Reports which are generated by Playwright so that they team can keep track of flaky and [historical test trends](https://app.circleci.com/insights/github/nasa/openmct/workflows/overall-circleci-commit-status/tests?branch=master&reporting-window=last-30-days)
|
||||||
|
|
||||||
|
We leverage Github Actions / Workflows to execute tests as it gives us the ability to run against multiple operating systems with greater control over git event triggers (i.e. Run on a PR Comment event).
|
||||||
|
|
||||||
|
Our CI environment consists of 3 main modes of operation:
|
||||||
|
|
||||||
|
#### 1. Per-Commit Testing
|
||||||
|
|
||||||
|
CircleCI
|
||||||
|
|
||||||
|
- Stable e2e tests against ubuntu and chrome
|
||||||
|
- Performance tests against ubuntu and chrome
|
||||||
|
- e2e tests are linted
|
||||||
|
|
||||||
|
#### 2. Per-Merge Testing
|
||||||
|
|
||||||
|
Github Actions / Workflow
|
||||||
|
|
||||||
|
- Full suite against all browsers/projects. Triggered with Github Label Event 'pr:e2e'
|
||||||
|
- Visual Tests. Triggered with Github Label Event 'pr:visual'
|
||||||
|
|
||||||
|
#### 3. Scheduled / Batch Testing
|
||||||
|
|
||||||
|
Nightly Testing in Circle CI
|
||||||
|
|
||||||
|
- Full e2e suite against ubuntu and chrome
|
||||||
|
- Performance tests against ubuntu and chrome
|
||||||
|
|
||||||
|
Github Actions / Workflow
|
||||||
|
|
||||||
|
- Visual Test baseline generation.
|
||||||
|
|
||||||
|
#### Parallelism and Fast Feedback
|
||||||
|
|
||||||
|
In order to provide fast feedback in the Per-Commit context, we try to keep total test feedback at 5 minutes or less. That is to say, A developer should have a pass/fail result in under 5 minutes.
|
||||||
|
|
||||||
|
Playwright has native support for semi-intelligent sharding. Read about it [here](https://playwright.dev/docs/test-parallel#shard-tests-between-multiple-machines).
|
||||||
|
|
||||||
|
We will be adjusting the parallelization of the Per-Commit tests to keep below the 5 minute total runtime threshold.
|
||||||
|
|
||||||
|
In addition to the Parallelization of Test Runners (Sharding), we're also running two concurrent threads on every Shard. This is the functional limit of what CircelCI Agents can support from a memory and CPU resource constraint.
|
||||||
|
|
||||||
|
So for every commit, Playwright is effectively running 4 x 2 concurrent browsercontexts to keep the overall runtime to a miminum.
|
||||||
|
|
||||||
|
At the same time, we don't want to waste CI resources on parallel runs, so we've configured each shard to fail after 5 test failures. Test failure logs are recorded and stored to allow fast triage.
|
||||||
|
|
||||||
|
#### Test Promotion
|
||||||
|
|
||||||
|
In order to maintain fast and reliable feedback, tests go through a promotion process. All new test cases or test suites must be labeled with the `@unstable` annotation. The Open MCT dev team runs these unstable tests in our private repos to ensure they work downstream and are reliable.
|
||||||
|
|
||||||
|
To run the stable tests, use the ```npm run test:e2e:stable``` command. To run the new and flaky tests, use the ```npm run test:e2e:unstable``` command.
|
||||||
|
|
||||||
|
A testcase and testsuite are to be unmarked as @unstable when:
|
||||||
|
|
||||||
|
1. They run as part of "full" run 5 times without failure.
|
||||||
|
2. They've been by a Open MCT Developer 5 times in the closed source repo without failure.
|
||||||
|
|
||||||
|
### Cross-browser and Cross-operating system
|
||||||
|
|
||||||
|
#### **What's supported:**
|
||||||
|
|
||||||
|
We are leveraging the `browserslist` project to declare our supported list of browsers.
|
||||||
|
|
||||||
|
#### **Where it's tested:**
|
||||||
|
|
||||||
|
We lint on `browserslist` to ensure that we're not implementing deprecated browser APIs and are aware of browser API improvements over time.
|
||||||
|
|
||||||
|
We also have the need to execute our e2e tests across this published list of browsers. Our browsers and browser version matrix is found inside of our `./playwright-*.config.js`, but mostly follows in order of bleeding edge to stable:
|
||||||
|
|
||||||
|
- `playwright-chromium channel:beta`
|
||||||
|
- A beta version of Chromium from official chromium channels. As close to the bleeding edge as we can get.
|
||||||
|
- `playwright-chromium`
|
||||||
|
- A stable version of Chromium from the official chromium channels. This is always at least 1 version ahead of desktop chrome.
|
||||||
|
- `playwright-chrome`
|
||||||
|
- The stable channel of Chrome from the official chrome channels. This is always 2 versions behind chromium.
|
||||||
|
|
||||||
|
#### **Mobile**
|
||||||
|
|
||||||
|
We have the Mission-need to support iPad. To run our iPad suite, please see our `playwright-*.config.js` with the 'iPad' project.
|
||||||
|
|
||||||
|
#### **Skipping or executing tests based on browser, os, and/os browser version:**
|
||||||
|
|
||||||
|
Conditionally skipping tests based on browser (**RECOMMENDED**):
|
||||||
|
|
||||||
|
```js
|
||||||
|
test('Can adjust image brightness/contrast by dragging the sliders', async ({ page, browserName }) => {
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip(browserName === 'firefox', 'This test needs to be updated to work with firefox');
|
||||||
|
|
||||||
|
// ...
|
||||||
|
```
|
||||||
|
|
||||||
|
Conditionally skipping tests based on OS:
|
||||||
|
|
||||||
|
```js
|
||||||
|
test('Can adjust image brightness/contrast by dragging the sliders', async ({ page }) => {
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip(process.platform === 'darwin', 'This test needs to be updated to work with MacOS');
|
||||||
|
|
||||||
|
// ...
|
||||||
|
```
|
||||||
|
|
||||||
|
Skipping based on browser version (Rarely used): <https://github.com/microsoft/playwright/discussions/17318>
|
||||||
|
|
||||||
|
## Test Design, Best Practices, and Tips & Tricks
|
||||||
|
|
||||||
|
### Test Design (TODO)
|
||||||
|
|
||||||
|
- How to make tests robust to function in other contexts (VISTA, VIPER, etc.)
|
||||||
|
- Leverage the use of `appActions.js` methods such as `createDomainObjectWithDefaults()`
|
||||||
|
- How to make tests faster and more resilient
|
||||||
|
- When possible, navigate directly by URL
|
||||||
|
- Leverage `await page.goto('./', { waitUntil: 'networkidle' });`
|
||||||
|
- Avoid repeated setup to test to test a single assertion. Write longer tests with multiple soft assertions.
|
||||||
|
|
||||||
|
### How to write a great test (TODO)
|
||||||
|
|
||||||
|
#### How to write a great visual test (TODO)
|
||||||
|
|
||||||
|
### Best Practices
|
||||||
|
|
||||||
|
For now, our best practices exist as self-tested, living documentation in our [exampleTemplate.e2e.spec.js](./tests/framework/exampleTemplate.e2e.spec.js) file.
|
||||||
|
|
||||||
|
### Tips & Tricks (TODO)
|
||||||
|
|
||||||
|
The following contains a list of tips and tricks which don't exactly fit into a FAQ or Best Practices doc.
|
||||||
|
|
||||||
|
- Working with multiple pages
|
||||||
|
There are instances where multiple browser pages will need to be opened to verify multi-page or multi-tab application behavior.
|
||||||
|
|
||||||
|
### Reporting
|
||||||
|
|
||||||
|
Test Reporting is done through official Playwright reporters and the CI Systems which execute them.
|
||||||
|
|
||||||
|
We leverage the following official Playwright reporters:
|
||||||
|
|
||||||
|
- HTML
|
||||||
|
- junit
|
||||||
|
- github annotations
|
||||||
|
- Tracefile
|
||||||
|
- Screenshots
|
||||||
|
|
||||||
|
When running the tests locally with the `npm run test:local` command, the html report will open automatically on failure. Inside this HTML report will be a complete summary of the finished tests. If the tests failed, you'll see embedded links to screenshot failure, execution logs, and the Tracefile.
|
||||||
|
|
||||||
|
When looking at the reports run in CI, you'll leverage this same HTML Report which is hosted either in CircleCI or Github Actions as a build artifact.
|
||||||
|
|
||||||
|
### 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:
|
||||||
|
|
||||||
|
```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.
|
||||||
|
|
||||||
|
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.
|
||||||
|
|
||||||
|
This e2e coverage is combined with our unit test report to give a comprehensive (if flawed) view of line coverage.
|
||||||
|
|
||||||
|
## Other
|
||||||
|
|
||||||
|
### About e2e testing
|
||||||
|
|
||||||
|
e2e testing is an industry-standard approach to automating the testing of web-based UIs such as Open MCT. Broadly speaking, e2e tests differentiate themselves from unit tests by preferring replication of real user interactions over execution of raw JavaScript functions.
|
||||||
|
|
||||||
|
Historically, the abstraction necessary to replicate real user behavior meant that:
|
||||||
|
|
||||||
|
- e2e tests were "expensive" due to how much code each test executed. The closer a test replicates the user, the more code is needed run during test execution. Unit tests could run smaller units of code more efficiently.
|
||||||
|
- e2e tests were flaky due to network conditions or the underlying protocols associated with testing a browser.
|
||||||
|
- e2e frameworks relied on a browser communication standard which lacked the observability and controls necessary needed to reach the code paths possible with unit and integration tests.
|
||||||
|
- e2e frameworks provided insufficient debug information on test failure
|
||||||
|
|
||||||
|
However, as the web ecosystem has matured to the point where mission-critical UIs can be written for the web (Open MCT), the e2e testing tools have matured as well. There are now fewer "trade-offs" when choosing to write an e2e test over any other type of test.
|
||||||
|
|
||||||
|
Modern e2e frameworks:
|
||||||
|
|
||||||
|
- Bypass the surface layer of the web-application-under-test and use a raw debugging protocol to observe and control application and browser state.
|
||||||
|
- These new browser-internal protocols enable near-instant, bi-directional communication between test code and the browser, speeding up test execution and making the tests as reliable as the application itself.
|
||||||
|
- Provide test debug tooling which enables developers to pinpoint failure
|
||||||
|
|
||||||
|
Furthermore, the abstraction necessary to run e2e tests as a user enables them to be extended to run within a variety of contexts. This matches the extensible design of Open MCT.
|
||||||
|
|
||||||
|
A single e2e test in Open MCT is extended to run:
|
||||||
|
|
||||||
|
- Against a matrix of browser versions.
|
||||||
|
- Against a matrix of OS platforms.
|
||||||
|
- Against a local development version of Open MCT.
|
||||||
|
- A version of Open MCT loaded as a dependency (VIPER, VISTA, etc)
|
||||||
|
- Against a variety of data sources or telemetry endpoints.
|
||||||
|
|
||||||
|
### Why Playwright?
|
||||||
|
|
||||||
|
[Playwright](https://playwright.dev/) was chosen as our e2e framework because it solves a few VIPER Mission needs:
|
||||||
|
|
||||||
|
1. First-class support for Automated Performance Testing
|
||||||
|
2. Official Chrome, Chrome Canary, and iPad Capabilities
|
||||||
|
3. Support for Browserless.io to run tests in a "hermetically sealed" environment
|
||||||
|
4. Ability to generate code coverage reports
|
||||||
|
|
||||||
|
### FAQ
|
||||||
|
|
||||||
|
- How does this help NASA missions?
|
||||||
|
- When should I write an e2e test instead of a unit test?
|
||||||
|
- When should I write a functional vs visual test?
|
||||||
|
- How is Open MCT extending default Playwright functionality?
|
||||||
|
- What about Component Testing?
|
||||||
|
|
||||||
|
### Troubleshooting
|
||||||
|
|
||||||
|
- 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...'
|
||||||
|
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```
|
336
e2e/appActions.js
Normal file
336
e2e/appActions.js
Normal file
@ -0,0 +1,336 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The fixtures in this file are to be used to consolidate common actions performed by the
|
||||||
|
* various test suites. The goal is only to avoid duplication of code across test suites and not to abstract
|
||||||
|
* away the underlying functionality of the application. For more about the App Action pattern, see /e2e/README.md)
|
||||||
|
*
|
||||||
|
* For example, if two functions are nearly identical in
|
||||||
|
* timer.e2e.spec.js and notebook.e2e.spec.js, that function should be generalized and moved into this file.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Defines parameters to be used in the creation of a domain object.
|
||||||
|
* @typedef {Object} CreateObjectOptions
|
||||||
|
* @property {string} type the type of domain object to create (e.g.: "Sine Wave Generator").
|
||||||
|
* @property {string} [name] the desired name of the created domain object.
|
||||||
|
* @property {string | import('../src/api/objects/ObjectAPI').Identifier} [parent] the Identifier or uuid of the parent object.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Contains information about the newly created domain object.
|
||||||
|
* @typedef {Object} CreatedObjectInfo
|
||||||
|
* @property {string} name the name of the created object
|
||||||
|
* @property {string} uuid the uuid of the created object
|
||||||
|
* @property {string} url the relative url to the object (for use with `page.goto()`)
|
||||||
|
*/
|
||||||
|
|
||||||
|
const Buffer = require('buffer').Buffer;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This common function creates a domain object with the default options. It is the preferred way of creating objects
|
||||||
|
* in the e2e suite when uninterested in properties of the objects themselves.
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {CreateObjectOptions} options
|
||||||
|
* @returns {Promise<CreatedObjectInfo>} An object containing information about the newly created domain object.
|
||||||
|
*/
|
||||||
|
async function createDomainObjectWithDefaults(page, { type, name, parent = 'mine' }) {
|
||||||
|
const parentUrl = await getHashUrlToDomainObject(page, parent);
|
||||||
|
|
||||||
|
// Navigate to the parent object. This is necessary to create the object
|
||||||
|
// in the correct location, such as a folder, layout, or plot.
|
||||||
|
await page.goto(`${parentUrl}?hideTree=true`);
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
|
||||||
|
//Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click the object specified by 'type'
|
||||||
|
await page.click(`li:text("${type}")`);
|
||||||
|
|
||||||
|
// Modify the name input field of the domain object to accept 'name'
|
||||||
|
if (name) {
|
||||||
|
const nameInput = page.locator('form[name="mctForm"] .first input[type="text"]');
|
||||||
|
await nameInput.fill("");
|
||||||
|
await nameInput.fill(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Click OK button and wait for Navigate event
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForLoadState(),
|
||||||
|
page.click('[aria-label="Save"]'),
|
||||||
|
// Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Wait until the URL is updated
|
||||||
|
await page.waitForURL(`**/${parent}/*`);
|
||||||
|
const uuid = await getFocusedObjectUuid(page);
|
||||||
|
const objectUrl = await getHashUrlToDomainObject(page, uuid);
|
||||||
|
|
||||||
|
if (await _isInEditMode(page, uuid)) {
|
||||||
|
// Save (exit edit mode)
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('li[title="Save and Finish Editing"]').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
name: name || `Unnamed ${type}`,
|
||||||
|
uuid: uuid,
|
||||||
|
url: objectUrl
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} name
|
||||||
|
*/
|
||||||
|
async function expandTreePaneItemByName(page, name) {
|
||||||
|
const treePane = page.locator('#tree-pane');
|
||||||
|
const treeItem = treePane.locator(`role=treeitem[expanded=false][name=/${name}/]`);
|
||||||
|
const expandTriangle = treeItem.locator('.c-disclosure-triangle');
|
||||||
|
await expandTriangle.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a Plan object from JSON with the provided options.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {*} options
|
||||||
|
* @returns {Promise<CreatedObjectInfo>} An object containing information about the newly created domain object.
|
||||||
|
*/
|
||||||
|
async function createPlanFromJSON(page, { name, json, parent = 'mine' }) {
|
||||||
|
const parentUrl = await getHashUrlToDomainObject(page, parent);
|
||||||
|
|
||||||
|
// Navigate to the parent object. This is necessary to create the object
|
||||||
|
// in the correct location, such as a folder, layout, or plot.
|
||||||
|
await page.goto(`${parentUrl}?hideTree=true`);
|
||||||
|
|
||||||
|
//Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click 'Plan' menu option
|
||||||
|
await page.click(`li:text("Plan")`);
|
||||||
|
|
||||||
|
// Modify the name input field of the domain object to accept 'name'
|
||||||
|
if (name) {
|
||||||
|
const nameInput = page.locator('form[name="mctForm"] .first input[type="text"]');
|
||||||
|
await nameInput.fill("");
|
||||||
|
await nameInput.fill(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Upload buffer from memory
|
||||||
|
await page.locator('input#fileElem').setInputFiles({
|
||||||
|
name: 'plan.txt',
|
||||||
|
mimeType: 'text/plain',
|
||||||
|
buffer: Buffer.from(JSON.stringify(json))
|
||||||
|
});
|
||||||
|
|
||||||
|
// Click OK button and wait for Navigate event
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForLoadState(),
|
||||||
|
page.click('[aria-label="Save"]'),
|
||||||
|
// Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Wait until the URL is updated
|
||||||
|
await page.waitForURL(`**/mine/*`);
|
||||||
|
const uuid = await getFocusedObjectUuid(page);
|
||||||
|
const objectUrl = await getHashUrlToDomainObject(page, uuid);
|
||||||
|
|
||||||
|
return {
|
||||||
|
uuid,
|
||||||
|
name,
|
||||||
|
url: objectUrl
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Open the given `domainObject`'s context menu from the object tree.
|
||||||
|
* Expands the path to the object and scrolls to it if necessary.
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} url the url to the object
|
||||||
|
*/
|
||||||
|
async function openObjectTreeContextMenu(page, url) {
|
||||||
|
await page.goto(url);
|
||||||
|
await page.click('button[title="Show selected item in tree"]');
|
||||||
|
await page.locator('.is-navigated-object').click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the UUID of the currently focused object by parsing the current URL
|
||||||
|
* and returning the last UUID in the path.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @returns {Promise<string>} the uuid of the focused object
|
||||||
|
*/
|
||||||
|
async function getFocusedObjectUuid(page) {
|
||||||
|
const UUIDv4Regexp = /[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}/gi;
|
||||||
|
const focusedObjectUuid = await page.evaluate((regexp) => {
|
||||||
|
return window.location.href.split('?')[0].match(regexp).at(-1);
|
||||||
|
}, UUIDv4Regexp);
|
||||||
|
|
||||||
|
return focusedObjectUuid;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the hashUrl to the domainObject given its uuid.
|
||||||
|
* Useful for directly navigating to the given domainObject.
|
||||||
|
*
|
||||||
|
* URLs returned will be of the form `'./browse/#/mine/<uuid0>/<uuid1>/...'`
|
||||||
|
*
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} uuid the uuid of the object to get the url for
|
||||||
|
* @returns {Promise<string>} the url of the object
|
||||||
|
*/
|
||||||
|
async function getHashUrlToDomainObject(page, uuid) {
|
||||||
|
const hashUrl = await page.evaluate(async (objectUuid) => {
|
||||||
|
const path = await window.openmct.objects.getOriginalPath(objectUuid);
|
||||||
|
let url = './#/browse/' + [...path].reverse()
|
||||||
|
.map((object) => window.openmct.objects.makeKeyString(object.identifier))
|
||||||
|
.join('/');
|
||||||
|
|
||||||
|
// Drop the vestigial '/ROOT' if it exists
|
||||||
|
if (url.includes('/ROOT')) {
|
||||||
|
url = url.split('/ROOT').join('');
|
||||||
|
}
|
||||||
|
|
||||||
|
return url;
|
||||||
|
}, uuid);
|
||||||
|
|
||||||
|
return hashUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utilizes the OpenMCT API to detect if the given object has an active transaction (is in Edit mode).
|
||||||
|
* @private
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string | import('../src/api/objects/ObjectAPI').Identifier} identifier
|
||||||
|
* @return {Promise<boolean>} true if the object has an active transaction, false otherwise
|
||||||
|
*/
|
||||||
|
async function _isInEditMode(page, identifier) {
|
||||||
|
// eslint-disable-next-line no-return-await
|
||||||
|
return await page.evaluate((objectIdentifier) => window.openmct.objects.isTransactionActive(objectIdentifier), identifier);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the time conductor mode to either fixed timespan or realtime mode.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {boolean} [isFixedTimespan=true] true for fixed timespan mode, false for realtime mode; default is true
|
||||||
|
*/
|
||||||
|
async function setTimeConductorMode(page, isFixedTimespan = true) {
|
||||||
|
// Click 'mode' button
|
||||||
|
await page.locator('.c-mode-button').click();
|
||||||
|
|
||||||
|
// Switch time conductor mode
|
||||||
|
if (isFixedTimespan) {
|
||||||
|
await page.locator('data-testid=conductor-modeOption-fixed').click();
|
||||||
|
} else {
|
||||||
|
await page.locator('data-testid=conductor-modeOption-realtime').click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the time conductor to fixed timespan mode
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function setFixedTimeMode(page) {
|
||||||
|
await setTimeConductorMode(page, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the time conductor to realtime mode
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function setRealTimeMode(page) {
|
||||||
|
await setTimeConductorMode(page, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} OffsetValues
|
||||||
|
* @property {string | undefined} hours
|
||||||
|
* @property {string | undefined} mins
|
||||||
|
* @property {string | undefined} secs
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the values (hours, mins, secs) for the TimeConductor offsets when in realtime mode
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {OffsetValues} offset
|
||||||
|
* @param {import('@playwright/test').Locator} offsetButton
|
||||||
|
*/
|
||||||
|
async function setTimeConductorOffset(page, {hours, mins, secs}, offsetButton) {
|
||||||
|
await offsetButton.click();
|
||||||
|
|
||||||
|
if (hours) {
|
||||||
|
await page.fill('.pr-time-controls__hrs', hours);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mins) {
|
||||||
|
await page.fill('.pr-time-controls__mins', mins);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (secs) {
|
||||||
|
await page.fill('.pr-time-controls__secs', secs);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Click the check button
|
||||||
|
await page.locator('.pr-time__buttons .icon-check').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the values (hours, mins, secs) for the start time offset when in realtime mode
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {OffsetValues} offset
|
||||||
|
*/
|
||||||
|
async function setStartOffset(page, offset) {
|
||||||
|
const startOffsetButton = page.locator('data-testid=conductor-start-offset-button');
|
||||||
|
await setTimeConductorOffset(page, offset, startOffsetButton);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the values (hours, mins, secs) for the end time offset when in realtime mode
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {OffsetValues} offset
|
||||||
|
*/
|
||||||
|
async function setEndOffset(page, offset) {
|
||||||
|
const endOffsetButton = page.locator('data-testid=conductor-end-offset-button');
|
||||||
|
await setTimeConductorOffset(page, offset, endOffsetButton);
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
module.exports = {
|
||||||
|
createDomainObjectWithDefaults,
|
||||||
|
expandTreePaneItemByName,
|
||||||
|
createPlanFromJSON,
|
||||||
|
openObjectTreeContextMenu,
|
||||||
|
getHashUrlToDomainObject,
|
||||||
|
getFocusedObjectUuid,
|
||||||
|
setFixedTimeMode,
|
||||||
|
setRealTimeMode,
|
||||||
|
setStartOffset,
|
||||||
|
setEndOffset
|
||||||
|
};
|
174
e2e/baseFixtures.js
Normal file
174
e2e/baseFixtures.js
Normal file
@ -0,0 +1,174 @@
|
|||||||
|
/* eslint-disable no-undef */
|
||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 file is dedicated to extending the base functionality of the `@playwright/test` framework.
|
||||||
|
* The functions in this file should be viewed as temporary or a shim to be removed as the RFEs in
|
||||||
|
* the Playwright GitHub repo are implemented. Functions which serve those RFEs are marked with corresponding
|
||||||
|
* GitHub issues.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const base = require('@playwright/test');
|
||||||
|
const { expect } = base;
|
||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
const { v4: uuid } = require('uuid');
|
||||||
|
const sinon = require('sinon');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes a `ConsoleMessage` and returns a formatted string. Used to enable console log error detection.
|
||||||
|
* @see {@link https://github.com/microsoft/playwright/discussions/11690 Github Discussion}
|
||||||
|
* @private
|
||||||
|
* @param {import('@playwright/test').ConsoleMessage} msg
|
||||||
|
* @returns {String} formatted string with message type, text, url, and line and column numbers
|
||||||
|
*/
|
||||||
|
function _consoleMessageToString(msg) {
|
||||||
|
const { url, lineNumber, columnNumber } = msg.location();
|
||||||
|
|
||||||
|
return `[${msg.type()}] ${msg.text()} at (${url} ${lineNumber}:${columnNumber})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wait for all animations within the given element and subtrees to finish. Useful when
|
||||||
|
* verifying that css transitions have completed.
|
||||||
|
* @see {@link https://github.com/microsoft/playwright/issues/15660 Github RFE}
|
||||||
|
* @param {import('@playwright/test').Locator} locator
|
||||||
|
* @return {Promise<Animation[]>}
|
||||||
|
*/
|
||||||
|
function waitForAnimations(locator) {
|
||||||
|
return locator
|
||||||
|
.evaluate((element) =>
|
||||||
|
Promise.all(
|
||||||
|
element
|
||||||
|
.getAnimations({ subtree: true })
|
||||||
|
.map((animation) => animation.finished)));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is part of our codecoverage shim.
|
||||||
|
* @see {@link https://github.com/mxschmitt/playwright-test-coverage Github Example Project}
|
||||||
|
* @constant {string}
|
||||||
|
*/
|
||||||
|
const istanbulCLIOutput = path.join(process.cwd(), '.nyc_output');
|
||||||
|
|
||||||
|
exports.test = base.test.extend({
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
* Usage:
|
||||||
|
* ```
|
||||||
|
* test.use({
|
||||||
|
* clockOptions: {
|
||||||
|
* now: 0,
|
||||||
|
* shouldAdvanceTime: true
|
||||||
|
* ```
|
||||||
|
* If clockOptions are provided, will override the default clock with fake timers provided by SinonJS.
|
||||||
|
*
|
||||||
|
* Default: `undefined`
|
||||||
|
*
|
||||||
|
* @see {@link https://github.com/microsoft/playwright/issues/6347 Github RFE}
|
||||||
|
* @see {@link https://github.com/sinonjs/fake-timers/#var-clock--faketimersinstallconfig SinonJS FakeTimers Config}
|
||||||
|
*/
|
||||||
|
clockOptions: [undefined, { option: true }],
|
||||||
|
overrideClock: [async ({ context, clockOptions }, use) => {
|
||||||
|
if (clockOptions !== undefined) {
|
||||||
|
await context.addInitScript({
|
||||||
|
path: path.join(__dirname, '../', './node_modules/sinon/pkg/sinon.js')
|
||||||
|
});
|
||||||
|
await context.addInitScript((options) => {
|
||||||
|
window.__clock = sinon.useFakeTimers(options);
|
||||||
|
}, clockOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
await use(context);
|
||||||
|
}, {
|
||||||
|
auto: true,
|
||||||
|
scope: 'test'
|
||||||
|
}],
|
||||||
|
/**
|
||||||
|
* Extends the base context class to add codecoverage shim.
|
||||||
|
* @see {@link https://github.com/mxschmitt/playwright-test-coverage Github Project}
|
||||||
|
*/
|
||||||
|
context: async ({ context }, use) => {
|
||||||
|
await context.addInitScript(() =>
|
||||||
|
window.addEventListener('beforeunload', () =>
|
||||||
|
(window).collectIstanbulCoverage(JSON.stringify((window).__coverage__))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
await fs.promises.mkdir(istanbulCLIOutput, { recursive: true });
|
||||||
|
await context.exposeFunction('collectIstanbulCoverage', (coverageJSON) => {
|
||||||
|
if (coverageJSON) {
|
||||||
|
fs.writeFileSync(path.join(istanbulCLIOutput, `playwright_coverage_${uuid()}.json`), coverageJSON);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
await use(context);
|
||||||
|
for (const page of context.pages()) {
|
||||||
|
await page.evaluate(() => (window).collectIstanbulCoverage(JSON.stringify((window).__coverage__)));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* If true, will assert against any console.error calls that occur during the test. Assertions occur
|
||||||
|
* during test teardown (after the test has completed).
|
||||||
|
*
|
||||||
|
* Default: `true`
|
||||||
|
*/
|
||||||
|
failOnConsoleError: [true, { option: true }],
|
||||||
|
/**
|
||||||
|
* Extends the base page class to enable console log error detection.
|
||||||
|
* @see {@link https://github.com/microsoft/playwright/discussions/11690 Github Discussion}
|
||||||
|
*/
|
||||||
|
page: async ({ page, failOnConsoleError }, use) => {
|
||||||
|
// Capture any console errors during test execution
|
||||||
|
const messages = [];
|
||||||
|
page.on('console', (msg) => messages.push(msg));
|
||||||
|
|
||||||
|
await use(page);
|
||||||
|
|
||||||
|
// Assert against console errors during teardown
|
||||||
|
if (failOnConsoleError) {
|
||||||
|
messages.forEach(
|
||||||
|
msg => expect.soft(msg.type(), `Console error detected: ${_consoleMessageToString(msg)}`).not.toEqual('error')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
/**
|
||||||
|
* Extends the base browser class to enable CDP connection definition in playwright.config.js. Once
|
||||||
|
* that RFE is implemented, this function can be removed.
|
||||||
|
* @see {@link https://github.com/microsoft/playwright/issues/8379 Github RFE}
|
||||||
|
*/
|
||||||
|
browser: async ({ playwright, browser }, use, workerInfo) => {
|
||||||
|
// Use browserless if configured
|
||||||
|
if (workerInfo.project.name.match(/browserless/)) {
|
||||||
|
const vBrowser = await playwright.chromium.connectOverCDP({
|
||||||
|
endpointURL: 'ws://localhost:3003'
|
||||||
|
});
|
||||||
|
await use(vBrowser);
|
||||||
|
} else {
|
||||||
|
// Use Local Browser for testing.
|
||||||
|
await use(browser);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
exports.expect = expect;
|
||||||
|
exports.waitForAnimations = waitForAnimations;
|
28
e2e/helper/addInitExampleFaultProvider.js
Normal file
28
e2e/helper/addInitExampleFaultProvider.js
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 Fault Provider, this will also install the FaultManagementPlugin (neither of which are installed by default).
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
openmct.install(openmct.plugins.example.ExampleFaultSource());
|
||||||
|
});
|
30
e2e/helper/addInitExampleFaultProviderStatic.js
Normal file
30
e2e/helper/addInitExampleFaultProviderStatic.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 Fault Provider, this will also install the FaultManagementPlugin (neither of which are installed by default).
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
const staticFaults = true;
|
||||||
|
|
||||||
|
openmct.install(openmct.plugins.example.ExampleFaultSource(staticFaults));
|
||||||
|
});
|
28
e2e/helper/addInitFaultManagementPlugin.js
Normal file
28
e2e/helper/addInitFaultManagementPlugin.js
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 Fault Provider, this will also install the FaultManagementPlugin (neither of which are installed by default).
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
openmct.install(openmct.plugins.FaultManagement());
|
||||||
|
});
|
30
e2e/helper/addInitRestrictedNotebook.js
Normal file
30
e2e/helper/addInitRestrictedNotebook.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 non-default Restricted Notebook plugin since it is not installed by default.
|
||||||
|
// e.g.
|
||||||
|
// await page.addInitScript({ path: path.join(__dirname, 'addInitRestrictedNotebook.js') });
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
openmct.install(openmct.plugins.RestrictedNotebook('CUSTOM_NAME'));
|
||||||
|
});
|
277
e2e/helper/faultUtils.js
Normal file
277
e2e/helper/faultUtils.js
Normal file
@ -0,0 +1,277 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function navigateToFaultManagementWithExample(page) {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, './', 'addInitExampleFaultProvider.js') });
|
||||||
|
|
||||||
|
await navigateToFaultItemInTree(page);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function navigateToFaultManagementWithStaticExample(page) {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, './', 'addInitExampleFaultProviderStatic.js') });
|
||||||
|
|
||||||
|
await navigateToFaultItemInTree(page);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function navigateToFaultManagementWithoutExample(page) {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, './', 'addInitFaultManagementPlugin.js') });
|
||||||
|
|
||||||
|
await navigateToFaultItemInTree(page);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function navigateToFaultItemInTree(page) {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Click text=Fault Management
|
||||||
|
await page.click('text=Fault Management'); // this verifies the plugin has been added
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function acknowledgeFault(page, rowNumber) {
|
||||||
|
await openFaultRowMenu(page, rowNumber);
|
||||||
|
await page.locator('.c-menu >> text="Acknowledge"').click();
|
||||||
|
// Click [aria-label="Save"]
|
||||||
|
await page.locator('[aria-label="Save"]').click();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function shelveMultipleFaults(page, ...nums) {
|
||||||
|
const selectRows = nums.map((num) => {
|
||||||
|
return selectFaultItem(page, num);
|
||||||
|
});
|
||||||
|
await Promise.all(selectRows);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Shelve")').click();
|
||||||
|
await page.locator('[aria-label="Save"]').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function acknowledgeMultipleFaults(page, ...nums) {
|
||||||
|
const selectRows = nums.map((num) => {
|
||||||
|
return selectFaultItem(page, num);
|
||||||
|
});
|
||||||
|
await Promise.all(selectRows);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Acknowledge")').click();
|
||||||
|
await page.locator('[aria-label="Save"]').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function shelveFault(page, rowNumber) {
|
||||||
|
await openFaultRowMenu(page, rowNumber);
|
||||||
|
await page.locator('.c-menu >> text="Shelve"').click();
|
||||||
|
// Click [aria-label="Save"]
|
||||||
|
await page.locator('[aria-label="Save"]').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function changeViewTo(page, view) {
|
||||||
|
await page.locator('.c-fault-mgmt__search-row select').first().selectOption(view);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function sortFaultsBy(page, sort) {
|
||||||
|
await page.locator('.c-fault-mgmt__list-header-sortButton select').selectOption(sort);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function enterSearchTerm(page, term) {
|
||||||
|
await page.locator('.c-fault-mgmt-search [aria-label="Search Input"]').fill(term);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function clearSearch(page) {
|
||||||
|
await enterSearchTerm(page, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function selectFaultItem(page, rowNumber) {
|
||||||
|
// eslint-disable-next-line playwright/no-force-option
|
||||||
|
await page.check(`.c-fault-mgmt-item > input >> nth=${rowNumber - 1}`, { force: true }); // this will not work without force true, saw this may be a pw bug
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getHighestSeverity(page) {
|
||||||
|
const criticalCount = await page.locator('[title=CRITICAL]').count();
|
||||||
|
const warningCount = await page.locator('[title=WARNING]').count();
|
||||||
|
|
||||||
|
if (criticalCount > 0) {
|
||||||
|
return 'CRITICAL';
|
||||||
|
} else if (warningCount > 0) {
|
||||||
|
return 'WARNING';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'WATCH';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getLowestSeverity(page) {
|
||||||
|
const warningCount = await page.locator('[title=WARNING]').count();
|
||||||
|
const watchCount = await page.locator('[title=WATCH]').count();
|
||||||
|
|
||||||
|
if (watchCount > 0) {
|
||||||
|
return 'WATCH';
|
||||||
|
} else if (warningCount > 0) {
|
||||||
|
return 'WARNING';
|
||||||
|
}
|
||||||
|
|
||||||
|
return 'CRITICAL';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getFaultResultCount(page) {
|
||||||
|
const count = await page.locator('.c-faults-list-view-item-body > .c-fault-mgmt__list').count();
|
||||||
|
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
function getFault(page, rowNumber) {
|
||||||
|
const fault = page.locator(`.c-faults-list-view-item-body > .c-fault-mgmt__list >> nth=${rowNumber - 1}`);
|
||||||
|
|
||||||
|
return fault;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
function getFaultByName(page, name) {
|
||||||
|
const fault = page.locator(`.c-fault-mgmt__list-faultname:has-text("${name}")`);
|
||||||
|
|
||||||
|
return fault;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getFaultName(page, rowNumber) {
|
||||||
|
const faultName = await page.locator(`.c-fault-mgmt__list-faultname >> nth=${rowNumber - 1}`).textContent();
|
||||||
|
|
||||||
|
return faultName;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getFaultSeverity(page, rowNumber) {
|
||||||
|
const faultSeverity = await page.locator(`.c-faults-list-view-item-body .c-fault-mgmt__list-severity >> nth=${rowNumber - 1}`).getAttribute('title');
|
||||||
|
|
||||||
|
return faultSeverity;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getFaultNamespace(page, rowNumber) {
|
||||||
|
const faultNamespace = await page.locator(`.c-fault-mgmt__list-path >> nth=${rowNumber - 1}`).textContent();
|
||||||
|
|
||||||
|
return faultNamespace;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function getFaultTriggerTime(page, rowNumber) {
|
||||||
|
const faultTriggerTime = await page.locator(`.c-fault-mgmt__list-trigTime >> nth=${rowNumber - 1} >> .c-fault-mgmt-item__value`).textContent();
|
||||||
|
|
||||||
|
return faultTriggerTime.toString().trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function openFaultRowMenu(page, rowNumber) {
|
||||||
|
// select
|
||||||
|
await page.locator(`.c-fault-mgmt-item > .c-fault-mgmt__list-action-button >> nth=${rowNumber - 1}`).click();
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
module.exports = {
|
||||||
|
navigateToFaultManagementWithExample,
|
||||||
|
navigateToFaultManagementWithStaticExample,
|
||||||
|
navigateToFaultManagementWithoutExample,
|
||||||
|
navigateToFaultItemInTree,
|
||||||
|
acknowledgeFault,
|
||||||
|
shelveMultipleFaults,
|
||||||
|
acknowledgeMultipleFaults,
|
||||||
|
shelveFault,
|
||||||
|
changeViewTo,
|
||||||
|
sortFaultsBy,
|
||||||
|
enterSearchTerm,
|
||||||
|
clearSearch,
|
||||||
|
selectFaultItem,
|
||||||
|
getHighestSeverity,
|
||||||
|
getLowestSeverity,
|
||||||
|
getFaultResultCount,
|
||||||
|
getFault,
|
||||||
|
getFaultByName,
|
||||||
|
getFaultName,
|
||||||
|
getFaultSeverity,
|
||||||
|
getFaultNamespace,
|
||||||
|
getFaultTriggerTime,
|
||||||
|
openFaultRowMenu
|
||||||
|
};
|
65
e2e/helper/notebookUtils.js
Normal file
65
e2e/helper/notebookUtils.js
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const NOTEBOOK_DROP_AREA = '.c-notebook__drag-area';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function enterTextEntry(page, text) {
|
||||||
|
// Click .c-notebook__drag-area
|
||||||
|
await page.locator(NOTEBOOK_DROP_AREA).click();
|
||||||
|
|
||||||
|
// enter text
|
||||||
|
await page.locator('div.c-ne__text').click();
|
||||||
|
await page.locator('div.c-ne__text').fill(text);
|
||||||
|
await page.locator('div.c-ne__text').press('Enter');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function dragAndDropEmbed(page, myItemsFolderName) {
|
||||||
|
// Click button:has-text("Create")
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
// Click li:has-text("Sine Wave Generator")
|
||||||
|
await page.locator('li:has-text("Sine Wave Generator")').click();
|
||||||
|
// Click form[name="mctForm"] >> text=My Items
|
||||||
|
await page.locator(`form[name="mctForm"] >> text=${myItemsFolderName}`).click();
|
||||||
|
// Click text=OK
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
// Click text=Open MCT My Items >> span >> nth=3
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
// Click text=Unnamed CUSTOM_NAME
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Unnamed CUSTOM_NAME').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.dragAndDrop('text=UNNAMED SINE WAVE GENERATOR', NOTEBOOK_DROP_AREA);
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
module.exports = {
|
||||||
|
enterTextEntry,
|
||||||
|
dragAndDropEmbed
|
||||||
|
};
|
30
e2e/helper/useSnowTheme.js
Normal file
30
e2e/helper/useSnowTheme.js
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 Snow theme for Open MCT. Espresso is the default
|
||||||
|
// e.g.
|
||||||
|
// await page.addInitScript({ path: path.join(__dirname, 'useSnowTheme.js') });
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', () => {
|
||||||
|
const openmct = window.openmct;
|
||||||
|
openmct.install(openmct.plugins.Snow());
|
||||||
|
});
|
@ -2,37 +2,44 @@
|
|||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
const { devices } = require('@playwright/test');
|
const { devices } = require('@playwright/test');
|
||||||
|
const MAX_FAILURES = 5;
|
||||||
|
const NUM_WORKERS = 2;
|
||||||
|
|
||||||
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
||||||
const config = {
|
const config = {
|
||||||
retries: 1,
|
retries: 2, //Retries 2 times for a total of 3 runs. When running sharded and with maxFailures = 5, this should ensure that flake is managed without failing the full suite
|
||||||
testDir: 'tests',
|
testDir: 'tests',
|
||||||
|
testIgnore: '**/*.perf.spec.js', //Ignore performance tests and define in playwright-perfromance.config.js
|
||||||
timeout: 60 * 1000,
|
timeout: 60 * 1000,
|
||||||
webServer: {
|
webServer: {
|
||||||
command: 'npm run start',
|
command: 'cross-env NODE_ENV=test npm run start',
|
||||||
port: 8080,
|
url: 'http://localhost:8080/#',
|
||||||
timeout: 200 * 1000,
|
timeout: 200 * 1000,
|
||||||
reuseExistingServer: !process.env.CI
|
reuseExistingServer: false
|
||||||
},
|
},
|
||||||
workers: 2, //Limit to 2 for CircleCI Agent
|
maxFailures: MAX_FAILURES, //Limits failures to 5 to reduce CI Waste
|
||||||
|
workers: NUM_WORKERS, //Limit to 2 for CircleCI Agent
|
||||||
use: {
|
use: {
|
||||||
baseURL: 'http://localhost:8080/',
|
baseURL: 'http://localhost:8080/',
|
||||||
headless: true,
|
headless: true,
|
||||||
ignoreHTTPSErrors: true,
|
ignoreHTTPSErrors: true,
|
||||||
screenshot: 'on',
|
screenshot: 'only-on-failure',
|
||||||
trace: 'on',
|
trace: 'on-first-retry',
|
||||||
video: 'on'
|
video: 'off'
|
||||||
},
|
},
|
||||||
projects: [
|
projects: [
|
||||||
{
|
{
|
||||||
name: 'chrome',
|
name: 'chrome',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
use: {
|
use: {
|
||||||
browserName: 'chromium'
|
browserName: 'chromium'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'MMOC',
|
name: 'MMOC',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
grepInvert: /@snapshot/,
|
grepInvert: /@snapshot/,
|
||||||
use: {
|
use: {
|
||||||
browserName: 'chromium',
|
browserName: 'chromium',
|
||||||
@ -41,19 +48,32 @@ const config = {
|
|||||||
height: 1440
|
height: 1440
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
/*{
|
{
|
||||||
name: 'ipad',
|
name: 'firefox',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
use: {
|
use: {
|
||||||
browserName: 'webkit',
|
browserName: 'firefox'
|
||||||
...devices['iPad (gen 7) landscape'] // Complete List https://github.com/microsoft/playwright/blob/main/packages/playwright-core/src/server/deviceDescriptorsSource.json
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'chrome-beta', //Only Chrome Beta is available on ubuntu -- not chrome canary
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium',
|
||||||
|
channel: 'chrome-beta'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}*/
|
|
||||||
],
|
],
|
||||||
reporter: [
|
reporter: [
|
||||||
['list'],
|
['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' }],
|
['junit', { outputFile: 'test-results/results.xml' }],
|
||||||
['allure-playwright'],
|
|
||||||
['github']
|
['github']
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
@ -2,18 +2,23 @@
|
|||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
const { devices } = require('@playwright/test');
|
const { devices } = require('@playwright/test');
|
||||||
|
|
||||||
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
||||||
const config = {
|
const config = {
|
||||||
retries: 0,
|
retries: 0,
|
||||||
testDir: 'tests',
|
testDir: 'tests',
|
||||||
|
testIgnore: '**/*.perf.spec.js',
|
||||||
timeout: 30 * 1000,
|
timeout: 30 * 1000,
|
||||||
webServer: {
|
webServer: {
|
||||||
|
env: {
|
||||||
|
NODE_ENV: 'test'
|
||||||
|
},
|
||||||
command: 'npm run start',
|
command: 'npm run start',
|
||||||
port: 8080,
|
url: 'http://localhost:8080/#',
|
||||||
timeout: 120 * 1000,
|
timeout: 120 * 1000,
|
||||||
reuseExistingServer: !process.env.CI
|
reuseExistingServer: true
|
||||||
},
|
},
|
||||||
workers: 1,
|
workers: 1,
|
||||||
use: {
|
use: {
|
||||||
@ -21,9 +26,9 @@ const config = {
|
|||||||
baseURL: 'http://localhost:8080/',
|
baseURL: 'http://localhost:8080/',
|
||||||
headless: false,
|
headless: false,
|
||||||
ignoreHTTPSErrors: true,
|
ignoreHTTPSErrors: true,
|
||||||
screenshot: 'on',
|
screenshot: 'only-on-failure',
|
||||||
trace: 'on',
|
trace: 'retain-on-failure',
|
||||||
video: 'on'
|
video: 'off'
|
||||||
},
|
},
|
||||||
projects: [
|
projects: [
|
||||||
{
|
{
|
||||||
@ -34,6 +39,7 @@ const config = {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'MMOC',
|
name: 'MMOC',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
grepInvert: /@snapshot/,
|
grepInvert: /@snapshot/,
|
||||||
use: {
|
use: {
|
||||||
browserName: 'chromium',
|
browserName: 'chromium',
|
||||||
@ -42,18 +48,59 @@ const config = {
|
|||||||
height: 1440
|
height: 1440
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'safari',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grep: /@ipad/, // only run ipad tests due to this bug https://github.com/microsoft/playwright/issues/8340
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
|
use: {
|
||||||
|
browserName: 'webkit'
|
||||||
}
|
}
|
||||||
/*{
|
},
|
||||||
|
{
|
||||||
|
name: 'firefox',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
|
use: {
|
||||||
|
browserName: 'firefox'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'canary',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium',
|
||||||
|
channel: 'chrome-canary' //Note this is not available in ubuntu/CircleCI
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'chrome-beta',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium',
|
||||||
|
channel: 'chrome-beta'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
name: 'ipad',
|
name: 'ipad',
|
||||||
|
testMatch: '**/*.e2e.spec.js', // only run e2e tests
|
||||||
|
grep: /@ipad/,
|
||||||
|
grepInvert: /@snapshot/,
|
||||||
use: {
|
use: {
|
||||||
browserName: 'webkit',
|
browserName: 'webkit',
|
||||||
...devices['iPad (gen 7) landscape'] // Complete List https://github.com/microsoft/playwright/blob/main/packages/playwright-core/src/server/deviceDescriptorsSource.json
|
...devices['iPad (gen 7) landscape'] // Complete List https://github.com/microsoft/playwright/blob/main/packages/playwright-core/src/server/deviceDescriptorsSource.json
|
||||||
}
|
}
|
||||||
}*/
|
}
|
||||||
],
|
],
|
||||||
reporter: [
|
reporter: [
|
||||||
['list'],
|
['list'],
|
||||||
['allure-playwright']
|
['html', {
|
||||||
|
open: 'on-failure',
|
||||||
|
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
||||||
|
}]
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
43
e2e/playwright-performance.config.js
Normal file
43
e2e/playwright-performance.config.js
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
/* eslint-disable no-undef */
|
||||||
|
// playwright.config.js
|
||||||
|
// @ts-check
|
||||||
|
|
||||||
|
const CI = process.env.CI === 'true';
|
||||||
|
|
||||||
|
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
||||||
|
const config = {
|
||||||
|
retries: 1, //Only for debugging purposes because trace is enabled only on first retry
|
||||||
|
testDir: 'tests/performance/',
|
||||||
|
timeout: 60 * 1000,
|
||||||
|
workers: 1, //Only run in serial with 1 worker
|
||||||
|
webServer: {
|
||||||
|
command: 'cross-env NODE_ENV=test npm run start',
|
||||||
|
url: 'http://localhost:8080/#',
|
||||||
|
timeout: 200 * 1000,
|
||||||
|
reuseExistingServer: !CI
|
||||||
|
},
|
||||||
|
use: {
|
||||||
|
browserName: "chromium",
|
||||||
|
baseURL: 'http://localhost:8080/',
|
||||||
|
headless: CI, //Only if running locally
|
||||||
|
ignoreHTTPSErrors: true,
|
||||||
|
screenshot: 'off',
|
||||||
|
trace: 'on-first-retry',
|
||||||
|
video: 'off'
|
||||||
|
},
|
||||||
|
projects: [
|
||||||
|
{
|
||||||
|
name: 'chrome',
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
reporter: [
|
||||||
|
['list'],
|
||||||
|
['junit', { outputFile: 'test-results/results.xml' }],
|
||||||
|
['json', { outputFile: 'test-results/results.json' }]
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = config;
|
@ -2,31 +2,49 @@
|
|||||||
// playwright.config.js
|
// playwright.config.js
|
||||||
// @ts-check
|
// @ts-check
|
||||||
|
|
||||||
/** @type {import('@playwright/test').PlaywrightTestConfig} */
|
/** @type {import('@playwright/test').PlaywrightTestConfig<{ theme: string }>} */
|
||||||
const config = {
|
const config = {
|
||||||
retries: 0,
|
retries: 1, // visual tests should never retry due to snapshot comparison errors. Leaving as a shim
|
||||||
testDir: 'tests',
|
testDir: 'tests/visual',
|
||||||
timeout: 90 * 1000,
|
testMatch: '**/*.visual.spec.js', // only run visual tests
|
||||||
workers: 1,
|
timeout: 60 * 1000,
|
||||||
|
workers: 1, //Lower stress on Circle CI Agent for Visual tests https://github.com/percy/cli/discussions/1067
|
||||||
webServer: {
|
webServer: {
|
||||||
command: 'npm run start',
|
command: 'cross-env NODE_ENV=test npm run start',
|
||||||
port: 8080,
|
url: 'http://localhost:8080/#',
|
||||||
timeout: 200 * 1000,
|
timeout: 200 * 1000,
|
||||||
reuseExistingServer: !process.env.CI
|
reuseExistingServer: !process.env.CI
|
||||||
},
|
},
|
||||||
use: {
|
use: {
|
||||||
browserName: "chromium",
|
|
||||||
baseURL: 'http://localhost:8080/',
|
baseURL: 'http://localhost:8080/',
|
||||||
headless: true,
|
headless: true, // this needs to remain headless to avoid visual changes due to GPU rendering in headed browsers
|
||||||
ignoreHTTPSErrors: true,
|
ignoreHTTPSErrors: true,
|
||||||
screenshot: 'on',
|
screenshot: 'only-on-failure',
|
||||||
trace: 'off',
|
trace: 'on-first-retry',
|
||||||
video: 'on'
|
video: 'off'
|
||||||
},
|
},
|
||||||
|
projects: [
|
||||||
|
{
|
||||||
|
name: 'chrome',
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'chrome-snow-theme',
|
||||||
|
use: {
|
||||||
|
browserName: 'chromium',
|
||||||
|
theme: 'snow'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
reporter: [
|
reporter: [
|
||||||
['list'],
|
['list'],
|
||||||
['junit', { outputFile: 'test-results/results.xml' }],
|
['junit', { outputFile: 'test-results/results.xml' }],
|
||||||
['allure-playwright']
|
['html', {
|
||||||
|
open: 'on-failure',
|
||||||
|
outputFolder: '../html-test-results' //Must be in different location due to https://github.com/microsoft/playwright/issues/12840
|
||||||
|
}]
|
||||||
]
|
]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
161
e2e/pluginFixtures.js
Normal file
161
e2e/pluginFixtures.js
Normal file
@ -0,0 +1,161 @@
|
|||||||
|
/* eslint-disable no-undef */
|
||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The file contains custom fixtures which extend the base functionality of the Playwright fixtures
|
||||||
|
* and appActions. These fixtures should be generalized across all plugins.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('./baseFixtures');
|
||||||
|
// const { createDomainObjectWithDefaults } = require('./appActions');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {Object} ObjectCreateOptions
|
||||||
|
* @property {string} type
|
||||||
|
* @property {string} name
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* **NOTE: This feature is a work-in-progress and should not currently be used.**
|
||||||
|
*
|
||||||
|
* Used to create a new domain object as a part of getOrCreateDomainObject.
|
||||||
|
* @type {Map<string, string>}
|
||||||
|
*/
|
||||||
|
// const createdObjects = new Map();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* **NOTE: This feature is a work-in-progress and should not currently be used.**
|
||||||
|
*
|
||||||
|
* This action will create a domain object for the test to reference and return the uuid. If an object
|
||||||
|
* of a given name already exists, it will return the uuid of that object to the test instead of creating
|
||||||
|
* a new file. The intent is to move object creation out of test suites which are not explicitly worried
|
||||||
|
* about object creation, while providing a consistent interface to retrieving objects in a persistentContext.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {ObjectCreateOptions} options
|
||||||
|
* @returns {Promise<string>} uuid of the domain object
|
||||||
|
*/
|
||||||
|
// async function getOrCreateDomainObject(page, options) {
|
||||||
|
// const { type, name } = options;
|
||||||
|
// const objectName = name ? `${type}:${name}` : type;
|
||||||
|
|
||||||
|
// if (createdObjects.has(objectName)) {
|
||||||
|
// return createdObjects.get(objectName);
|
||||||
|
// }
|
||||||
|
|
||||||
|
// await createDomainObjectWithDefaults(page, type, name);
|
||||||
|
|
||||||
|
// // Once object is created, get the uuid from the url
|
||||||
|
// const uuid = await page.evaluate(() => {
|
||||||
|
// return window.location.href.match(/[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}/)[0];
|
||||||
|
// });
|
||||||
|
|
||||||
|
// createdObjects.set(objectName, uuid);
|
||||||
|
|
||||||
|
// return uuid;
|
||||||
|
// }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* **NOTE: This feature is a work-in-progress and should not currently be used.**
|
||||||
|
*
|
||||||
|
* If provided, these options will be used to get or create the desired domain object before
|
||||||
|
* any tests or test hooks have run.
|
||||||
|
* The `uuid` of the `domainObject` will then be available to use within the scoped tests.
|
||||||
|
*
|
||||||
|
* ### Example:
|
||||||
|
* ```js
|
||||||
|
* test.describe("My test suite", () => {
|
||||||
|
* test.use({ objectCreateOptions: { type: "Telemetry Table", name: "My Telemetry Table" }});
|
||||||
|
* test("'My Telemetry Table' is created and provides a uuid", async ({ page, domainObject }) => {
|
||||||
|
* const { uuid } = domainObject;
|
||||||
|
* expect(uuid).toBeDefined();
|
||||||
|
* }))
|
||||||
|
* });
|
||||||
|
* ```
|
||||||
|
* @type {ObjectCreateOptions}
|
||||||
|
*/
|
||||||
|
// const objectCreateOptions = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The default theme for VIPER and Open MCT is the 'espresso' theme. Overriding this value with 'snow' in our playwright config.js
|
||||||
|
* will override the default theme by injecting the 'snow' theme on launch.
|
||||||
|
*
|
||||||
|
* ### Example:
|
||||||
|
* ```js
|
||||||
|
* projects: [
|
||||||
|
* {
|
||||||
|
* name: 'chrome-snow-theme',
|
||||||
|
* use: {
|
||||||
|
* browserName: 'chromium',
|
||||||
|
* theme: 'snow'
|
||||||
|
* ```
|
||||||
|
* @type {'snow' | 'espresso'}
|
||||||
|
*/
|
||||||
|
const theme = 'espresso';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The name of the "My Items" folder in the domain object tree.
|
||||||
|
*
|
||||||
|
* Default: `"My Items"`
|
||||||
|
*
|
||||||
|
* @type {string}
|
||||||
|
*/
|
||||||
|
const myItemsFolderName = "My Items";
|
||||||
|
|
||||||
|
exports.test = test.extend({
|
||||||
|
// This should follow in the Project's configuration. Can be set to 'snow' in playwright config.js
|
||||||
|
theme: [theme, { option: true }],
|
||||||
|
// eslint-disable-next-line no-shadow
|
||||||
|
page: async ({ page, theme }, use) => {
|
||||||
|
// eslint-disable-next-line playwright/no-conditional-in-test
|
||||||
|
if (theme === 'snow') {
|
||||||
|
//inject snow theme
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, './helper', './useSnowTheme.js') });
|
||||||
|
}
|
||||||
|
|
||||||
|
await use(page);
|
||||||
|
},
|
||||||
|
myItemsFolderName: [myItemsFolderName, { option: true }],
|
||||||
|
// eslint-disable-next-line no-shadow
|
||||||
|
openmctConfig: async ({ myItemsFolderName }, use) => {
|
||||||
|
await use({ myItemsFolderName });
|
||||||
|
}
|
||||||
|
// objectCreateOptions: [objectCreateOptions, {option: true}],
|
||||||
|
// eslint-disable-next-line no-shadow
|
||||||
|
// domainObject: [async ({ page, objectCreateOptions }, use) => {
|
||||||
|
// // FIXME: This is a false-positive caused by a bug in the eslint-plugin-playwright rule.
|
||||||
|
// // eslint-disable-next-line playwright/no-conditional-in-test
|
||||||
|
// if (objectCreateOptions === null) {
|
||||||
|
// await use(page);
|
||||||
|
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// //Go to baseURL
|
||||||
|
// await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// const uuid = await getOrCreateDomainObject(page, objectCreateOptions);
|
||||||
|
// await use({ uuid });
|
||||||
|
// }, { auto: true }]
|
||||||
|
});
|
||||||
|
exports.expect = expect;
|
1
e2e/test-data/PerformanceDisplayLayout.json
Normal file
1
e2e/test-data/PerformanceDisplayLayout.json
Normal file
@ -0,0 +1 @@
|
|||||||
|
{"openmct":{"b3cee102-86dd-4c0a-8eec-4d5d276f8691":{"identifier":{"key":"b3cee102-86dd-4c0a-8eec-4d5d276f8691","namespace":""},"name":"Performance Display Layout","type":"layout","composition":[{"key":"9666e7b4-be0c-47a5-94b8-99accad7155e","namespace":""}],"configuration":{"items":[{"width":32,"height":18,"x":12,"y":9,"identifier":{"key":"9666e7b4-be0c-47a5-94b8-99accad7155e","namespace":""},"hasFrame":true,"fontSize":"default","font":"default","type":"subobject-view","id":"23ca351d-a67d-46aa-a762-290eb742d2f1"}],"layoutGrid":[10,10]},"modified":1654299875432,"location":"mine","persisted":1654299878751},"9666e7b4-be0c-47a5-94b8-99accad7155e":{"identifier":{"key":"9666e7b4-be0c-47a5-94b8-99accad7155e","namespace":""},"name":"Performance Example Imagery","type":"example.imagery","configuration":{"imageLocation":"","imageLoadDelayInMilliSeconds":20000,"imageSamples":[],"layers":[{"source":"dist/imagery/example-imagery-layer-16x9.png","name":"16:9","visible":false},{"source":"dist/imagery/example-imagery-layer-safe.png","name":"Safe","visible":false},{"source":"dist/imagery/example-imagery-layer-scale.png","name":"Scale","visible":false}]},"telemetry":{"values":[{"name":"Name","key":"name"},{"name":"Time","key":"utc","format":"utc","hints":{"domain":2}},{"name":"Local Time","key":"local","format":"local-format","hints":{"domain":1}},{"name":"Image","key":"url","format":"image","hints":{"image":1},"layers":[{"source":"dist/imagery/example-imagery-layer-16x9.png","name":"16:9"},{"source":"dist/imagery/example-imagery-layer-safe.png","name":"Safe"},{"source":"dist/imagery/example-imagery-layer-scale.png","name":"Scale"}]},{"name":"Image Download Name","key":"imageDownloadName","format":"imageDownloadName","hints":{"imageDownloadName":1}}]},"modified":1654299840077,"location":"b3cee102-86dd-4c0a-8eec-4d5d276f8691","persisted":1654299840078}},"rootId":"b3cee102-86dd-4c0a-8eec-4d5d276f8691"}
|
1
e2e/test-data/PerformanceNotebook.json
Normal file
1
e2e/test-data/PerformanceNotebook.json
Normal file
@ -0,0 +1 @@
|
|||||||
|
{"openmct":{"6d2fa9fd-f2aa-461a-a1e1-164ac44bec9d":{"identifier":{"key":"6d2fa9fd-f2aa-461a-a1e1-164ac44bec9d","namespace":""},"name":"Performance Notebook","type":"notebook","configuration":{"defaultSort":"oldest","entries":{"3e31c412-33ba-4757-8ade-e9821f6ba321":{"8c8f6035-631c-45af-8c24-786c60295335":[{"id":"entry-1652815305457","createdOn":1652815305457,"createdBy":"","text":"Existing Entry 1","embeds":[]},{"id":"entry-1652815313465","createdOn":1652815313465,"createdBy":"","text":"Existing Entry 2","embeds":[]},{"id":"entry-1652815399955","createdOn":1652815399955,"createdBy":"","text":"Existing Entry 3","embeds":[]}]}},"imageMigrationVer":"v1","pageTitle":"Page","sections":[{"id":"3e31c412-33ba-4757-8ade-e9821f6ba321","isDefault":false,"isSelected":false,"name":"Section1","pages":[{"id":"8c8f6035-631c-45af-8c24-786c60295335","isDefault":false,"isSelected":false,"name":"Page1","pageTitle":"Page"},{"id":"36555942-c9aa-439c-bbdb-0aaf50db50f5","isDefault":false,"isSelected":false,"name":"Page2","pageTitle":"Page"}],"sectionTitle":"Section"},{"id":"dab0bd1d-2c5a-405c-987f-107123d6189a","isDefault":false,"isSelected":true,"name":"Section2","pages":[{"id":"f625a86a-cb99-4898-8082-80543c8de534","isDefault":false,"isSelected":false,"name":"Page1","pageTitle":"Page"},{"id":"e77ef810-f785-42a7-942e-07e999b79c59","isDefault":false,"isSelected":true,"name":"Page2","pageTitle":"Page"}],"sectionTitle":"Section"}],"sectionTitle":"Section","type":"General","showTime":"0"},"modified":1652815915219,"location":"mine","persisted":1652815915222}},"rootId":"6d2fa9fd-f2aa-461a-a1e1-164ac44bec9d"}
|
22
e2e/test-data/VisualTestData_storage.json
Normal file
22
e2e/test-data/VisualTestData_storage.json
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
"cookies": [],
|
||||||
|
"origins": [
|
||||||
|
{
|
||||||
|
"origin": "http://localhost:8080",
|
||||||
|
"localStorage": [
|
||||||
|
{
|
||||||
|
"name": "tcHistory",
|
||||||
|
"value": "{\"utc\":[{\"start\":1658617611983,\"end\":1658619411983}]}"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "mct",
|
||||||
|
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"7fa5749b-8969-494c-9d85-c272516d333c\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"persisted\":1658619412848,\"modified\":1658619412848},\"7fa5749b-8969-494c-9d85-c272516d333c\":{\"identifier\":{\"key\":\"7fa5749b-8969-494c-9d85-c272516d333c\",\"namespace\":\"\"},\"name\":\"Unnamed Overlay Plot\",\"type\":\"telemetry.plot.overlay\",\"composition\":[{\"key\":\"67cbb9fc-af46-4148-b9e5-aea11179ae4b\",\"namespace\":\"\"}],\"configuration\":{\"series\":[{\"identifier\":{\"key\":\"67cbb9fc-af46-4148-b9e5-aea11179ae4b\",\"namespace\":\"\"}}]},\"modified\":1658619413566,\"location\":\"mine\",\"persisted\":1658619413567},\"67cbb9fc-af46-4148-b9e5-aea11179ae4b\":{\"name\":\"Unnamed Sine Wave Generator\",\"type\":\"generator\",\"identifier\":{\"key\":\"67cbb9fc-af46-4148-b9e5-aea11179ae4b\",\"namespace\":\"\"},\"telemetry\":{\"period\":10,\"amplitude\":1,\"offset\":0,\"dataRateInHz\":1,\"phase\":0,\"randomness\":0,\"loadDelay\":\"5000\"},\"modified\":1658619413552,\"location\":\"7fa5749b-8969-494c-9d85-c272516d333c\",\"persisted\":1658619413552}}"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "mct-tree-expanded",
|
||||||
|
"value": "[\"/browse/mine\"]"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
22
e2e/test-data/recycled_local_storage.json
Normal file
22
e2e/test-data/recycled_local_storage.json
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
"cookies": [],
|
||||||
|
"origins": [
|
||||||
|
{
|
||||||
|
"origin": "http://localhost:8080",
|
||||||
|
"localStorage": [
|
||||||
|
{
|
||||||
|
"name": "tcHistory",
|
||||||
|
"value": "{\"utc\":[{\"start\":1658617494563,\"end\":1658619294563},{\"start\":1658617090044,\"end\":1658618890044},{\"start\":1658616460484,\"end\":1658618260484},{\"start\":1658608882159,\"end\":1658610682159},{\"start\":1654537164464,\"end\":1654538964464},{\"start\":1652301954635,\"end\":1652303754635}]}"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "mct",
|
||||||
|
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"f64bea3b-58a7-4586-8c05-8b651e5f0bfd\",\"namespace\":\"\"},{\"key\":\"2d02a680-eb7e-4645-bba2-dd298f76efb8\",\"namespace\":\"\"},{\"key\":\"72a5f66b-39a7-4f62-8c40-4a99a33d6a8e\",\"namespace\":\"\"},{\"key\":\"8e4d20f1-9a04-4de5-8db5-c7e08d27f70d\",\"namespace\":\"\"},{\"key\":\"3e294eae-6124-409b-a870-554d1bdcdd6f\",\"namespace\":\"\"},{\"key\":\"ec24d05d-5df5-4c96-9241-b73636cd19a9\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"persisted\":1658619295366,\"modified\":1658619295366},\"f64bea3b-58a7-4586-8c05-8b651e5f0bfd\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"f64bea3b-58a7-4586-8c05-8b651e5f0bfd\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"73f2d9ae-d1f3-4561-b7fc-ecd5df557249\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1652303755999,\"location\":\"mine\",\"persisted\":1652303756002},\"2d02a680-eb7e-4645-bba2-dd298f76efb8\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"2d02a680-eb7e-4645-bba2-dd298f76efb8\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"4291d80c-303c-4d8d-85e1-10f012b864fb\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1654538965702,\"location\":\"mine\",\"persisted\":1654538965702},\"72a5f66b-39a7-4f62-8c40-4a99a33d6a8e\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"72a5f66b-39a7-4f62-8c40-4a99a33d6a8e\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"2b6bf89f-877b-42b8-acc1-a9a575efdbe1\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1658610682787,\"location\":\"mine\",\"persisted\":1658610682787},\"8e4d20f1-9a04-4de5-8db5-c7e08d27f70d\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"8e4d20f1-9a04-4de5-8db5-c7e08d27f70d\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"b9a9c413-4b94-401d-b0c7-5e404f182616\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1658618261112,\"location\":\"mine\",\"persisted\":1658618261112},\"3e294eae-6124-409b-a870-554d1bdcdd6f\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"3e294eae-6124-409b-a870-554d1bdcdd6f\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"108043b1-9c88-4e1d-8deb-fbf2cdb528f9\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1658618890910,\"location\":\"mine\",\"persisted\":1658618890910},\"ec24d05d-5df5-4c96-9241-b73636cd19a9\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"ec24d05d-5df5-4c96-9241-b73636cd19a9\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"4062bd9b-b788-43dd-ab0a-8fa10a78d4b3\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1658619295363,\"location\":\"mine\",\"persisted\":1658619295363}}"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "mct-tree-expanded",
|
||||||
|
"value": "[]"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -1,166 +0,0 @@
|
|||||||
/*****************************************************************************
|
|
||||||
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding conditionSets.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
|
||||||
|
|
||||||
test.describe('Sine Wave Generator', () => {
|
|
||||||
test('Create new Sine Wave Generator Object and validate create Form Logic', async ({ page }) => {
|
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
//Click the Create button
|
|
||||||
await page.click('button:has-text("Create")');
|
|
||||||
|
|
||||||
// Click Sine Wave Generator
|
|
||||||
await page.click('text=Sine Wave Generator');
|
|
||||||
|
|
||||||
// Verify that the each required field has required indicator
|
|
||||||
// Title
|
|
||||||
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Verify that the Notes row does not have a required indicator
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(3) .form-row .c-form-row__state-indicator')).not.toContain('.req');
|
|
||||||
|
|
||||||
// Period
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(4) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Amplitude
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(5) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Offset
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(6) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Data Rate
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(7) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Phase
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(8) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Randomness
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(9) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req']);
|
|
||||||
|
|
||||||
// Verify that by removing value from required text field shows invalid indicator
|
|
||||||
await page.locator('text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]').fill('');
|
|
||||||
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(['c-form-row__state-indicator req invalid']);
|
|
||||||
|
|
||||||
// Verify that by adding value to empty required text field changes invalid to valid indicator
|
|
||||||
await page.locator('text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]').fill('non empty');
|
|
||||||
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(['c-form-row__state-indicator req valid']);
|
|
||||||
|
|
||||||
// Verify that by removing value from required number field shows invalid indicator
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().fill('');
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(4) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req invalid']);
|
|
||||||
|
|
||||||
// Verify that by adding value to empty required number field changes invalid to valid indicator
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().fill('3');
|
|
||||||
await expect(page.locator('.c-form__section div:nth-child(4) .form-row .c-form-row__state-indicator')).toHaveClass(['c-form-row__state-indicator req valid']);
|
|
||||||
|
|
||||||
// Verify that can change value of number field by up/down arrows keys
|
|
||||||
// Click .field.control.l-input-sm input >> nth=0
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().click();
|
|
||||||
// Press ArrowUp 3 times to change value from 3 to 6
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().press('ArrowUp');
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().press('ArrowUp');
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().press('ArrowUp');
|
|
||||||
|
|
||||||
const value = await page.locator('.field.control.l-input-sm input').first().inputValue();
|
|
||||||
await expect(value).toBe('6');
|
|
||||||
|
|
||||||
// Click .c-form-row__state-indicator.grows
|
|
||||||
await page.locator('.c-form-row__state-indicator.grows').click();
|
|
||||||
|
|
||||||
// Click text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]
|
|
||||||
await page.locator('text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]').click();
|
|
||||||
|
|
||||||
// Click .c-form-row__state-indicator >> nth=0
|
|
||||||
await page.locator('.c-form-row__state-indicator').first().click();
|
|
||||||
|
|
||||||
// Fill text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]
|
|
||||||
await page.locator('text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]').fill('New Sine Wave Generator');
|
|
||||||
|
|
||||||
// Double click div:nth-child(4) .form-row .c-form-row__controls
|
|
||||||
await page.locator('div:nth-child(4) .form-row .c-form-row__controls').dblclick();
|
|
||||||
|
|
||||||
// Click .field.control.l-input-sm input >> nth=0
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().click();
|
|
||||||
|
|
||||||
// Click div:nth-child(4) .form-row .c-form-row__state-indicator
|
|
||||||
await page.locator('div:nth-child(4) .form-row .c-form-row__state-indicator').click();
|
|
||||||
|
|
||||||
// Click .field.control.l-input-sm input >> nth=0
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().click();
|
|
||||||
|
|
||||||
// Click .field.control.l-input-sm input >> nth=0
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().click();
|
|
||||||
|
|
||||||
// Click div:nth-child(5) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(5) .form-row .c-form-row__controls .form-control .field input').click();
|
|
||||||
|
|
||||||
// Click div:nth-child(5) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(5) .form-row .c-form-row__controls .form-control .field input').click();
|
|
||||||
|
|
||||||
// Click div:nth-child(5) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(5) .form-row .c-form-row__controls .form-control .field input').click();
|
|
||||||
|
|
||||||
// Click div:nth-child(6) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(6) .form-row .c-form-row__controls .form-control .field input').click();
|
|
||||||
|
|
||||||
// Double click div:nth-child(7) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(7) .form-row .c-form-row__controls .form-control .field input').dblclick();
|
|
||||||
|
|
||||||
// Click div:nth-child(7) .form-row .c-form-row__state-indicator
|
|
||||||
await page.locator('div:nth-child(7) .form-row .c-form-row__state-indicator').click();
|
|
||||||
|
|
||||||
// Click div:nth-child(7) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(7) .form-row .c-form-row__controls .form-control .field input').click();
|
|
||||||
|
|
||||||
// Fill div:nth-child(7) .form-row .c-form-row__controls .form-control .field input
|
|
||||||
await page.locator('div:nth-child(7) .form-row .c-form-row__controls .form-control .field input').fill('3');
|
|
||||||
|
|
||||||
//Click text=OK
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.click('text=OK')
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Verify that the Sine Wave Generator is displayed and correct
|
|
||||||
// Verify object properties
|
|
||||||
await expect(page.locator('.l-browse-bar__object-name')).toContainText('New Sine Wave Generator');
|
|
||||||
|
|
||||||
// Verify canvas rendered
|
|
||||||
await page.locator('canvas').nth(1).click({
|
|
||||||
position: {
|
|
||||||
x: 341,
|
|
||||||
y: 28
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Verify that where we click on canvas shows the number we clicked on
|
|
||||||
// Note that any number will do, we just care that a number exists
|
|
||||||
await expect(page.locator('.value-to-display-nearestValue')).toContainText(/[+-]?([0-9]*[.])?[0-9]+/);
|
|
||||||
|
|
||||||
});
|
|
||||||
});
|
|
88
e2e/tests/framework/appActions.e2e.spec.js
Normal file
88
e2e/tests/framework/appActions.e2e.spec.js
Normal file
@ -0,0 +1,88 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../baseFixtures.js');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions.js');
|
||||||
|
|
||||||
|
test.describe('AppActions', () => {
|
||||||
|
test('createDomainObjectsWithDefaults', async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
const e2eFolder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'e2e folder'
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Create multiple flat objects in a row', async () => {
|
||||||
|
const timer1 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Timer',
|
||||||
|
name: 'Timer Foo',
|
||||||
|
parent: e2eFolder.uuid
|
||||||
|
});
|
||||||
|
const timer2 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Timer',
|
||||||
|
name: 'Timer Bar',
|
||||||
|
parent: e2eFolder.uuid
|
||||||
|
});
|
||||||
|
const timer3 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Timer',
|
||||||
|
name: 'Timer Baz',
|
||||||
|
parent: e2eFolder.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(timer1.url, { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toHaveText('Timer Foo');
|
||||||
|
await page.goto(timer2.url, { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toHaveText('Timer Bar');
|
||||||
|
await page.goto(timer3.url, { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toHaveText('Timer Baz');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step('Create multiple nested objects in a row', async () => {
|
||||||
|
const folder1 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Folder Foo',
|
||||||
|
parent: e2eFolder.uuid
|
||||||
|
});
|
||||||
|
const folder2 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Folder Bar',
|
||||||
|
parent: folder1.uuid
|
||||||
|
});
|
||||||
|
const folder3 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Folder Baz',
|
||||||
|
parent: folder2.uuid
|
||||||
|
});
|
||||||
|
await page.goto(folder1.url, { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toHaveText('Folder Foo');
|
||||||
|
await page.goto(folder2.url, { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toHaveText('Folder Bar');
|
||||||
|
await page.goto(folder3.url, { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toHaveText('Folder Baz');
|
||||||
|
|
||||||
|
expect(folder1.url).toBe(`${e2eFolder.url}/${folder1.uuid}`);
|
||||||
|
expect(folder2.url).toBe(`${e2eFolder.url}/${folder1.uuid}/${folder2.uuid}`);
|
||||||
|
expect(folder3.url).toBe(`${e2eFolder.url}/${folder1.uuid}/${folder2.uuid}/${folder3.uuid}`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
55
e2e/tests/framework/baseFixtures.e2e.spec.js
Normal file
55
e2e/tests/framework/baseFixtures.e2e.spec.js
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 testing our use of the playwright framework as it
|
||||||
|
relates to how we've extended it (i.e. ./e2e/baseFixtures.js) and assumptions made in our dev environment
|
||||||
|
(app.js and ./e2e/webpack-dev-middleware.js)
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test } = require('../../baseFixtures.js');
|
||||||
|
|
||||||
|
test.describe('baseFixtures tests', () => {
|
||||||
|
test('Verify that tests fail if console.error is thrown', async ({ page }) => {
|
||||||
|
test.fail();
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
//Verify that ../fixtures.js detects console log errors
|
||||||
|
await Promise.all([
|
||||||
|
page.evaluate(() => console.error('This should result in a failure')),
|
||||||
|
page.waitForEvent('console') // always wait for the event to happen while triggering it!
|
||||||
|
]);
|
||||||
|
|
||||||
|
});
|
||||||
|
test('Verify that tests pass if console.warn is thrown', async ({ page }) => {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
//Verify that ../fixtures.js detects console log errors
|
||||||
|
await Promise.all([
|
||||||
|
page.evaluate(() => console.warn('This should result in a pass')),
|
||||||
|
page.waitForEvent('console') // always wait for the event to happen while triggering it!
|
||||||
|
]);
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
148
e2e/tests/framework/exampleTemplate.e2e.spec.js
Normal file
148
e2e/tests/framework/exampleTemplate.e2e.spec.js
Normal file
@ -0,0 +1,148 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 template is to be used when creating new test suites. It will be kept up to date with the latest improvements
|
||||||
|
* made by the Open MCT team. It will also follow our best pratices as those evolve. Please use this structure as a _reference_ and clear
|
||||||
|
* or update any references when creating a new test suite!
|
||||||
|
*
|
||||||
|
* To illustrate current best practices, we've included a mocked up test suite for Renaming a Timer domain object.
|
||||||
|
*
|
||||||
|
* Demonstrated:
|
||||||
|
* - Using appActions to leverage existing functions
|
||||||
|
* - Structure
|
||||||
|
* - @unstable annotation
|
||||||
|
* - await, expect, test, describe syntax
|
||||||
|
* - Writing a custom function for a test suite
|
||||||
|
* - Test stub for unfinished test coverage (test.fixme)
|
||||||
|
*
|
||||||
|
* The structure should follow
|
||||||
|
* 1. imports
|
||||||
|
* 2. test.describe()
|
||||||
|
* 3. -> test1
|
||||||
|
* -> test2
|
||||||
|
* -> test3(stub)
|
||||||
|
* 4. Any custom functions
|
||||||
|
*/
|
||||||
|
|
||||||
|
// Structure: Some standard Imports. Please update the required pathing.
|
||||||
|
const { test, expect } = require('../../baseFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Structure:
|
||||||
|
* Try to keep a single describe block per logical groups of tests.
|
||||||
|
* If your test runtime exceeds 5 minutes or 500 lines, it's likely that it will need to be split.
|
||||||
|
*
|
||||||
|
* Annotations:
|
||||||
|
* Please use the @unstable tag at the end of the test title so that our automation can pick it up
|
||||||
|
* as a part of our test promotion pipeline.
|
||||||
|
*/
|
||||||
|
test.describe('Renaming Timer Object', () => {
|
||||||
|
// Top-level declaration of the Timer object created in beforeEach().
|
||||||
|
// We can then use this throughout the entire test suite.
|
||||||
|
let timer;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Open a browser, navigate to the main page, and wait until all network events to resolve
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// We provide some helper functions in appActions like `createDomainObjectWithDefaults()`.
|
||||||
|
// This example will create a Timer object with default properties, under the root folder:
|
||||||
|
timer = await createDomainObjectWithDefaults(page, { type: 'Timer' });
|
||||||
|
|
||||||
|
// Assert the object to be created and check its name in the title
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(timer.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make sure to use testcase names which are descriptive and easy to understand.
|
||||||
|
* A good testcase name concisely describes the test's goal(s) and should give
|
||||||
|
* some hint as to what went wrong if the test fails.
|
||||||
|
*/
|
||||||
|
test('An existing Timer object can be renamed via the 3dot actions menu', async ({ page }) => {
|
||||||
|
const newObjectName = "Renamed Timer";
|
||||||
|
|
||||||
|
// We've created an example of a shared function which pases the page and newObjectName values
|
||||||
|
await renameTimerFrom3DotMenu(page, timer.url, newObjectName);
|
||||||
|
|
||||||
|
// Assert that the name has changed in the browser bar to the value we assigned above
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(newObjectName);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('An existing Timer object can be renamed twice', async ({ page }) => {
|
||||||
|
const newObjectName = "Renamed Timer";
|
||||||
|
const newObjectName2 = "Re-Renamed Timer";
|
||||||
|
|
||||||
|
await renameTimerFrom3DotMenu(page, timer.url, newObjectName);
|
||||||
|
|
||||||
|
// Assert that the name has changed in the browser bar to the value we assigned above
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(newObjectName);
|
||||||
|
|
||||||
|
// Rename the Timer object again
|
||||||
|
await renameTimerFrom3DotMenu(page, timer.url, newObjectName2);
|
||||||
|
|
||||||
|
// Assert that the name has changed in the browser bar to the second value
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(newObjectName2);
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If you run out of time to write new tests, please stub in the missing tests
|
||||||
|
* in-place with a test.fixme and BDD-style test steps.
|
||||||
|
* Someone will carry the baton!
|
||||||
|
*/
|
||||||
|
test.fixme('Can Rename Timer Object from Tree', async ({ page }) => {
|
||||||
|
//Create a new object
|
||||||
|
//Copy this object
|
||||||
|
//Delete first object
|
||||||
|
//Expect copied object to persist
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Structure:
|
||||||
|
* Custom functions should be declared last.
|
||||||
|
* We are leaning on JSDoc pretty heavily to describe functionality. It is not required, but highly recommended.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is an example of a function which is shared between testcases in this test suite. When refactoring, we'll be looking
|
||||||
|
* for common functionality which makes sense to generalize for the entire test framework.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} timerUrl The URL of the timer object to be renamed
|
||||||
|
* @param {string} newNameForTimer New name for object
|
||||||
|
*/
|
||||||
|
async function renameTimerFrom3DotMenu(page, timerUrl, newNameForTimer) {
|
||||||
|
// Navigate to the timer object
|
||||||
|
await page.goto(timerUrl);
|
||||||
|
|
||||||
|
// Click on 3 Dot Menu
|
||||||
|
await page.locator('button[title="More options"]').click();
|
||||||
|
|
||||||
|
// Click text=Edit Properties...
|
||||||
|
await page.locator('text=Edit Properties...').click();
|
||||||
|
|
||||||
|
// Rename the timer object
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(newNameForTimer);
|
||||||
|
|
||||||
|
// Click Ok button to Save
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
}
|
64
e2e/tests/framework/generateVisualTestData.e2e.spec.js
Normal file
64
e2e/tests/framework/generateVisualTestData.e2e.spec.js
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 generating LocalStorage via Session Storage to be used
|
||||||
|
in some visual test suites like controlledClock.visual.spec.js. This suite should run to completion
|
||||||
|
and generate an artifact named ./e2e/test-data/VisualTestData_storage.json . This will run
|
||||||
|
on every Commit to ensure that this object still loads into tests correctly and will retain the
|
||||||
|
.e2e.spec.js suffix.
|
||||||
|
|
||||||
|
TODO: Provide additional validation of object properties as it grows.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions.js');
|
||||||
|
const { test, expect } = require('../../pluginFixtures.js');
|
||||||
|
|
||||||
|
test('Generate Visual Test Data @localStorage', async ({ page, context }) => {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
const overlayPlot = await createDomainObjectWithDefaults(page, { type: 'Overlay Plot' });
|
||||||
|
|
||||||
|
// click create button
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
|
||||||
|
// add sine wave generator with defaults
|
||||||
|
await page.locator('li:has-text("Sine Wave Generator")').click();
|
||||||
|
|
||||||
|
//Add a 5000 ms Delay
|
||||||
|
await page.locator('[aria-label="Loading Delay \\(ms\\)"]').fill('5000');
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=OK').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// focus the overlay plot
|
||||||
|
await page.goto(overlayPlot.url);
|
||||||
|
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Overlay Plot');
|
||||||
|
//Save localStorage for future test execution
|
||||||
|
await context.storageState({ path: './e2e/test-data/VisualTestData_storage.json' });
|
||||||
|
});
|
46
e2e/tests/framework/pluginFixtures.e2e.spec.js
Normal file
46
e2e/tests/framework/pluginFixtures.e2e.spec.js
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 testing our use of our custom fixtures to verify
|
||||||
|
that they are working as expected.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test } = require('../../pluginFixtures.js');
|
||||||
|
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.describe.skip('pluginFixtures tests', () => {
|
||||||
|
// test.use({ domainObjectName: 'Timer' });
|
||||||
|
// let timerUUID;
|
||||||
|
|
||||||
|
// test('Creates a timer object @framework @unstable', ({ domainObject }) => {
|
||||||
|
// const { uuid } = domainObject;
|
||||||
|
// const uuidRegexp = /[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}/;
|
||||||
|
// expect(uuid).toMatch(uuidRegexp);
|
||||||
|
// timerUUID = uuid;
|
||||||
|
// });
|
||||||
|
|
||||||
|
// test('Provides same uuid for subsequent uses of the same object @framework', ({ domainObject }) => {
|
||||||
|
// const { uuid } = domainObject;
|
||||||
|
// expect(uuid).toEqual(timerUUID);
|
||||||
|
// });
|
||||||
|
});
|
36
e2e/tests/framework/testData.e2e.spec.js
Normal file
36
e2e/tests/framework/testData.e2e.spec.js
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 template is to be used when verifying Test Data files found in /e2e/test-data/
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test } = require('../../baseFixtures');
|
||||||
|
|
||||||
|
test.describe('recycled_local_storage @localStorage', () => {
|
||||||
|
//We may want to do some additional level of verification of this file. For now, we just verify that it exists and can be used in a test suite.
|
||||||
|
test.use({ storageState: './e2e/test-data/recycled_local_storage.json' });
|
||||||
|
test('Can use recycled_local_storage file', async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
@ -24,30 +24,30 @@
|
|||||||
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('@playwright/test');
|
const { test, expect } = require('../../baseFixtures.js');
|
||||||
|
|
||||||
test.describe('Branding tests', () => {
|
test.describe('Branding tests', () => {
|
||||||
test('About Modal launches with basic branding properties', async ({ page }) => {
|
test('About Modal launches with basic branding properties', async ({ page }) => {
|
||||||
// Go to baseURL
|
// Go to baseURL
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
// Click About button
|
// Click About button
|
||||||
await page.click('.l-shell__app-logo');
|
await page.click('.l-shell__app-logo');
|
||||||
|
|
||||||
// Verify that the NASA Logo Appears
|
// Verify that the NASA Logo Appears
|
||||||
await expect(await page.locator('.c-about__image')).toBeVisible();
|
await expect(page.locator('.c-about__image')).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');
|
const versionInformationLocator = page.locator('ul.t-info.l-info.s-info').first();
|
||||||
await expect(versionInformationLocator).toBeEnabled();
|
await expect(versionInformationLocator).toBeEnabled();
|
||||||
await expect.soft(versionInformationLocator).toContainText(/Version: \d/);
|
await expect.soft(versionInformationLocator).toContainText(/Version: \d/);
|
||||||
await expect.soft(versionInformationLocator).toContainText(/Build Date: ((?:Mon|Tue|Wed|Thu|Fri|Sat|Sun))/);
|
await expect.soft(versionInformationLocator).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(versionInformationLocator).toContainText(/Revision: \b[0-9a-f]{5,40}\b/);
|
||||||
await expect.soft(versionInformationLocator).toContainText(/Branch: ./);
|
await expect.soft(versionInformationLocator).toContainText(/Branch: ./);
|
||||||
});
|
});
|
||||||
test('Verify Links in About Modal', async ({ page }) => {
|
test('Verify Links in About Modal @2p', async ({ page }) => {
|
||||||
// Go to baseURL
|
// Go to baseURL
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
// Click About button
|
// Click About button
|
||||||
await page.click('.l-shell__app-logo');
|
await page.click('.l-shell__app-logo');
|
||||||
@ -57,6 +57,7 @@ test.describe('Branding tests', () => {
|
|||||||
page.waitForEvent('popup'),
|
page.waitForEvent('popup'),
|
||||||
page.locator('text=click here for third party licensing information').click()
|
page.locator('text=click here for third party licensing information').click()
|
||||||
]);
|
]);
|
||||||
expect(page2.waitForURL('**\/licenses**')).toBeTruthy();
|
await page2.waitForLoadState('networkidle'); //Avoids timing issues with juggler/firefox
|
||||||
|
expect(page2.waitForURL('**/licenses**')).toBeTruthy();
|
||||||
});
|
});
|
||||||
});
|
});
|
108
e2e/tests/functional/couchdb.e2e.spec.js
Normal file
108
e2e/tests/functional/couchdb.e2e.spec.js
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 meant to be executed against a couchdb container. More doc to come
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../baseFixtures');
|
||||||
|
|
||||||
|
test.describe("CouchDB Status Indicator @couchdb", () => {
|
||||||
|
test.use({ failOnConsoleError: false });
|
||||||
|
//TODO BeforeAll Verify CouchDB Connectivity with APIContext
|
||||||
|
test('Shows green if connected', async ({ page }) => {
|
||||||
|
await page.route('**/openmct/mine', route => {
|
||||||
|
route.fulfill({
|
||||||
|
status: 200,
|
||||||
|
contentType: 'application/json',
|
||||||
|
body: JSON.stringify({})
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./#/browse/mine?hideTree=true&hideInspector=true', { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('div:has-text("CouchDB is connected")').nth(3)).toBeVisible();
|
||||||
|
});
|
||||||
|
test('Shows red if not connected', async ({ page }) => {
|
||||||
|
await page.route('**/openmct/**', route => {
|
||||||
|
route.fulfill({
|
||||||
|
status: 503,
|
||||||
|
contentType: 'application/json',
|
||||||
|
body: JSON.stringify({})
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./#/browse/mine?hideTree=true&hideInspector=true', { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('div:has-text("CouchDB is offline")').nth(3)).toBeVisible();
|
||||||
|
});
|
||||||
|
test('Shows unknown if it receives an unexpected response code', async ({ page }) => {
|
||||||
|
await page.route('**/openmct/mine', route => {
|
||||||
|
route.fulfill({
|
||||||
|
status: 418,
|
||||||
|
contentType: 'application/json',
|
||||||
|
body: JSON.stringify({})
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./#/browse/mine?hideTree=true&hideInspector=true', { waitUntil: 'networkidle' });
|
||||||
|
await expect(page.locator('div:has-text("CouchDB connectivity unknown")').nth(3)).toBeVisible();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe("CouchDB initialization @couchdb", () => {
|
||||||
|
test.use({ failOnConsoleError: false });
|
||||||
|
test("'My Items' folder is created if it doesn't exist", async ({ page }) => {
|
||||||
|
// Store any relevant PUT requests that happen on the page
|
||||||
|
const createMineFolderRequests = [];
|
||||||
|
page.on('request', req => {
|
||||||
|
// eslint-disable-next-line playwright/no-conditional-in-test
|
||||||
|
if (req.method() === 'PUT' && req.url().endsWith('openmct/mine')) {
|
||||||
|
createMineFolderRequests.push(req);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Override the first request to GET openmct/mine to return a 404
|
||||||
|
await page.route('**/openmct/mine', route => {
|
||||||
|
route.fulfill({
|
||||||
|
status: 404,
|
||||||
|
contentType: 'application/json',
|
||||||
|
body: JSON.stringify({})
|
||||||
|
});
|
||||||
|
}, { times: 1 });
|
||||||
|
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Verify that error banner is displayed
|
||||||
|
const bannerMessage = await page.locator('.c-message-banner__message').innerText();
|
||||||
|
expect(bannerMessage).toEqual('Failed to retrieve object mine');
|
||||||
|
|
||||||
|
// Verify that a PUT request to create "My Items" folder was made
|
||||||
|
await expect.poll(() => createMineFolderRequests.length, {
|
||||||
|
message: 'Verify that PUT request to create "mine" folder was made',
|
||||||
|
timeout: 1000
|
||||||
|
}).toBeGreaterThanOrEqual(1);
|
||||||
|
});
|
||||||
|
});
|
@ -24,33 +24,30 @@
|
|||||||
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('@playwright/test');
|
const { test, expect } = require('../../../baseFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../appActions');
|
||||||
|
|
||||||
test.describe('Example Event Generator Operations', () => {
|
test.describe('Example Event Generator CRUD Operations', () => {
|
||||||
test('Can create example event generator with a name', async ({ page }) => {
|
test('Can create a Test Event Generator and it results in the table View', async ({ page }) => {
|
||||||
//Go to baseURL
|
//Go to baseURL
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
// let's make an event generator
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
|
||||||
// Click li:has-text("Event Message Generator")
|
|
||||||
await page.locator('li:has-text("Event Message Generator")').click();
|
|
||||||
// Click text=Properties Title Notes >> input[type="text"]
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
|
||||||
// Fill text=Properties Title Notes >> input[type="text"]
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').fill('Test Event Generator');
|
|
||||||
// Press Enter
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').press('Enter');
|
|
||||||
// Click text=OK
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation({ url: /.*&view=table/ }),
|
|
||||||
page.locator('text=OK').click()
|
|
||||||
]);
|
|
||||||
|
|
||||||
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Test Event Generator');
|
//Create a name for the object
|
||||||
// Click button:has-text("Fixed Timespan")
|
const newObjectName = 'Test Event Generator';
|
||||||
await page.locator('button:has-text("Fixed Timespan")').click();
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Event Message Generator',
|
||||||
|
name: newObjectName
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//Assertions against newly created object which define standard behavior
|
||||||
|
await expect(page.waitForURL(/.*&view=table/)).toBeTruthy();
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(newObjectName);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Example Event Generator Telemetry Event Verficiation', () => {
|
||||||
|
|
||||||
test.fixme('telemetry is coming in for test event', async ({ page }) => {
|
test.fixme('telemetry is coming in for test event', async ({ page }) => {
|
||||||
// Go to object created in step one
|
// Go to object created in step one
|
||||||
// Verify the telemetry table is filled with > 1 row
|
// Verify the telemetry table is filled with > 1 row
|
@ -0,0 +1,119 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding conditionSets.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
|
||||||
|
test.describe('Sine Wave Generator', () => {
|
||||||
|
test('Create new Sine Wave Generator Object and validate create Form Logic', async ({ page, browserName }) => {
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip(browserName === 'firefox', 'This test needs to be updated to work with firefox');
|
||||||
|
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
//Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click Sine Wave Generator
|
||||||
|
await page.click('text=Sine Wave Generator');
|
||||||
|
|
||||||
|
// Verify that the each required field has required indicator
|
||||||
|
// Title
|
||||||
|
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Verify that the Notes row does not have a required indicator
|
||||||
|
await expect(page.locator('.c-form__section div:nth-child(3) .form-row .c-form-row__state-indicator')).not.toContain('.req');
|
||||||
|
await page.locator('textarea[type="text"]').fill('Optional Note Text');
|
||||||
|
|
||||||
|
// Period
|
||||||
|
await expect(page.locator('div:nth-child(4) .c-form-row__state-indicator')).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Amplitude
|
||||||
|
await expect(page.locator('div:nth-child(5) .c-form-row__state-indicator')).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Offset
|
||||||
|
await expect(page.locator('div:nth-child(6) .c-form-row__state-indicator')).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Data Rate
|
||||||
|
await expect(page.locator('div:nth-child(7) .c-form-row__state-indicator')).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Phase
|
||||||
|
await expect(page.locator('div:nth-child(8) .c-form-row__state-indicator')).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Randomness
|
||||||
|
await expect(page.locator('div:nth-child(9) .c-form-row__state-indicator')).toHaveClass(/req/);
|
||||||
|
|
||||||
|
// Verify that by removing value from required text field shows invalid indicator
|
||||||
|
await page.locator('text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]').fill('');
|
||||||
|
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(/invalid/);
|
||||||
|
|
||||||
|
// Verify that by adding value to empty required text field changes invalid to valid indicator
|
||||||
|
await page.locator('text=Properties Title Notes Period Amplitude Offset Data Rate (hz) Phase (radians) Ra >> input[type="text"]').fill('New Sine Wave Generator');
|
||||||
|
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(/valid/);
|
||||||
|
|
||||||
|
// Verify that by removing value from required number field shows invalid indicator
|
||||||
|
await page.locator('.field.control.l-input-sm input').first().fill('');
|
||||||
|
await expect(page.locator('div:nth-child(4) .c-form-row__state-indicator')).toHaveClass(/invalid/);
|
||||||
|
|
||||||
|
// Verify that by adding value to empty required number field changes invalid to valid indicator
|
||||||
|
await page.locator('.field.control.l-input-sm input').first().fill('3');
|
||||||
|
await expect(page.locator('div:nth-child(4) .c-form-row__state-indicator')).toHaveClass(/valid/);
|
||||||
|
|
||||||
|
// Verify that can change value of number field by up/down arrows keys
|
||||||
|
// Click .field.control.l-input-sm input >> nth=0
|
||||||
|
await page.locator('.field.control.l-input-sm input').first().click();
|
||||||
|
// Press ArrowUp 3 times to change value from 3 to 6
|
||||||
|
await page.locator('.field.control.l-input-sm input').first().press('ArrowUp');
|
||||||
|
await page.locator('.field.control.l-input-sm input').first().press('ArrowUp');
|
||||||
|
await page.locator('.field.control.l-input-sm input').first().press('ArrowUp');
|
||||||
|
|
||||||
|
const value = await page.locator('.field.control.l-input-sm input').first().inputValue();
|
||||||
|
await expect(value).toBe('6');
|
||||||
|
|
||||||
|
//Click text=OK
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.click('text=OK')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Verify that the Sine Wave Generator is displayed and correct
|
||||||
|
// Verify object properties
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('New Sine Wave Generator');
|
||||||
|
|
||||||
|
// Verify canvas rendered and can be interacted with
|
||||||
|
await page.locator('canvas').nth(1).click({
|
||||||
|
position: {
|
||||||
|
x: 341,
|
||||||
|
y: 28
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify that where we click on canvas shows the number we clicked on
|
||||||
|
// Note that any number will do, we just care that a number exists
|
||||||
|
await expect(page.locator('.value-to-display-nearestValue')).toContainText(/[+-]?([0-9]*[.])?[0-9]+/);
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
100
e2e/tests/functional/forms.e2e.spec.js
Normal file
100
e2e/tests/functional/forms.e2e.spec.js
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 form functionality in isolation
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../baseFixtures');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const TEST_FOLDER = 'test folder';
|
||||||
|
|
||||||
|
test.describe('Form Validation Behavior', () => {
|
||||||
|
test('Required Field indicators appear if title is empty and can be corrected', async ({ page }) => {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
await page.click(':nth-match(:text("Folder"), 2)');
|
||||||
|
|
||||||
|
// Fill in empty string into title and trigger validation with 'Tab'
|
||||||
|
await page.click('text=Properties Title Notes >> input[type="text"]');
|
||||||
|
await page.fill('text=Properties Title Notes >> input[type="text"]', '');
|
||||||
|
await page.press('text=Properties Title Notes >> input[type="text"]', 'Tab');
|
||||||
|
|
||||||
|
//Required Field Form Validation
|
||||||
|
await expect(page.locator('text=OK')).toBeDisabled();
|
||||||
|
await expect(page.locator('.c-form-row__state-indicator').first()).toHaveClass(/invalid/);
|
||||||
|
|
||||||
|
//Correct Form Validation for missing title and trigger validation with 'Tab'
|
||||||
|
await page.click('text=Properties Title Notes >> input[type="text"]');
|
||||||
|
await page.fill('text=Properties Title Notes >> input[type="text"]', TEST_FOLDER);
|
||||||
|
await page.press('text=Properties Title Notes >> input[type="text"]', 'Tab');
|
||||||
|
|
||||||
|
//Required Field Form Validation is corrected
|
||||||
|
await expect(page.locator('text=OK')).toBeEnabled();
|
||||||
|
await expect(page.locator('.c-form-row__state-indicator').first()).not.toHaveClass(/invalid/);
|
||||||
|
|
||||||
|
//Finish Creating Domain Object
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.click('text=OK')
|
||||||
|
]);
|
||||||
|
|
||||||
|
//Verify that the Domain Object has been created with the corrected title property
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(TEST_FOLDER);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Persistence operations @addInit', () => {
|
||||||
|
// add non persistable root item
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, '../../helper', 'addNoneditableObject.js') });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Persistability should be respected in the create form location field', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/4323'
|
||||||
|
});
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
await page.click('text=Condition Set');
|
||||||
|
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Persistence Testing').click();
|
||||||
|
|
||||||
|
const okButton = page.locator('button:has-text("OK")');
|
||||||
|
await expect(okButton).toBeDisabled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Form Correctness by Object Type', () => {
|
||||||
|
test.fixme('Verify correct behavior of number object (SWG)', async ({page}) => {});
|
||||||
|
test.fixme('Verify correct behavior of number object Timer', async ({page}) => {});
|
||||||
|
test.fixme('Verify correct behavior of number object Plan View', async ({page}) => {});
|
||||||
|
test.fixme('Verify correct behavior of number object Clock', async ({page}) => {});
|
||||||
|
test.fixme('Verify correct behavior of number object Hyperlink', async ({page}) => {});
|
||||||
|
});
|
@ -21,57 +21,30 @@
|
|||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This test suite is dedicated to tests which verify the basic operations surrounding conditionSets.
|
This test suite is dedicated to tests which verify persistability checks
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
const { test, expect } = require('../../baseFixtures.js');
|
||||||
|
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
|
|
||||||
// https://github.com/nasa/openmct/issues/4323#issuecomment-1067282651
|
test.describe('Persistence operations @addInit', () => {
|
||||||
|
|
||||||
test.describe('Persistence operations', () => {
|
|
||||||
// add non persistable root item
|
// add non persistable root item
|
||||||
test.beforeEach(async ({ page }) => {
|
test.beforeEach(async ({ page }) => {
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
await page.addInitScript({ path: path.join(__dirname, 'addNoneditableObject.js') });
|
await page.addInitScript({ path: path.join(__dirname, '../../helper', 'addNoneditableObject.js') });
|
||||||
});
|
});
|
||||||
|
|
||||||
test('Persistability should be respected in the create form location field', async ({ page }) => {
|
|
||||||
// Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Click the Create button
|
|
||||||
await page.click('button:has-text("Create")');
|
|
||||||
|
|
||||||
// Click text=Condition Set
|
|
||||||
await page.click('text=Condition Set');
|
|
||||||
|
|
||||||
// Click form[name="mctForm"] >> text=Persistence Testing
|
|
||||||
await page.locator('form[name="mctForm"] >> text=Persistence Testing').click();
|
|
||||||
|
|
||||||
// Check that "OK" button is disabled
|
|
||||||
const okButton = page.locator('button:has-text("OK")');
|
|
||||||
await expect(okButton).toBeDisabled();
|
|
||||||
});
|
|
||||||
test('Non-persistable objects should not show persistence related actions', async ({ page }) => {
|
test('Non-persistable objects should not show persistence related actions', async ({ page }) => {
|
||||||
// Go to baseURL
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Click text=Persistence Testing >> nth=0
|
|
||||||
await page.locator('text=Persistence Testing').first().click({
|
await page.locator('text=Persistence Testing').first().click({
|
||||||
button: 'right'
|
button: 'right'
|
||||||
});
|
});
|
||||||
|
|
||||||
const menuOptions = page.locator('.c-menu ul');
|
const menuOptions = page.locator('.c-menu li');
|
||||||
|
|
||||||
await expect.soft(menuOptions).toContainText(['Open In New Tab', 'View', 'Create Link']);
|
await expect.soft(menuOptions).toContainText(['Open In New Tab', 'View', 'Create Link']);
|
||||||
await expect(menuOptions).not.toContainText(['Move', 'Duplicate', 'Remove', 'Add New Folder', 'Edit Properties...', 'Export as JSON', 'Import from JSON']);
|
await expect(menuOptions).not.toContainText(['Move', 'Duplicate', 'Remove', 'Add New Folder', 'Edit Properties...', 'Export as JSON', 'Import from JSON']);
|
||||||
});
|
});
|
||||||
test.fixme('Cannot move a previously created domain object to non-peristable boject in Move Modal', async ({ page }) => {
|
|
||||||
//Create a domain object
|
|
||||||
//Save Domain object
|
|
||||||
//Move Object and verify that cannot select non-persistable object
|
|
||||||
//Move Object to My Items
|
|
||||||
//Verify successful move
|
|
||||||
});
|
|
||||||
});
|
});
|
212
e2e/tests/functional/moveAndLinkObjects.e2e.spec.js
Normal file
212
e2e/tests/functional/moveAndLinkObjects.e2e.spec.js
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding moving & linking objects.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
|
|
||||||
|
test.describe('Move & link item tests', () => {
|
||||||
|
test('Create a basic object and verify that it can be moved to another folder', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
// Go to Open MCT
|
||||||
|
await page.goto('./');
|
||||||
|
|
||||||
|
const parentFolder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Parent Folder'
|
||||||
|
});
|
||||||
|
const childFolder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Child Folder',
|
||||||
|
parent: parentFolder.uuid
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Grandchild Folder',
|
||||||
|
parent: childFolder.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
// Attempt to move parent to its own grandparent
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
await page.locator('.c-disclosure-triangle >> nth=0').click();
|
||||||
|
|
||||||
|
await page.locator(`a:has-text("Parent Folder") >> nth=0`).click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.locator('li.icon-move').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> .c-disclosure-triangle >> nth=0').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Parent Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('form[name="mctForm"] >> .c-disclosure-triangle >> nth=1').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Child Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('form[name="mctForm"] >> .c-disclosure-triangle >> nth=2').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Grandchild Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Parent Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('[aria-label="Cancel"]').click();
|
||||||
|
|
||||||
|
// Move Child Folder from Parent Folder to My Items
|
||||||
|
await page.locator('.c-disclosure-triangle >> nth=0').click();
|
||||||
|
await page.locator('.c-disclosure-triangle >> nth=1').click();
|
||||||
|
|
||||||
|
await page.locator(`a:has-text("Child Folder") >> nth=0`).click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
await page.locator('li.icon-move').click();
|
||||||
|
await page.locator(`form[name="mctForm"] >> text=${myItemsFolderName}`).click();
|
||||||
|
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
// Expect that Child Folder is in My Items, the root folder
|
||||||
|
expect(page.locator(`text=${myItemsFolderName} >> nth=0:has(text=Child Folder)`)).toBeTruthy();
|
||||||
|
});
|
||||||
|
test('Create a basic object and verify that it cannot be moved to telemetry object without Composition Provider', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
// Go to Open MCT
|
||||||
|
await page.goto('./');
|
||||||
|
|
||||||
|
// Create Telemetry Table
|
||||||
|
let telemetryTable = 'Test Telemetry Table';
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li:has-text("Telemetry Table")').click();
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(telemetryTable);
|
||||||
|
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
// Finish editing and save Telemetry Table
|
||||||
|
await page.locator('.c-button--menu.c-button--major.icon-save').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
// Create New Folder Basic Domain Object
|
||||||
|
let folder = 'Test Folder';
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li:has-text("Folder")').click();
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(folder);
|
||||||
|
|
||||||
|
// See if it's possible to put the folder in the Telemetry object during creation (Soft Assert)
|
||||||
|
await page.locator(`form[name="mctForm"] >> text=${telemetryTable}`).click();
|
||||||
|
let okButton = await page.locator('button.c-button.c-button--major:has-text("OK")');
|
||||||
|
let okButtonStateDisabled = await okButton.isDisabled();
|
||||||
|
expect.soft(okButtonStateDisabled).toBeTruthy();
|
||||||
|
|
||||||
|
// Continue test regardless of assertion and create it in My Items
|
||||||
|
await page.locator(`form[name="mctForm"] >> text=${myItemsFolderName}`).click();
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
// Open My Items
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
|
||||||
|
// Select Folder Object and select Move from context menu
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator(`a:has-text("${folder}")`).click()
|
||||||
|
]);
|
||||||
|
await page.locator('.c-tree__item.is-navigated-object .c-tree__item__label .c-tree__item__type-icon').click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
await page.locator('li.icon-move').click();
|
||||||
|
|
||||||
|
// See if it's possible to put the folder in the Telemetry object after creation
|
||||||
|
await page.locator(`text=Location Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
await page.locator(`form[name="mctForm"] >> text=${telemetryTable}`).click();
|
||||||
|
let okButton2 = await page.locator('button.c-button.c-button--major:has-text("OK")');
|
||||||
|
let okButtonStateDisabled2 = await okButton2.isDisabled();
|
||||||
|
expect(okButtonStateDisabled2).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Create a basic object and verify that it can be linked to another folder', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
// Go to Open MCT
|
||||||
|
await page.goto('./');
|
||||||
|
|
||||||
|
const parentFolder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Parent Folder'
|
||||||
|
});
|
||||||
|
const childFolder = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Child Folder',
|
||||||
|
parent: parentFolder.uuid
|
||||||
|
});
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Grandchild Folder',
|
||||||
|
parent: childFolder.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
// Attempt to link parent to its own grandparent
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
await page.locator('.c-disclosure-triangle >> nth=0').click();
|
||||||
|
|
||||||
|
await page.locator(`a:has-text("Parent Folder") >> nth=0`).click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.locator('li.icon-link').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> .c-disclosure-triangle >> nth=0').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Parent Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('form[name="mctForm"] >> .c-disclosure-triangle >> nth=1').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Child Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('form[name="mctForm"] >> .c-disclosure-triangle >> nth=2').click();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Grandchild Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Parent Folder').click();
|
||||||
|
await expect(page.locator('[aria-label="Save"]')).toBeDisabled();
|
||||||
|
await page.locator('[aria-label="Cancel"]').click();
|
||||||
|
|
||||||
|
// Link Child Folder from Parent Folder to My Items
|
||||||
|
await page.locator('.c-disclosure-triangle >> nth=0').click();
|
||||||
|
await page.locator('.c-disclosure-triangle >> nth=1').click();
|
||||||
|
|
||||||
|
await page.locator(`a:has-text("Child Folder") >> nth=0`).click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
await page.locator('li.icon-link').click();
|
||||||
|
await page.locator(`form[name="mctForm"] >> text=${myItemsFolderName}`).click();
|
||||||
|
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
// Expect that Child Folder is in My Items, the root folder
|
||||||
|
expect(page.locator(`text=${myItemsFolderName} >> nth=0:has(text=Child Folder)`)).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.fixme('Cannot move a previously created domain object to non-peristable object in Move Modal', async ({ page }) => {
|
||||||
|
//Create a domain object
|
||||||
|
//Save Domain object
|
||||||
|
//Move Object and verify that cannot select non-persistable object
|
||||||
|
//Move Object to My Items
|
||||||
|
//Verify successful move
|
||||||
|
});
|
87
e2e/tests/functional/planning/plan.e2e.spec.js
Normal file
87
e2e/tests/functional/planning/plan.e2e.spec.js
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
const { test, expect } = require('../../../pluginFixtures');
|
||||||
|
const { createPlanFromJSON } = require('../../../appActions');
|
||||||
|
|
||||||
|
const testPlan = {
|
||||||
|
"TEST_GROUP": [
|
||||||
|
{
|
||||||
|
"name": "Past event 1",
|
||||||
|
"start": 1660320408000,
|
||||||
|
"end": 1660343797000,
|
||||||
|
"type": "TEST-GROUP",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 2",
|
||||||
|
"start": 1660406808000,
|
||||||
|
"end": 1660429160000,
|
||||||
|
"type": "TEST-GROUP",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"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("Plan", () => {
|
||||||
|
test("Create a Plan and display all plan events @unstable", async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
const plan = await createPlanFromJSON(page, {
|
||||||
|
name: 'Test Plan',
|
||||||
|
json: testPlan
|
||||||
|
});
|
||||||
|
const startBound = testPlan.TEST_GROUP[0].start;
|
||||||
|
const endBound = testPlan.TEST_GROUP[testPlan.TEST_GROUP.length - 1].end;
|
||||||
|
|
||||||
|
// 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`);
|
||||||
|
const eventCount = await page.locator('.activity-bounds').count();
|
||||||
|
expect(eventCount).toEqual(testPlan.TEST_GROUP.length);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
181
e2e/tests/functional/planning/timestrip.e2e.spec.js
Normal file
181
e2e/tests/functional/planning/timestrip.e2e.spec.js
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults, createPlanFromJSON } = require('../../../appActions');
|
||||||
|
|
||||||
|
const testPlan = {
|
||||||
|
"TEST_GROUP": [
|
||||||
|
{
|
||||||
|
"name": "Past event 1",
|
||||||
|
"start": 1660320408000,
|
||||||
|
"end": 1660343797000,
|
||||||
|
"type": "TEST-GROUP",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Past event 2",
|
||||||
|
"start": 1660406808000,
|
||||||
|
"end": 1660429160000,
|
||||||
|
"type": "TEST-GROUP",
|
||||||
|
"color": "orange",
|
||||||
|
"textColor": "white"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"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 Strip", () => {
|
||||||
|
test("Create two Time Strips, add a single Plan to both, and verify they can have separate Indepdenent Time Contexts @unstable", async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5627'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Constant locators
|
||||||
|
const independentTimeConductorInputs = page.locator('.l-shell__main-independent-time-conductor .c-input--datetime');
|
||||||
|
const activityBounds = page.locator('.activity-bounds');
|
||||||
|
|
||||||
|
// Goto baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
const timestrip = await test.step("Create a Time Strip", async () => {
|
||||||
|
const createdTimeStrip = await createDomainObjectWithDefaults(page, { type: 'Time Strip' });
|
||||||
|
const objectName = await page.locator('.l-browse-bar__object-name').innerText();
|
||||||
|
expect(objectName).toBe(createdTimeStrip.name);
|
||||||
|
|
||||||
|
return createdTimeStrip;
|
||||||
|
});
|
||||||
|
|
||||||
|
const plan = await test.step("Create a Plan and add it to the timestrip", async () => {
|
||||||
|
const createdPlan = await createPlanFromJSON(page, {
|
||||||
|
name: 'Test Plan',
|
||||||
|
json: testPlan
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(timestrip.url);
|
||||||
|
// Expand the tree to show the plan
|
||||||
|
await page.click("button[title='Show selected item in tree']");
|
||||||
|
await page.dragAndDrop(`role=treeitem[name=/${createdPlan.name}/]`, '.c-object-view');
|
||||||
|
await page.click("button[title='Save']");
|
||||||
|
await page.click("li[title='Save and Finish Editing']");
|
||||||
|
const startBound = testPlan.TEST_GROUP[0].start;
|
||||||
|
const endBound = testPlan.TEST_GROUP[testPlan.TEST_GROUP.length - 1].end;
|
||||||
|
|
||||||
|
// Switch to fixed time mode with all plan events within the bounds
|
||||||
|
await page.goto(`${timestrip.url}?tc.mode=fixed&tc.startBound=${startBound}&tc.endBound=${endBound}&tc.timeSystem=utc&view=time-strip.view`);
|
||||||
|
|
||||||
|
// Verify all events are displayed
|
||||||
|
const eventCount = await page.locator('.activity-bounds').count();
|
||||||
|
expect(eventCount).toEqual(testPlan.TEST_GROUP.length);
|
||||||
|
|
||||||
|
return createdPlan;
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step("TimeStrip can use the Independent Time Conductor", async () => {
|
||||||
|
// Activate Independent Time Conductor in Fixed Time Mode
|
||||||
|
await page.click('.c-toggle-switch__slider');
|
||||||
|
expect(await activityBounds.count()).toEqual(0);
|
||||||
|
|
||||||
|
// Set the independent time bounds so that only one event is shown
|
||||||
|
const startBound = testPlan.TEST_GROUP[0].start;
|
||||||
|
const endBound = testPlan.TEST_GROUP[0].end;
|
||||||
|
const startBoundString = new Date(startBound).toISOString().replace('T', ' ');
|
||||||
|
const endBoundString = new Date(endBound).toISOString().replace('T', ' ');
|
||||||
|
|
||||||
|
await independentTimeConductorInputs.nth(0).fill('');
|
||||||
|
await independentTimeConductorInputs.nth(0).fill(startBoundString);
|
||||||
|
await page.keyboard.press('Enter');
|
||||||
|
await independentTimeConductorInputs.nth(1).fill('');
|
||||||
|
await independentTimeConductorInputs.nth(1).fill(endBoundString);
|
||||||
|
await page.keyboard.press('Enter');
|
||||||
|
expect(await activityBounds.count()).toEqual(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step("Can have multiple TimeStrips with the same plan linked and different Independent Time Contexts", async () => {
|
||||||
|
// Create another Time Strip and verify that it has been created
|
||||||
|
const createdTimeStrip = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Time Strip',
|
||||||
|
name: "Another Time Strip"
|
||||||
|
});
|
||||||
|
|
||||||
|
const objectName = await page.locator('.l-browse-bar__object-name').innerText();
|
||||||
|
expect(objectName).toBe(createdTimeStrip.name);
|
||||||
|
|
||||||
|
// Drag the existing Plan onto the newly created Time Strip, and save.
|
||||||
|
await page.dragAndDrop(`role=treeitem[name=/${plan.name}/]`, '.c-object-view');
|
||||||
|
await page.click("button[title='Save']");
|
||||||
|
await page.click("li[title='Save and Finish Editing']");
|
||||||
|
|
||||||
|
// Activate Independent Time Conductor in Fixed Time Mode
|
||||||
|
await page.click('.c-toggle-switch__slider');
|
||||||
|
|
||||||
|
// All events should be displayed at this point because the
|
||||||
|
// initial independent context bounds will match the global bounds
|
||||||
|
expect(await activityBounds.count()).toEqual(5);
|
||||||
|
|
||||||
|
// Set the independent time bounds so that two events are shown
|
||||||
|
const startBound = testPlan.TEST_GROUP[0].start;
|
||||||
|
const endBound = testPlan.TEST_GROUP[1].end;
|
||||||
|
const startBoundString = new Date(startBound).toISOString().replace('T', ' ');
|
||||||
|
const endBoundString = new Date(endBound).toISOString().replace('T', ' ');
|
||||||
|
|
||||||
|
await independentTimeConductorInputs.nth(0).fill('');
|
||||||
|
await independentTimeConductorInputs.nth(0).fill(startBoundString);
|
||||||
|
await page.keyboard.press('Enter');
|
||||||
|
await independentTimeConductorInputs.nth(1).fill('');
|
||||||
|
await independentTimeConductorInputs.nth(1).fill(endBoundString);
|
||||||
|
await page.keyboard.press('Enter');
|
||||||
|
|
||||||
|
// Verify that two events are displayed
|
||||||
|
expect(await activityBounds.count()).toEqual(2);
|
||||||
|
|
||||||
|
// Switch to the previous Time Strip and verify that only one event is displayed
|
||||||
|
await page.goto(timestrip.url);
|
||||||
|
expect(await activityBounds.count()).toEqual(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
@ -24,9 +24,9 @@
|
|||||||
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('@playwright/test');
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
|
||||||
test.describe('Clock Generator', () => {
|
test.describe('Clock Generator CRUD Operations', () => {
|
||||||
|
|
||||||
test('Timezone dropdown will collapse when clicked outside or on dropdown icon again', async ({ page }) => {
|
test('Timezone dropdown will collapse when clicked outside or on dropdown icon again', async ({ page }) => {
|
||||||
test.info().annotations.push({
|
test.info().annotations.push({
|
||||||
@ -34,7 +34,7 @@ test.describe('Clock Generator', () => {
|
|||||||
description: 'https://github.com/nasa/openmct/issues/4878'
|
description: 'https://github.com/nasa/openmct/issues/4878'
|
||||||
});
|
});
|
||||||
//Go to baseURL
|
//Go to baseURL
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.click('button:has-text("Create")');
|
||||||
@ -45,22 +45,22 @@ test.describe('Clock Generator', () => {
|
|||||||
// Click .icon-arrow-down
|
// Click .icon-arrow-down
|
||||||
await page.locator('.icon-arrow-down').click();
|
await page.locator('.icon-arrow-down').click();
|
||||||
//verify if the autocomplete dropdown is visible
|
//verify if the autocomplete dropdown is visible
|
||||||
await expect(page.locator(".optionPreSelected")).toBeVisible();
|
await expect(page.locator(".c-input--autocomplete__options")).toBeVisible();
|
||||||
// Click .icon-arrow-down
|
// Click .icon-arrow-down
|
||||||
await page.locator('.icon-arrow-down').click();
|
await page.locator('.icon-arrow-down').click();
|
||||||
|
|
||||||
// Verify clicking on the autocomplete arrow collapses the dropdown
|
// Verify clicking on the autocomplete arrow collapses the dropdown
|
||||||
await expect(page.locator(".optionPreSelected")).not.toBeVisible();
|
await expect(page.locator(".c-input--autocomplete__options")).toBeHidden();
|
||||||
|
|
||||||
// Click timezone input to open dropdown
|
// Click timezone input to open dropdown
|
||||||
await page.locator('.autocompleteInput').click();
|
await page.locator('.c-input--autocomplete__input').click();
|
||||||
//verify if the autocomplete dropdown is visible
|
//verify if the autocomplete dropdown is visible
|
||||||
await expect(page.locator(".optionPreSelected")).toBeVisible();
|
await expect(page.locator(".c-input--autocomplete__options")).toBeVisible();
|
||||||
|
|
||||||
// Verify clicking outside the autocomplete dropdown collapses it
|
// Verify clicking outside the autocomplete dropdown collapses it
|
||||||
await page.locator('text=Timezone').click();
|
await page.locator('text=Timezone').click();
|
||||||
// Verify clicking on the autocomplete arrow collapses the dropdown
|
// Verify clicking on the autocomplete arrow collapses the dropdown
|
||||||
await expect(page.locator(".optionPreSelected")).not.toBeVisible();
|
await expect(page.locator(".c-input--autocomplete__options")).toBeHidden();
|
||||||
|
|
||||||
});
|
});
|
||||||
});
|
});
|
41
e2e/tests/functional/plugins/clocks/remoteClock.e2e.spec.js
Normal file
41
e2e/tests/functional/plugins/clocks/remoteClock.e2e.spec.js
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
// FIXME: Remove this eslint exception once tests are implemented
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
|
||||||
|
test.describe('Remote Clock', () => {
|
||||||
|
// eslint-disable-next-line require-await
|
||||||
|
test.fixme('blocks historical requests until first tick is received', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5221'
|
||||||
|
});
|
||||||
|
// addInitScript to with remote clock
|
||||||
|
// Switch time conductor mode to 'remote clock'
|
||||||
|
// Navigate to telemetry
|
||||||
|
// Verify that the plot renders historical data within the correct bounds
|
||||||
|
// Refresh the page
|
||||||
|
// Verify again that the plot renders historical data within the correct bounds
|
||||||
|
});
|
||||||
|
});
|
@ -26,54 +26,52 @@ suite is sharing state between tests which is considered an anti-pattern. Implim
|
|||||||
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('@playwright/test');
|
const { test, expect } = require('../../../../pluginFixtures.js');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
|
||||||
let conditionSetUrl;
|
let conditionSetUrl;
|
||||||
let getConditionSetIdentifierFromUrl;
|
let getConditionSetIdentifierFromUrl;
|
||||||
|
|
||||||
test('Create new Condition Set object and store @localStorage', async ({ page, context }) => {
|
test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
||||||
//Go to baseURL
|
test.beforeAll(async ({ browser}) => {
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
//TODO: This needs to be refactored
|
||||||
|
const context = await browser.newContext();
|
||||||
//Click the Create button
|
const page = await context.newPage();
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
await page.click('button:has-text("Create")');
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
// Click text=Condition Set
|
await page.locator('li:has-text("Condition Set")').click();
|
||||||
await page.click('text=Condition Set');
|
|
||||||
|
|
||||||
// Click text=OK
|
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
page.waitForNavigation(),
|
page.waitForNavigation(),
|
||||||
page.click('text=OK')
|
page.click('text=OK')
|
||||||
]);
|
]);
|
||||||
|
|
||||||
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
|
||||||
//Save localStorage for future test execution
|
//Save localStorage for future test execution
|
||||||
await context.storageState({ path: './e2e/tests/recycled_storage.json' });
|
await context.storageState({ path: './e2e/test-data/recycled_local_storage.json' });
|
||||||
|
|
||||||
//Set object identifier from url
|
//Set object identifier from url
|
||||||
conditionSetUrl = await page.url();
|
conditionSetUrl = page.url();
|
||||||
console.log('conditionSetUrl ' + conditionSetUrl);
|
console.log('conditionSetUrl ' + conditionSetUrl);
|
||||||
|
|
||||||
getConditionSetIdentifierFromUrl = await conditionSetUrl.split('/').pop().split('?')[0];
|
getConditionSetIdentifierFromUrl = conditionSetUrl.split('/').pop().split('?')[0];
|
||||||
console.log('getConditionSetIdentifierFromUrl ' + getConditionSetIdentifierFromUrl);
|
console.debug('getConditionSetIdentifierFromUrl ' + getConditionSetIdentifierFromUrl);
|
||||||
|
await page.close();
|
||||||
|
});
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|
||||||
//Load localStorage for subsequent tests
|
//Load localStorage for subsequent tests
|
||||||
test.use({ storageState: './e2e/tests/recycled_storage.json' });
|
test.use({ storageState: './e2e/test-data/recycled_local_storage.json' });
|
||||||
|
|
||||||
//Begin suite of tests again localStorage
|
//Begin suite of tests again localStorage
|
||||||
test('Condition set object properties persist in main view and inspector', async ({ page }) => {
|
test('Condition set object properties persist in main view and inspector @localStorage', async ({ page }) => {
|
||||||
//Navigate to baseURL with injected localStorage
|
//Navigate to baseURL with injected localStorage
|
||||||
await page.goto(conditionSetUrl, { waitUntil: 'networkidle' });
|
await page.goto(conditionSetUrl, { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
//Assertions on loaded Condition Set in main view
|
//Assertions on loaded Condition Set in main view. This is a stateful transition step after page.goto()
|
||||||
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
||||||
|
|
||||||
//Assertions on loaded Condition Set in Inspector
|
//Assertions on loaded Condition Set in Inspector
|
||||||
await expect.soft(page.locator('_vue=item.name=Unnamed Condition Set')).toBeTruthy;
|
expect.soft(page.locator('_vue=item.name=Unnamed Condition Set')).toBeTruthy();
|
||||||
|
|
||||||
//Reload Page
|
//Reload Page
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
@ -84,13 +82,15 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
//Re-verify after reload
|
//Re-verify after reload
|
||||||
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
||||||
//Assertions on loaded Condition Set in Inspector
|
//Assertions on loaded Condition Set in Inspector
|
||||||
await 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 }) => {
|
test('condition set object can be modified on @localStorage', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
await page.goto(conditionSetUrl, { waitUntil: 'networkidle' });
|
await page.goto(conditionSetUrl, { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
//Assertions on loaded Condition Set in main view
|
//Assertions on loaded Condition Set in main view. This is a stateful transition step after page.goto()
|
||||||
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Condition Set');
|
||||||
|
|
||||||
//Update the Condition Set properties
|
//Update the Condition Set properties
|
||||||
@ -110,18 +110,18 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
|
|
||||||
// Verify Inspector properties
|
// Verify Inspector properties
|
||||||
// Verify Inspector has updated Name property
|
// Verify Inspector has updated Name property
|
||||||
await expect.soft(page.locator('text=Renamed Condition Set').nth(1)).toBeTruthy();
|
expect.soft(page.locator('text=Renamed Condition Set').nth(1)).toBeTruthy();
|
||||||
// Verify Inspector Details has updated Name property
|
// Verify Inspector Details has updated Name property
|
||||||
await expect.soft(page.locator('text=Renamed Condition Set').nth(2)).toBeTruthy();
|
expect.soft(page.locator('text=Renamed Condition Set').nth(2)).toBeTruthy();
|
||||||
|
|
||||||
// Verify Tree reflects updated Name proprety
|
// Verify Tree reflects updated Name proprety
|
||||||
// Expand Tree
|
// Expand Tree
|
||||||
await page.locator('text=Open MCT My Items >> span >> nth=3').click();
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span >> nth=3`).click();
|
||||||
// Verify Condition Set Object is renamed in Tree
|
// Verify Condition Set Object is renamed in Tree
|
||||||
await expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
||||||
// Verify Search Tree reflects renamed Name property
|
// Verify Search Tree reflects renamed Name property
|
||||||
await page.locator('input[type="search"]').fill('Renamed');
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Renamed');
|
||||||
await expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
||||||
|
|
||||||
//Reload Page
|
//Reload Page
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
@ -134,45 +134,43 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
|
|
||||||
// Verify Inspector properties
|
// Verify Inspector properties
|
||||||
// Verify Inspector has updated Name property
|
// Verify Inspector has updated Name property
|
||||||
await expect.soft(page.locator('text=Renamed Condition Set').nth(1)).toBeTruthy();
|
expect.soft(page.locator('text=Renamed Condition Set').nth(1)).toBeTruthy();
|
||||||
// Verify Inspector Details has updated Name property
|
// Verify Inspector Details has updated Name property
|
||||||
await expect.soft(page.locator('text=Renamed Condition Set').nth(2)).toBeTruthy();
|
expect.soft(page.locator('text=Renamed Condition Set').nth(2)).toBeTruthy();
|
||||||
|
|
||||||
// Verify Tree reflects updated Name proprety
|
// Verify Tree reflects updated Name proprety
|
||||||
// Expand Tree
|
// Expand Tree
|
||||||
await page.locator('text=Open MCT My Items >> span >> nth=3').click();
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span >> nth=3`).click();
|
||||||
// Verify Condition Set Object is renamed in Tree
|
// Verify Condition Set Object is renamed in Tree
|
||||||
await expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
||||||
// Verify Search Tree reflects renamed Name property
|
// Verify Search Tree reflects renamed Name property
|
||||||
await page.locator('input[type="search"]').fill('Renamed');
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Renamed');
|
||||||
await expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
expect(page.locator('a:has-text("Renamed Condition Set")')).toBeTruthy();
|
||||||
});
|
});
|
||||||
test('condition set object can be deleted by Search Tree Actions menu on @localStorage', async ({ page }) => {
|
test('condition set object can be deleted by Search Tree Actions menu on @localStorage', async ({ page }) => {
|
||||||
//Navigate to baseURL
|
//Navigate to baseURL
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
//Expect Unnamed Condition Set to be visible in Main View
|
//Assertions on loaded Condition Set in main view. This is a stateful transition step after page.goto()
|
||||||
await expect(page.locator('a:has-text("Unnamed Condition Set Condition Set")')).toBeVisible();
|
await expect(page.locator('a:has-text("Unnamed Condition Set Condition Set") >> nth=0')).toBeVisible();
|
||||||
|
|
||||||
|
const numberOfConditionSetsToStart = await page.locator('a:has-text("Unnamed Condition Set Condition Set")').count();
|
||||||
|
|
||||||
// Search for Unnamed Condition Set
|
// Search for Unnamed Condition Set
|
||||||
await page.locator('input[type="search"]').fill('Unnamed Condition Set');
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Unnamed Condition Set');
|
||||||
// Right Click to Open Actions Menu
|
// Click Search Result
|
||||||
await page.locator('a:has-text("Unnamed Condition Set")').click({
|
await page.locator('[aria-label="OpenMCT Search"] >> text=Unnamed Condition Set').first().click();
|
||||||
button: 'right'
|
// Click hamburger button
|
||||||
});
|
await page.locator('[title="More options"]').click();
|
||||||
// Click Remove Action
|
|
||||||
await page.locator('text=Remove').click();
|
|
||||||
|
|
||||||
|
// Click text=Remove
|
||||||
|
await page.locator('text=Remove').click();
|
||||||
await page.locator('text=OK').click();
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
//Expect Unnamed Condition Set to be removed in Main View
|
//Expect Unnamed Condition Set to be removed in Main View
|
||||||
await expect(page.locator('a:has-text("Unnamed Condition Set Condition Set")')).not.toBeVisible();
|
const numberOfConditionSetsAtEnd = await page.locator('a:has-text("Unnamed Condition Set Condition Set")').count();
|
||||||
|
|
||||||
await page.locator('.c-search__clear-input').click();
|
expect(numberOfConditionSetsAtEnd).toEqual(numberOfConditionSetsToStart - 1);
|
||||||
// Search for Unnamed Condition Set
|
|
||||||
await page.locator('input[type="search"]').fill('Unnamed Condition Set');
|
|
||||||
// Expect Unnamed Condition Set to be removed
|
|
||||||
await expect(page.locator('a:has-text("Unnamed Condition Set")')).not.toBeVisible();
|
|
||||||
|
|
||||||
//Feature?
|
//Feature?
|
||||||
//Domain Object is still available by direct URL after delete
|
//Domain Object is still available by direct URL after delete
|
||||||
@ -181,3 +179,24 @@ test.describe.serial('Condition Set CRUD Operations on @localStorage', () => {
|
|||||||
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test.describe('Basic Condition Set Use', () => {
|
||||||
|
test('Can add a condition', async ({ page }) => {
|
||||||
|
//Navigate to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create a new condition set
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Condition Set',
|
||||||
|
name: "Test Condition Set"
|
||||||
|
});
|
||||||
|
// Change the object to edit mode
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
|
||||||
|
// Click Add Condition button
|
||||||
|
await page.locator('#addCondition').click();
|
||||||
|
// Check that the new Unnamed Condition section appears
|
||||||
|
const numOfUnnamedConditions = await page.locator('text=Unnamed Condition').count();
|
||||||
|
expect(numOfUnnamedConditions).toEqual(1);
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,186 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults, setStartOffset, setFixedTimeMode, setRealTimeMode } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Testing Display Layout @unstable', () => {
|
||||||
|
let sineWaveObject;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// Create Sine Wave Generator
|
||||||
|
sineWaveObject = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: "Test Sine Wave Generator"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('alpha-numeric widget telemetry value exactly matches latest telemetry value received in real time', async ({ page }) => {
|
||||||
|
// Create a Display Layout
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: "Test Display Layout"
|
||||||
|
});
|
||||||
|
// Edit Display Layout
|
||||||
|
await page.locator('[title="Edit"]').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 Display Layout and save changes
|
||||||
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.l-layout__grid-holder');
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
// Subscribe to the Sine Wave Generator data
|
||||||
|
// On getting data, check if the value found in the Display Layout is the most recent value
|
||||||
|
// from the Sine Wave Generator
|
||||||
|
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
||||||
|
const formattedTelemetryValue = await getTelemValuePromise;
|
||||||
|
const displayLayoutValuePromise = await page.waitForSelector(`text="${formattedTelemetryValue}"`);
|
||||||
|
const displayLayoutValue = await displayLayoutValuePromise.textContent();
|
||||||
|
const trimmedDisplayValue = displayLayoutValue.trim();
|
||||||
|
|
||||||
|
await expect(trimmedDisplayValue).toBe(formattedTelemetryValue);
|
||||||
|
});
|
||||||
|
test('alpha-numeric widget telemetry value exactly matches latest telemetry value received in fixed time', async ({ page }) => {
|
||||||
|
// Create a Display Layout
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: "Test Display Layout"
|
||||||
|
});
|
||||||
|
// Edit Display Layout
|
||||||
|
await page.locator('[title="Edit"]').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 Display Layout and save changes
|
||||||
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.l-layout__grid-holder');
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
// Subscribe to the Sine Wave Generator data
|
||||||
|
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
||||||
|
// Set an offset of 1 minute and then change the time mode to fixed to set a 1 minute historical window
|
||||||
|
await setStartOffset(page, { mins: '1' });
|
||||||
|
await setFixedTimeMode(page);
|
||||||
|
|
||||||
|
// On getting data, check if the value found in the Display Layout is the most recent value
|
||||||
|
// from the Sine Wave Generator
|
||||||
|
const formattedTelemetryValue = await getTelemValuePromise;
|
||||||
|
const displayLayoutValuePromise = await page.waitForSelector(`text="${formattedTelemetryValue}"`);
|
||||||
|
const displayLayoutValue = await displayLayoutValuePromise.textContent();
|
||||||
|
const trimmedDisplayValue = displayLayoutValue.trim();
|
||||||
|
|
||||||
|
await expect(trimmedDisplayValue).toBe(formattedTelemetryValue);
|
||||||
|
});
|
||||||
|
test('items in a display layout can be removed with object tree context menu when viewing the display layout', async ({ page }) => {
|
||||||
|
// Create a Display Layout
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: "Test Display Layout"
|
||||||
|
});
|
||||||
|
// Edit Display Layout
|
||||||
|
await page.locator('[title="Edit"]').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 Display Layout and save changes
|
||||||
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.l-layout__grid-holder');
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(1);
|
||||||
|
|
||||||
|
// Expand the Display Layout so we can remove the sine wave generator
|
||||||
|
await page.locator('.c-tree__item.is-navigated-object .c-disclosure-triangle').click();
|
||||||
|
|
||||||
|
// Bring up context menu and remove
|
||||||
|
await page.locator('.c-tree__item.is-alias .c-tree__item__name:text("Test Sine Wave Generator")').first().click({ button: 'right' });
|
||||||
|
await page.locator('text=Remove').click();
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
// delete
|
||||||
|
|
||||||
|
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(0);
|
||||||
|
});
|
||||||
|
test('items in a display layout can be removed with object tree context menu when viewing another item', async ({ page }) => {
|
||||||
|
// Create a Display Layout
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Display Layout',
|
||||||
|
name: "Test Display Layout"
|
||||||
|
});
|
||||||
|
// Edit Display Layout
|
||||||
|
await page.locator('[title="Edit"]').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 Display Layout and save changes
|
||||||
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.l-layout__grid-holder');
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(1);
|
||||||
|
|
||||||
|
// Expand the Display Layout so we can remove the sine wave generator
|
||||||
|
await page.locator('.c-tree__item.is-navigated-object .c-disclosure-triangle').click();
|
||||||
|
|
||||||
|
// Click the original Sine Wave Generator to navigate away from the Display Layout
|
||||||
|
await page.locator('.c-tree__item .c-tree__item__name:text("Test Sine Wave Generator")').click();
|
||||||
|
|
||||||
|
// Bring up context menu and remove
|
||||||
|
await page.locator('.c-tree__item.is-alias .c-tree__item__name:text("Test Sine Wave Generator")').click({ button: 'right' });
|
||||||
|
await page.locator('text=Remove').click();
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
// navigate back to the display layout to confirm it has been removed
|
||||||
|
await page.locator('.c-tree__item .c-tree__item__name:text("Test Display Layout")').click();
|
||||||
|
|
||||||
|
expect.soft(await page.locator('.l-layout .l-layout__frame').count()).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Util for subscribing to a telemetry object by object identifier
|
||||||
|
* Limitations: Currently only works to return telemetry once to the node scope
|
||||||
|
* To Do: See if there's a way to await this multiple times to allow for multiple
|
||||||
|
* values to be returned over time
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} objectIdentifier identifier for object
|
||||||
|
* @returns {Promise<string>} the formatted sin telemetry value
|
||||||
|
*/
|
||||||
|
async function subscribeToTelemetry(page, objectIdentifier) {
|
||||||
|
const getTelemValuePromise = new Promise(resolve => page.exposeFunction('getTelemValue', resolve));
|
||||||
|
|
||||||
|
await page.evaluate(async (telemetryIdentifier) => {
|
||||||
|
const telemetryObject = await window.openmct.objects.get(telemetryIdentifier);
|
||||||
|
const metadata = window.openmct.telemetry.getMetadata(telemetryObject);
|
||||||
|
const formats = await window.openmct.telemetry.getFormatMap(metadata);
|
||||||
|
window.openmct.telemetry.subscribe(telemetryObject, (obj) => {
|
||||||
|
const sinVal = obj.sin;
|
||||||
|
const formattedSinVal = formats.sin.format(sinVal);
|
||||||
|
window.getTelemValue(formattedSinVal);
|
||||||
|
});
|
||||||
|
}, objectIdentifier);
|
||||||
|
|
||||||
|
return getTelemValuePromise;
|
||||||
|
}
|
@ -0,0 +1,237 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const utils = require('../../../../helper/faultUtils');
|
||||||
|
|
||||||
|
test.describe('The Fault Management Plugin using example faults', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await utils.navigateToFaultManagementWithExample(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Shows a criticality icon for every fault @unstable', async ({ page }) => {
|
||||||
|
const faultCount = await page.locator('c-fault-mgmt__list').count();
|
||||||
|
const criticalityIconCount = await page.locator('c-fault-mgmt__list-severity').count();
|
||||||
|
|
||||||
|
expect.soft(faultCount).toEqual(criticalityIconCount);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('When selecting a fault, it has an "is-selected" class and it\'s information shows in the inspector @unstable', async ({ page }) => {
|
||||||
|
await utils.selectFaultItem(page, 1);
|
||||||
|
|
||||||
|
const selectedFaultName = await page.locator('.c-fault-mgmt__list.is-selected .c-fault-mgmt__list-faultname').textContent();
|
||||||
|
const inspectorFaultNameCount = await page.locator(`.c-inspector__properties >> :text("${selectedFaultName}")`).count();
|
||||||
|
|
||||||
|
await expect.soft(page.locator('.c-faults-list-view-item-body > .c-fault-mgmt__list').first()).toHaveClass(/is-selected/);
|
||||||
|
expect.soft(inspectorFaultNameCount).toEqual(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('When selecting multiple faults, no specific fault information is shown in the inspector @unstable', async ({ page }) => {
|
||||||
|
await utils.selectFaultItem(page, 1);
|
||||||
|
await utils.selectFaultItem(page, 2);
|
||||||
|
|
||||||
|
const selectedRows = page.locator('.c-fault-mgmt__list.is-selected .c-fault-mgmt__list-faultname');
|
||||||
|
expect.soft(await selectedRows.count()).toEqual(2);
|
||||||
|
|
||||||
|
const firstSelectedFaultName = await selectedRows.nth(0).textContent();
|
||||||
|
const secondSelectedFaultName = await selectedRows.nth(1).textContent();
|
||||||
|
const firstNameInInspectorCount = await page.locator(`.c-inspector__properties >> :text("${firstSelectedFaultName}")`).count();
|
||||||
|
const secondNameInInspectorCount = await page.locator(`.c-inspector__properties >> :text("${secondSelectedFaultName}")`).count();
|
||||||
|
|
||||||
|
expect.soft(firstNameInInspectorCount).toEqual(0);
|
||||||
|
expect.soft(secondNameInInspectorCount).toEqual(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows you to shelve a fault @unstable', async ({ page }) => {
|
||||||
|
const shelvedFaultName = await utils.getFaultName(page, 2);
|
||||||
|
const beforeShelvedFault = utils.getFaultByName(page, shelvedFaultName);
|
||||||
|
|
||||||
|
expect.soft(await beforeShelvedFault.count()).toBe(1);
|
||||||
|
|
||||||
|
await utils.shelveFault(page, 2);
|
||||||
|
|
||||||
|
// check it is removed from standard view
|
||||||
|
const afterShelvedFault = utils.getFaultByName(page, shelvedFaultName);
|
||||||
|
expect.soft(await afterShelvedFault.count()).toBe(0);
|
||||||
|
|
||||||
|
await utils.changeViewTo(page, 'shelved');
|
||||||
|
|
||||||
|
const shelvedViewFault = utils.getFaultByName(page, shelvedFaultName);
|
||||||
|
|
||||||
|
expect.soft(await shelvedViewFault.count()).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows you to acknowledge a fault @unstable', async ({ page }) => {
|
||||||
|
const acknowledgedFaultName = await utils.getFaultName(page, 3);
|
||||||
|
|
||||||
|
await utils.acknowledgeFault(page, 3);
|
||||||
|
|
||||||
|
const fault = utils.getFault(page, 3);
|
||||||
|
await expect.soft(fault).toHaveClass(/is-acknowledged/);
|
||||||
|
|
||||||
|
await utils.changeViewTo(page, 'acknowledged');
|
||||||
|
|
||||||
|
const acknowledgedViewFaultName = await utils.getFaultName(page, 1);
|
||||||
|
expect.soft(acknowledgedFaultName).toEqual(acknowledgedViewFaultName);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows you to shelve multiple faults @unstable', async ({ page }) => {
|
||||||
|
const shelvedFaultNameOne = await utils.getFaultName(page, 1);
|
||||||
|
const shelvedFaultNameFour = await utils.getFaultName(page, 4);
|
||||||
|
|
||||||
|
const beforeShelvedFaultOne = utils.getFaultByName(page, shelvedFaultNameOne);
|
||||||
|
const beforeShelvedFaultFour = utils.getFaultByName(page, shelvedFaultNameFour);
|
||||||
|
|
||||||
|
expect.soft(await beforeShelvedFaultOne.count()).toBe(1);
|
||||||
|
expect.soft(await beforeShelvedFaultFour.count()).toBe(1);
|
||||||
|
|
||||||
|
await utils.shelveMultipleFaults(page, 1, 4);
|
||||||
|
|
||||||
|
// check it is removed from standard view
|
||||||
|
const afterShelvedFaultOne = utils.getFaultByName(page, shelvedFaultNameOne);
|
||||||
|
const afterShelvedFaultFour = utils.getFaultByName(page, shelvedFaultNameFour);
|
||||||
|
expect.soft(await afterShelvedFaultOne.count()).toBe(0);
|
||||||
|
expect.soft(await afterShelvedFaultFour.count()).toBe(0);
|
||||||
|
|
||||||
|
await utils.changeViewTo(page, 'shelved');
|
||||||
|
|
||||||
|
const shelvedViewFaultOne = utils.getFaultByName(page, shelvedFaultNameOne);
|
||||||
|
const shelvedViewFaultFour = utils.getFaultByName(page, shelvedFaultNameFour);
|
||||||
|
|
||||||
|
expect.soft(await shelvedViewFaultOne.count()).toBe(1);
|
||||||
|
expect.soft(await shelvedViewFaultFour.count()).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows you to acknowledge multiple faults @unstable', async ({ page }) => {
|
||||||
|
const acknowledgedFaultNameTwo = await utils.getFaultName(page, 2);
|
||||||
|
const acknowledgedFaultNameFive = await utils.getFaultName(page, 5);
|
||||||
|
|
||||||
|
await utils.acknowledgeMultipleFaults(page, 2, 5);
|
||||||
|
|
||||||
|
const faultTwo = utils.getFault(page, 2);
|
||||||
|
const faultFive = utils.getFault(page, 5);
|
||||||
|
|
||||||
|
// check they have been acknowledged
|
||||||
|
await expect.soft(faultTwo).toHaveClass(/is-acknowledged/);
|
||||||
|
await expect.soft(faultFive).toHaveClass(/is-acknowledged/);
|
||||||
|
|
||||||
|
await utils.changeViewTo(page, 'acknowledged');
|
||||||
|
|
||||||
|
const acknowledgedViewFaultTwo = utils.getFaultByName(page, acknowledgedFaultNameTwo);
|
||||||
|
const acknowledgedViewFaultFive = utils.getFaultByName(page, acknowledgedFaultNameFive);
|
||||||
|
|
||||||
|
expect.soft(await acknowledgedViewFaultTwo.count()).toBe(1);
|
||||||
|
expect.soft(await acknowledgedViewFaultFive.count()).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows you to search faults @unstable', async ({ page }) => {
|
||||||
|
const faultThreeNamespace = await utils.getFaultNamespace(page, 3);
|
||||||
|
const faultTwoName = await utils.getFaultName(page, 2);
|
||||||
|
const faultFiveTriggerTime = await utils.getFaultTriggerTime(page, 5);
|
||||||
|
|
||||||
|
// should be all faults (5)
|
||||||
|
let faultResultCount = await utils.getFaultResultCount(page);
|
||||||
|
expect.soft(faultResultCount).toEqual(5);
|
||||||
|
|
||||||
|
// search namespace
|
||||||
|
await utils.enterSearchTerm(page, faultThreeNamespace);
|
||||||
|
|
||||||
|
faultResultCount = await utils.getFaultResultCount(page);
|
||||||
|
expect.soft(faultResultCount).toEqual(1);
|
||||||
|
expect.soft(await utils.getFaultNamespace(page, 1)).toEqual(faultThreeNamespace);
|
||||||
|
|
||||||
|
// all faults
|
||||||
|
await utils.clearSearch(page);
|
||||||
|
faultResultCount = await utils.getFaultResultCount(page);
|
||||||
|
expect.soft(faultResultCount).toEqual(5);
|
||||||
|
|
||||||
|
// search name
|
||||||
|
await utils.enterSearchTerm(page, faultTwoName);
|
||||||
|
|
||||||
|
faultResultCount = await utils.getFaultResultCount(page);
|
||||||
|
expect.soft(faultResultCount).toEqual(1);
|
||||||
|
expect.soft(await utils.getFaultName(page, 1)).toEqual(faultTwoName);
|
||||||
|
|
||||||
|
// all faults
|
||||||
|
await utils.clearSearch(page);
|
||||||
|
faultResultCount = await utils.getFaultResultCount(page);
|
||||||
|
expect.soft(faultResultCount).toEqual(5);
|
||||||
|
|
||||||
|
// search triggerTime
|
||||||
|
await utils.enterSearchTerm(page, faultFiveTriggerTime);
|
||||||
|
|
||||||
|
faultResultCount = await utils.getFaultResultCount(page);
|
||||||
|
expect.soft(faultResultCount).toEqual(1);
|
||||||
|
expect.soft(await utils.getFaultTriggerTime(page, 1)).toEqual(faultFiveTriggerTime);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows you to sort faults @unstable', async ({ page }) => {
|
||||||
|
const highestSeverity = await utils.getHighestSeverity(page);
|
||||||
|
const lowestSeverity = await utils.getLowestSeverity(page);
|
||||||
|
const faultOneName = 'Example Fault 1';
|
||||||
|
const faultFiveName = 'Example Fault 5';
|
||||||
|
let firstFaultName = await utils.getFaultName(page, 1);
|
||||||
|
|
||||||
|
expect.soft(firstFaultName).toEqual(faultOneName);
|
||||||
|
|
||||||
|
await utils.sortFaultsBy(page, 'oldest-first');
|
||||||
|
|
||||||
|
firstFaultName = await utils.getFaultName(page, 1);
|
||||||
|
expect.soft(firstFaultName).toEqual(faultFiveName);
|
||||||
|
|
||||||
|
await utils.sortFaultsBy(page, 'severity');
|
||||||
|
|
||||||
|
const sortedHighestSeverity = await utils.getFaultSeverity(page, 1);
|
||||||
|
const sortedLowestSeverity = await utils.getFaultSeverity(page, 5);
|
||||||
|
expect.soft(sortedHighestSeverity).toEqual(highestSeverity);
|
||||||
|
expect.soft(sortedLowestSeverity).toEqual(lowestSeverity);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('The Fault Management Plugin without using example faults', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await utils.navigateToFaultManagementWithoutExample(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Shows no faults when no faults are provided @unstable', async ({ page }) => {
|
||||||
|
const faultCount = await page.locator('c-fault-mgmt__list').count();
|
||||||
|
|
||||||
|
expect.soft(faultCount).toEqual(0);
|
||||||
|
|
||||||
|
await utils.changeViewTo(page, 'acknowledged');
|
||||||
|
const acknowledgedCount = await page.locator('c-fault-mgmt__list').count();
|
||||||
|
expect.soft(acknowledgedCount).toEqual(0);
|
||||||
|
|
||||||
|
await utils.changeViewTo(page, 'shelved');
|
||||||
|
const shelvedCount = await page.locator('c-fault-mgmt__list').count();
|
||||||
|
expect.soft(shelvedCount).toEqual(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Will return no faults when searching @unstable', async ({ page }) => {
|
||||||
|
await utils.enterSearchTerm(page, 'fault');
|
||||||
|
|
||||||
|
const faultCount = await page.locator('c-fault-mgmt__list').count();
|
||||||
|
|
||||||
|
expect.soft(faultCount).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,66 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Testing Flexible Layout @unstable', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create Sine Wave Generator
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: "Test Sine Wave Generator"
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create Clock Object
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Clock',
|
||||||
|
name: "Test Clock"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('panes have the appropriate draggable attribute while in Edit and Browse modes', async ({ page }) => {
|
||||||
|
// Create a Flexible Layout
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Flexible Layout',
|
||||||
|
name: "Test Flexible Layout"
|
||||||
|
});
|
||||||
|
// Edit Flexible Layout
|
||||||
|
await page.locator('[title="Edit"]').click();
|
||||||
|
|
||||||
|
// Expand the 'My Items' folder in the left tree
|
||||||
|
await page.locator('.c-tree__item__view-control.c-disclosure-triangle').first().click();
|
||||||
|
// Add the Sine Wave Generator and Clock to the Flexible Layout
|
||||||
|
await page.dragAndDrop('text=Test Sine Wave Generator', '.c-fl__container.is-empty');
|
||||||
|
await page.dragAndDrop('text=Test Clock', '.c-fl__container.is-empty');
|
||||||
|
// Check that panes can be dragged while Flexible Layout is in Edit mode
|
||||||
|
let dragWrapper = await page.locator('.c-fl-container__frames-holder .c-fl-frame__drag-wrapper').first();
|
||||||
|
await expect(dragWrapper).toHaveAttribute('draggable', 'true');
|
||||||
|
// Save Flexible Layout
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
// Check that panes are not draggable while Flexible Layout is in Browse mode
|
||||||
|
dragWrapper = await page.locator('.c-fl-container__frames-holder .c-fl-frame__drag-wrapper').first();
|
||||||
|
await expect(dragWrapper).toHaveAttribute('draggable', 'false');
|
||||||
|
});
|
||||||
|
});
|
720
e2e/tests/functional/plugins/imagery/exampleImagery.e2e.spec.js
Normal file
720
e2e/tests/functional/plugins/imagery/exampleImagery.e2e.spec.js
Normal file
@ -0,0 +1,720 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding imagery,
|
||||||
|
but only assume that example imagery is present.
|
||||||
|
*/
|
||||||
|
/* globals process */
|
||||||
|
const { v4: uuid } = require('uuid');
|
||||||
|
const { waitForAnimations } = require('../../../../baseFixtures');
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
||||||
|
const panHotkey = process.platform === 'linux' ? ['Control', 'Alt'] : ['Alt'];
|
||||||
|
const expectedAltText = process.platform === 'linux' ? 'Ctrl+Alt drag to pan' : 'Alt drag to pan';
|
||||||
|
|
||||||
|
//The following block of tests verifies the basic functionality of example imagery and serves as a template for Imagery objects embedded in other objects.
|
||||||
|
test.describe('Example Imagery Object', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create a default 'Example Imagery' object
|
||||||
|
await createDomainObjectWithDefaults(page, { type: 'Example Imagery' });
|
||||||
|
|
||||||
|
// Verify that the created object is focused
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Example Imagery');
|
||||||
|
await page.locator(backgroundImageSelector).hover({trial: true});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can use Mouse Wheel to zoom in and out of latest image', async ({ page }) => {
|
||||||
|
// Zoom in x2 and assert
|
||||||
|
await mouseZoomOnImageAndAssert(page, 2);
|
||||||
|
|
||||||
|
// Zoom out x2 and assert
|
||||||
|
await mouseZoomOnImageAndAssert(page, -2);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can adjust image brightness/contrast by dragging the sliders', async ({ page, browserName }) => {
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip(browserName === 'firefox', 'This test needs to be updated to work with firefox');
|
||||||
|
// Open the image filter menu
|
||||||
|
await page.locator('[role=toolbar] button[title="Brightness and contrast"]').click();
|
||||||
|
|
||||||
|
// Drag the brightness and contrast sliders around and assert filter values
|
||||||
|
await dragBrightnessSliderAndAssertFilterValues(page);
|
||||||
|
await dragContrastSliderAndAssertFilterValues(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can use alt+drag to move around image once zoomed in', async ({ page }) => {
|
||||||
|
const deltaYStep = 100; //equivalent to 1x zoom
|
||||||
|
|
||||||
|
await page.locator(backgroundImageSelector).hover({trial: true});
|
||||||
|
|
||||||
|
// zoom in
|
||||||
|
await page.mouse.wheel(0, deltaYStep * 2);
|
||||||
|
await page.locator(backgroundImageSelector).hover({trial: true});
|
||||||
|
const zoomedBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
const imageCenterX = zoomedBoundingBox.x + zoomedBoundingBox.width / 2;
|
||||||
|
const imageCenterY = zoomedBoundingBox.y + zoomedBoundingBox.height / 2;
|
||||||
|
// move to the right
|
||||||
|
|
||||||
|
// center the mouse pointer
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY);
|
||||||
|
|
||||||
|
//Get Diagnostic info about process environment
|
||||||
|
console.log('process.platform is ' + process.platform);
|
||||||
|
const getUA = await page.evaluate(() => navigator.userAgent);
|
||||||
|
console.log('navigator.userAgent ' + getUA);
|
||||||
|
// Pan Imagery Hints
|
||||||
|
const imageryHintsText = await page.locator('.c-imagery__hints').innerText();
|
||||||
|
expect(expectedAltText).toEqual(imageryHintsText);
|
||||||
|
|
||||||
|
// pan right
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX - 200, imageCenterY, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterRightPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(zoomedBoundingBox.x).toBeGreaterThan(afterRightPanBoundingBox.x);
|
||||||
|
|
||||||
|
// pan left
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterLeftPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(afterRightPanBoundingBox.x).toBeLessThan(afterLeftPanBoundingBox.x);
|
||||||
|
|
||||||
|
// pan up
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY);
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY + 200, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterUpPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(afterUpPanBoundingBox.y).toBeGreaterThan(afterLeftPanBoundingBox.y);
|
||||||
|
|
||||||
|
// pan down
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY - 200, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterDownPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(afterDownPanBoundingBox.y).toBeLessThan(afterUpPanBoundingBox.y);
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can use + - buttons to zoom on the image @unstable', async ({ page }) => {
|
||||||
|
await buttonZoomOnImageAndAssert(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can use the reset button to reset the image @unstable', async ({ page }, testInfo) => {
|
||||||
|
test.slow(testInfo.project === 'chrome-beta', "This test is slow in chrome-beta");
|
||||||
|
// Get initial image dimensions
|
||||||
|
const initialBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
|
||||||
|
// Zoom in twice via button
|
||||||
|
await zoomIntoImageryByButton(page);
|
||||||
|
await zoomIntoImageryByButton(page);
|
||||||
|
|
||||||
|
// Get and assert zoomed in image dimensions
|
||||||
|
const zoomedInBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect.soft(zoomedInBoundingBox.height).toBeGreaterThan(initialBoundingBox.height);
|
||||||
|
expect.soft(zoomedInBoundingBox.width).toBeGreaterThan(initialBoundingBox.width);
|
||||||
|
|
||||||
|
// Reset pan and zoom and assert against initial image dimensions
|
||||||
|
await resetImageryPanAndZoom(page);
|
||||||
|
const finalBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(finalBoundingBox).toEqual(initialBoundingBox);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Using the zoom features does not pause telemetry', async ({ page }) => {
|
||||||
|
const pausePlayButton = page.locator('.c-button.pause-play');
|
||||||
|
|
||||||
|
// open the time conductor drop down
|
||||||
|
await page.locator('.c-mode-button').click();
|
||||||
|
|
||||||
|
// Click local clock
|
||||||
|
await page.locator('[data-testid="conductor-modeOption-realtime"]').click();
|
||||||
|
await expect.soft(pausePlayButton).not.toHaveClass(/is-paused/);
|
||||||
|
|
||||||
|
// Zoom in via button
|
||||||
|
await zoomIntoImageryByButton(page);
|
||||||
|
await expect(pausePlayButton).not.toHaveClass(/is-paused/);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Uses low fetch priority', async ({ page }) => {
|
||||||
|
const priority = await page.locator('.js-imageryView-image').getAttribute('fetchpriority');
|
||||||
|
await expect(priority).toBe('low');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Example Imagery in Display Layout', () => {
|
||||||
|
let displayLayout;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
displayLayout = await createDomainObjectWithDefaults(page, { type: 'Display Layout' });
|
||||||
|
await page.goto(displayLayout.url);
|
||||||
|
|
||||||
|
/* Create Sine Wave Generator with minimum Image Load Delay */
|
||||||
|
// Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click text=Example Imagery
|
||||||
|
await page.click('text=Example Imagery');
|
||||||
|
|
||||||
|
// Clear and set Image load delay to minimum value
|
||||||
|
await page.locator('input[type="number"]').fill('');
|
||||||
|
await page.locator('input[type="number"]').fill('5000');
|
||||||
|
|
||||||
|
// Click text=OK
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({waitUntil: 'networkidle'}),
|
||||||
|
page.click('text=OK'),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Example Imagery');
|
||||||
|
|
||||||
|
await page.goto(displayLayout.url);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Imagery View operations @unstable', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5265'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Edit mode
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
// Click on example imagery to expose toolbar
|
||||||
|
await page.locator('.c-so-view__header').click();
|
||||||
|
|
||||||
|
// Adjust object height
|
||||||
|
await page.locator('div[title="Resize object height"] > input').click();
|
||||||
|
await page.locator('div[title="Resize object height"] > input').fill('50');
|
||||||
|
|
||||||
|
// Adjust object width
|
||||||
|
await page.locator('div[title="Resize object width"] > input').click();
|
||||||
|
await page.locator('div[title="Resize object width"] > input').fill('50');
|
||||||
|
|
||||||
|
await performImageryViewOperationsAndAssert(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Resizing the layout changes thumbnail visibility and size', async ({ page }) => {
|
||||||
|
const thumbsWrapperLocator = page.locator('.c-imagery__thumbs-wrapper');
|
||||||
|
// Edit mode
|
||||||
|
await page.click('button[title="Edit"]');
|
||||||
|
|
||||||
|
// Click on example imagery to expose toolbar
|
||||||
|
await page.locator('.c-so-view__header').click();
|
||||||
|
|
||||||
|
// expect thumbnails not be visible when first added
|
||||||
|
expect.soft(thumbsWrapperLocator.isHidden()).toBeTruthy();
|
||||||
|
|
||||||
|
// Resize the example imagery vertically to change the thumbnail visibility
|
||||||
|
/*
|
||||||
|
The following arbitrary values are added to observe the separate visual
|
||||||
|
conditions of the thumbnails (hidden, small thumbnails, regular thumbnails).
|
||||||
|
Specifically, height is set to 50px for small thumbs and 100px for regular
|
||||||
|
*/
|
||||||
|
await page.locator('div[title="Resize object height"] > input').click();
|
||||||
|
await page.locator('div[title="Resize object height"] > input').fill('50');
|
||||||
|
|
||||||
|
expect(thumbsWrapperLocator.isVisible()).toBeTruthy();
|
||||||
|
await expect(thumbsWrapperLocator).toHaveClass(/is-small-thumbs/);
|
||||||
|
|
||||||
|
// Resize the example imagery vertically to change the thumbnail visibility
|
||||||
|
await page.locator('div[title="Resize object height"] > input').click();
|
||||||
|
await page.locator('div[title="Resize object height"] > input').fill('100');
|
||||||
|
|
||||||
|
expect(thumbsWrapperLocator.isVisible()).toBeTruthy();
|
||||||
|
await expect(thumbsWrapperLocator).not.toHaveClass(/is-small-thumbs/);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Example Imagery in Flexible layout', () => {
|
||||||
|
let flexibleLayout;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
flexibleLayout = await createDomainObjectWithDefaults(page, { type: 'Flexible Layout' });
|
||||||
|
await page.goto(flexibleLayout.url);
|
||||||
|
|
||||||
|
/* Create Sine Wave Generator with minimum Image Load Delay */
|
||||||
|
// Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click text=Example Imagery
|
||||||
|
await page.click('text=Example Imagery');
|
||||||
|
|
||||||
|
// Clear and set Image load delay to minimum value
|
||||||
|
await page.locator('input[type="number"]').fill('');
|
||||||
|
await page.locator('input[type="number"]').fill('5000');
|
||||||
|
|
||||||
|
// Click text=OK
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({waitUntil: 'networkidle'}),
|
||||||
|
page.click('text=OK'),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Example Imagery');
|
||||||
|
|
||||||
|
await page.goto(flexibleLayout.url);
|
||||||
|
});
|
||||||
|
test('Imagery View operations @unstable', async ({ page, browserName }) => {
|
||||||
|
test.fixme(browserName === 'firefox', 'This test needs to be updated to work with firefox');
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5326'
|
||||||
|
});
|
||||||
|
|
||||||
|
await performImageryViewOperationsAndAssert(page);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Example Imagery in Tabs View', () => {
|
||||||
|
let tabsView;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
tabsView = await createDomainObjectWithDefaults(page, { type: 'Tabs View' });
|
||||||
|
await page.goto(tabsView.url);
|
||||||
|
|
||||||
|
/* Create Sine Wave Generator with minimum Image Load Delay */
|
||||||
|
// Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
// Click text=Example Imagery
|
||||||
|
await page.click('text=Example Imagery');
|
||||||
|
|
||||||
|
// Clear and set Image load delay to minimum value
|
||||||
|
await page.locator('input[type="number"]').fill('');
|
||||||
|
await page.locator('input[type="number"]').fill('5000');
|
||||||
|
|
||||||
|
// Click text=OK
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({waitUntil: 'networkidle'}),
|
||||||
|
page.click('text=OK'),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Example Imagery');
|
||||||
|
|
||||||
|
await page.goto(tabsView.url);
|
||||||
|
});
|
||||||
|
test('Imagery View operations @unstable', async ({ page }) => {
|
||||||
|
await performImageryViewOperationsAndAssert(page);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Example Imagery in Time Strip', () => {
|
||||||
|
let timeStripObject;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
timeStripObject = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Time Strip',
|
||||||
|
name: 'Time Strip'.concat(' ', uuid())
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Example Imagery',
|
||||||
|
name: 'Example Imagery'.concat(' ', uuid()),
|
||||||
|
parent: timeStripObject.uuid
|
||||||
|
});
|
||||||
|
// Navigate to timestrip
|
||||||
|
await page.goto(timeStripObject.url);
|
||||||
|
});
|
||||||
|
test('Clicking a thumbnail loads the image in large view', async ({ page, browserName }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5632'
|
||||||
|
});
|
||||||
|
await page.locator('.c-imagery-tsv-container').hover();
|
||||||
|
// get url of the hovered image
|
||||||
|
const hoveredImg = page.locator('.c-imagery-tsv div.c-imagery-tsv__image-wrapper:hover img');
|
||||||
|
const hoveredImgSrc = await hoveredImg.getAttribute('src');
|
||||||
|
expect(hoveredImgSrc).toBeTruthy();
|
||||||
|
await page.locator('.c-imagery-tsv-container').click();
|
||||||
|
// get image of view large container
|
||||||
|
const viewLargeImg = page.locator('img.c-imagery__main-image__image');
|
||||||
|
const viewLargeImgSrc = await viewLargeImg.getAttribute('src');
|
||||||
|
expect(viewLargeImgSrc).toBeTruthy();
|
||||||
|
expect(viewLargeImgSrc).toEqual(hoveredImgSrc);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Perform the common actions and assertions for the Imagery View.
|
||||||
|
* This function verifies the following in order:
|
||||||
|
* 1. Can zoom in/out using the zoom buttons
|
||||||
|
* 2. Can zoom in/out using the mouse wheel
|
||||||
|
* 3. Can pan the image using the pan hotkey + mouse drag
|
||||||
|
* 4. Clicking on the left arrow button pauses imagery and moves to the previous image
|
||||||
|
* 5. Imagery is updated as new images stream in, regardless of pause status
|
||||||
|
* 6. Old images are discarded when new images stream in
|
||||||
|
* 7. Image brightness/contrast can be adjusted by dragging the sliders
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function performImageryViewOperationsAndAssert(page) {
|
||||||
|
// Click previous image button
|
||||||
|
const previousImageButton = page.locator('.c-nav--prev');
|
||||||
|
await previousImageButton.click();
|
||||||
|
|
||||||
|
// Verify previous image
|
||||||
|
const selectedImage = page.locator('.selected');
|
||||||
|
await expect(selectedImage).toBeVisible();
|
||||||
|
|
||||||
|
// Use the zoom buttons to zoom in and out
|
||||||
|
await buttonZoomOnImageAndAssert(page);
|
||||||
|
|
||||||
|
// Use Mouse Wheel to zoom in to previous image
|
||||||
|
await mouseZoomOnImageAndAssert(page, 2);
|
||||||
|
|
||||||
|
// Use alt+drag to move around image once zoomed in
|
||||||
|
await panZoomAndAssertImageProperties(page);
|
||||||
|
|
||||||
|
// Use Mouse Wheel to zoom out of previous image
|
||||||
|
await mouseZoomOnImageAndAssert(page, -2);
|
||||||
|
|
||||||
|
// Click next image button
|
||||||
|
const nextImageButton = page.locator('.c-nav--next');
|
||||||
|
await nextImageButton.click();
|
||||||
|
|
||||||
|
// Click time conductor mode button
|
||||||
|
await page.locator('.c-mode-button').click();
|
||||||
|
|
||||||
|
// Select local clock mode
|
||||||
|
await page.locator('[data-testid=conductor-modeOption-realtime]').click();
|
||||||
|
|
||||||
|
// Zoom in on next image
|
||||||
|
await mouseZoomOnImageAndAssert(page, 2);
|
||||||
|
|
||||||
|
// Clicking on the left arrow should pause the imagery and go to previous image
|
||||||
|
await previousImageButton.click();
|
||||||
|
await expect(page.locator('.c-button.pause-play')).toHaveClass(/is-paused/);
|
||||||
|
await expect(selectedImage).toBeVisible();
|
||||||
|
|
||||||
|
// The imagery view should be updated when new images come in
|
||||||
|
const imageCount = await page.locator('.c-imagery__thumb').count();
|
||||||
|
await expect.poll(async () => {
|
||||||
|
const newImageCount = await page.locator('.c-imagery__thumb').count();
|
||||||
|
|
||||||
|
return newImageCount;
|
||||||
|
}, {
|
||||||
|
message: "verify that old images are discarded",
|
||||||
|
timeout: 7 * 1000
|
||||||
|
}).toBe(imageCount);
|
||||||
|
|
||||||
|
// Verify selected image is still displayed
|
||||||
|
await expect(selectedImage).toBeVisible();
|
||||||
|
|
||||||
|
// Unpause imagery
|
||||||
|
await page.locator('.pause-play').click();
|
||||||
|
|
||||||
|
//Get background-image url from background-image css prop
|
||||||
|
await assertBackgroundImageUrlFromBackgroundCss(page);
|
||||||
|
|
||||||
|
// Open the image filter menu
|
||||||
|
await page.locator('[role=toolbar] button[title="Brightness and contrast"]').click();
|
||||||
|
|
||||||
|
// Drag the brightness and contrast sliders around and assert filter values
|
||||||
|
await dragBrightnessSliderAndAssertFilterValues(page);
|
||||||
|
await dragContrastSliderAndAssertFilterValues(page);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Drag the brightness slider to max, min, and midpoint and assert the filter values
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function dragBrightnessSliderAndAssertFilterValues(page) {
|
||||||
|
const brightnessSlider = 'div.c-image-controls__slider-wrapper.icon-brightness > input';
|
||||||
|
const brightnessBoundingBox = await page.locator(brightnessSlider).boundingBox();
|
||||||
|
const brightnessMidX = brightnessBoundingBox.x + brightnessBoundingBox.width / 2;
|
||||||
|
const brightnessMidY = brightnessBoundingBox.y + brightnessBoundingBox.height / 2;
|
||||||
|
|
||||||
|
await page.locator(brightnessSlider).hover({trial: true});
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(brightnessBoundingBox.x + brightnessBoundingBox.width, brightnessMidY);
|
||||||
|
await assertBackgroundImageBrightness(page, '500');
|
||||||
|
await page.mouse.move(brightnessBoundingBox.x, brightnessMidY);
|
||||||
|
await assertBackgroundImageBrightness(page, '0');
|
||||||
|
await page.mouse.move(brightnessMidX, brightnessMidY);
|
||||||
|
await assertBackgroundImageBrightness(page, '250');
|
||||||
|
await page.mouse.up();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Drag the contrast slider to max, min, and midpoint and assert the filter values
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function dragContrastSliderAndAssertFilterValues(page) {
|
||||||
|
const contrastSlider = 'div.c-image-controls__slider-wrapper.icon-contrast > input';
|
||||||
|
const contrastBoundingBox = await page.locator(contrastSlider).boundingBox();
|
||||||
|
const contrastMidX = contrastBoundingBox.x + contrastBoundingBox.width / 2;
|
||||||
|
const contrastMidY = contrastBoundingBox.y + contrastBoundingBox.height / 2;
|
||||||
|
|
||||||
|
await page.locator(contrastSlider).hover({trial: true});
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(contrastBoundingBox.x + contrastBoundingBox.width, contrastMidY);
|
||||||
|
await assertBackgroundImageContrast(page, '500');
|
||||||
|
await page.mouse.move(contrastBoundingBox.x, contrastMidY);
|
||||||
|
await assertBackgroundImageContrast(page, '0');
|
||||||
|
await page.mouse.move(contrastMidX, contrastMidY);
|
||||||
|
await assertBackgroundImageContrast(page, '250');
|
||||||
|
await page.mouse.up();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the filter:brightness value of the current background-image and
|
||||||
|
* asserts against an expected value
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {String} expected The expected brightness value
|
||||||
|
*/
|
||||||
|
async function assertBackgroundImageBrightness(page, expected) {
|
||||||
|
const backgroundImage = page.locator('.c-imagery__main-image__background-image');
|
||||||
|
|
||||||
|
// Get the brightness filter value (i.e: filter: brightness(500%) => "500")
|
||||||
|
const actual = await backgroundImage.evaluate((el) => {
|
||||||
|
return el.style.filter.match(/brightness\((\d{1,3})%\)/)[1];
|
||||||
|
});
|
||||||
|
expect(actual).toBe(expected);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function assertBackgroundImageUrlFromBackgroundCss(page) {
|
||||||
|
const backgroundImage = page.locator('.c-imagery__main-image__background-image');
|
||||||
|
let backgroundImageUrl = await backgroundImage.evaluate((el) => {
|
||||||
|
return window.getComputedStyle(el).getPropertyValue('background-image').match(/url\(([^)]+)\)/)[1];
|
||||||
|
});
|
||||||
|
let backgroundImageUrl1 = backgroundImageUrl.slice(1, -1); //forgive me, padre
|
||||||
|
console.log('backgroundImageUrl1 ' + backgroundImageUrl1);
|
||||||
|
|
||||||
|
let backgroundImageUrl2;
|
||||||
|
await expect.poll(async () => {
|
||||||
|
// Verify next image has updated
|
||||||
|
let backgroundImageUrlNext = await backgroundImage.evaluate((el) => {
|
||||||
|
return window.getComputedStyle(el).getPropertyValue('background-image').match(/url\(([^)]+)\)/)[1];
|
||||||
|
});
|
||||||
|
backgroundImageUrl2 = backgroundImageUrlNext.slice(1, -1); //forgive me, padre
|
||||||
|
|
||||||
|
return backgroundImageUrl2;
|
||||||
|
}, {
|
||||||
|
message: "verify next image has updated",
|
||||||
|
timeout: 7 * 1000
|
||||||
|
}).not.toBe(backgroundImageUrl1);
|
||||||
|
console.log('backgroundImageUrl2 ' + backgroundImageUrl2);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function panZoomAndAssertImageProperties(page) {
|
||||||
|
const imageryHintsText = await page.locator('.c-imagery__hints').innerText();
|
||||||
|
expect(expectedAltText).toEqual(imageryHintsText);
|
||||||
|
const zoomedBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
const imageCenterX = zoomedBoundingBox.x + zoomedBoundingBox.width / 2;
|
||||||
|
const imageCenterY = zoomedBoundingBox.y + zoomedBoundingBox.height / 2;
|
||||||
|
|
||||||
|
// Pan right
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX - 200, imageCenterY, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterRightPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(zoomedBoundingBox.x).toBeGreaterThan(afterRightPanBoundingBox.x);
|
||||||
|
|
||||||
|
// Pan left
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterLeftPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(afterRightPanBoundingBox.x).toBeLessThan(afterLeftPanBoundingBox.x);
|
||||||
|
|
||||||
|
// Pan up
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY);
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY + 200, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterUpPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(afterUpPanBoundingBox.y).toBeGreaterThanOrEqual(afterLeftPanBoundingBox.y);
|
||||||
|
|
||||||
|
// Pan down
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
||||||
|
await page.mouse.down();
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY - 200, 10);
|
||||||
|
await page.mouse.up();
|
||||||
|
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
||||||
|
const afterDownPanBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(afterDownPanBoundingBox.y).toBeLessThanOrEqual(afterUpPanBoundingBox.y);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use the mouse wheel to zoom in or out of an image and assert that the image
|
||||||
|
* has successfully zoomed in or out.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {number} [factor = 2] The zoom factor. Positive for zoom in, negative for zoom out.
|
||||||
|
*/
|
||||||
|
async function mouseZoomOnImageAndAssert(page, factor = 2) {
|
||||||
|
// Zoom in
|
||||||
|
const originalImageDimensions = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
await page.locator(backgroundImageSelector).hover({trial: true});
|
||||||
|
const deltaYStep = 100; // equivalent to 1x zoom
|
||||||
|
await page.mouse.wheel(0, deltaYStep * factor);
|
||||||
|
const zoomedBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
const imageCenterX = zoomedBoundingBox.x + zoomedBoundingBox.width / 2;
|
||||||
|
const imageCenterY = zoomedBoundingBox.y + zoomedBoundingBox.height / 2;
|
||||||
|
|
||||||
|
// center the mouse pointer
|
||||||
|
await page.mouse.move(imageCenterX, imageCenterY);
|
||||||
|
|
||||||
|
// Wait for zoom animation to finish
|
||||||
|
await page.locator(backgroundImageSelector).hover({trial: true});
|
||||||
|
const imageMouseZoomed = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
|
||||||
|
if (factor > 0) {
|
||||||
|
expect(imageMouseZoomed.height).toBeGreaterThan(originalImageDimensions.height);
|
||||||
|
expect(imageMouseZoomed.width).toBeGreaterThan(originalImageDimensions.width);
|
||||||
|
} else {
|
||||||
|
expect(imageMouseZoomed.height).toBeLessThan(originalImageDimensions.height);
|
||||||
|
expect(imageMouseZoomed.width).toBeLessThan(originalImageDimensions.width);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Zoom in and out of the image using the buttons, and assert that the image has
|
||||||
|
* been successfully zoomed in or out.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function buttonZoomOnImageAndAssert(page) {
|
||||||
|
// Get initial image dimensions
|
||||||
|
const initialBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
|
||||||
|
// Zoom in twice via button
|
||||||
|
await zoomIntoImageryByButton(page);
|
||||||
|
await zoomIntoImageryByButton(page);
|
||||||
|
|
||||||
|
// Get and assert zoomed in image dimensions
|
||||||
|
const zoomedInBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(zoomedInBoundingBox.height).toBeGreaterThan(initialBoundingBox.height);
|
||||||
|
expect(zoomedInBoundingBox.width).toBeGreaterThan(initialBoundingBox.width);
|
||||||
|
|
||||||
|
// Zoom out once via button
|
||||||
|
await zoomOutOfImageryByButton(page);
|
||||||
|
|
||||||
|
// Get and assert zoomed out image dimensions
|
||||||
|
const zoomedOutBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(zoomedOutBoundingBox.height).toBeLessThan(zoomedInBoundingBox.height);
|
||||||
|
expect(zoomedOutBoundingBox.width).toBeLessThan(zoomedInBoundingBox.width);
|
||||||
|
|
||||||
|
// Zoom out again via button, assert against the initial image dimensions
|
||||||
|
await zoomOutOfImageryByButton(page);
|
||||||
|
const finalBoundingBox = await page.locator(backgroundImageSelector).boundingBox();
|
||||||
|
expect(finalBoundingBox).toEqual(initialBoundingBox);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets the filter:contrast value of the current background-image and
|
||||||
|
* asserts against an expected value
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {String} expected The expected contrast value
|
||||||
|
*/
|
||||||
|
async function assertBackgroundImageContrast(page, expected) {
|
||||||
|
const backgroundImage = page.locator('.c-imagery__main-image__background-image');
|
||||||
|
|
||||||
|
// Get the contrast filter value (i.e: filter: contrast(500%) => "500")
|
||||||
|
const actual = await backgroundImage.evaluate((el) => {
|
||||||
|
return el.style.filter.match(/contrast\((\d{1,3})%\)/)[1];
|
||||||
|
});
|
||||||
|
expect(actual).toBe(expected);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use the '+' button to zoom in. Hovers first if the toolbar is not visible
|
||||||
|
* and waits for the zoom animation to finish afterwards.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function zoomIntoImageryByButton(page) {
|
||||||
|
// FIXME: There should only be one set of imagery buttons, but there are two?
|
||||||
|
const zoomInBtn = page.locator("[role='toolbar'][aria-label='Image controls'] .t-btn-zoom-in").nth(0);
|
||||||
|
const backgroundImage = page.locator(backgroundImageSelector);
|
||||||
|
if (!(await zoomInBtn.isVisible())) {
|
||||||
|
await backgroundImage.hover({trial: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
await zoomInBtn.click();
|
||||||
|
await waitForAnimations(backgroundImage);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use the '-' button to zoom out. Hovers first if the toolbar is not visible
|
||||||
|
* and waits for the zoom animation to finish afterwards.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function zoomOutOfImageryByButton(page) {
|
||||||
|
// FIXME: There should only be one set of imagery buttons, but there are two?
|
||||||
|
const zoomOutBtn = page.locator("[role='toolbar'][aria-label='Image controls'] .t-btn-zoom-out").nth(0);
|
||||||
|
const backgroundImage = page.locator(backgroundImageSelector);
|
||||||
|
if (!(await zoomOutBtn.isVisible())) {
|
||||||
|
await backgroundImage.hover({trial: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
await zoomOutBtn.click();
|
||||||
|
await waitForAnimations(backgroundImage);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Use the reset button to reset image pan and zoom. Hovers first if the toolbar is not visible
|
||||||
|
* and waits for the zoom animation to finish afterwards.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function resetImageryPanAndZoom(page) {
|
||||||
|
// FIXME: There should only be one set of imagery buttons, but there are two?
|
||||||
|
const panZoomResetBtn = page.locator("[role='toolbar'][aria-label='Image controls'] .t-btn-zoom-reset").nth(0);
|
||||||
|
const backgroundImage = page.locator(backgroundImageSelector);
|
||||||
|
if (!(await panZoomResetBtn.isVisible())) {
|
||||||
|
await backgroundImage.hover({trial: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
await panZoomResetBtn.click();
|
||||||
|
await waitForAnimations(backgroundImage);
|
||||||
|
}
|
@ -24,7 +24,9 @@
|
|||||||
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.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
// FIXME: Remove this eslint exception once tests are implemented
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
|
||||||
test.describe('ExportAsJSON', () => {
|
test.describe('ExportAsJSON', () => {
|
||||||
test.fixme('Create a basic object and verify that it can be exported as JSON from Tree', async ({ page }) => {
|
test.fixme('Create a basic object and verify that it can be exported as JSON from Tree', async ({ page }) => {
|
@ -24,7 +24,9 @@
|
|||||||
This test suite is dedicated to tests which verify the basic operations surrounding importAsJSON.
|
This test suite is dedicated to tests which verify the basic operations surrounding importAsJSON.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
// FIXME: Remove this eslint exception once tests are implemented
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
|
||||||
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 }) => {
|
120
e2e/tests/functional/plugins/lad/lad.e2e.spec.js
Normal file
120
e2e/tests/functional/plugins/lad/lad.e2e.spec.js
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults, setStartOffset, setFixedTimeMode, setRealTimeMode } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Testing LAD table @unstable', () => {
|
||||||
|
let sineWaveObject;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// Create Sine Wave Generator
|
||||||
|
sineWaveObject = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
name: "Test Sine Wave Generator"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('telemetry value exactly matches latest telemetry value received in real time', async ({ page }) => {
|
||||||
|
// Create LAD table
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'LAD Table',
|
||||||
|
name: "Test LAD Table"
|
||||||
|
});
|
||||||
|
// Edit LAD table
|
||||||
|
await page.locator('[title="Edit"]').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('text=Test Sine Wave Generator', '.c-lad-table-wrapper');
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
// Subscribe to the Sine Wave Generator data
|
||||||
|
// On getting data, check if the value found in the LAD table is the most recent value
|
||||||
|
// from the Sine Wave Generator
|
||||||
|
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
||||||
|
const subscribeTelemValue = await getTelemValuePromise;
|
||||||
|
const ladTableValuePromise = await page.waitForSelector(`text="${subscribeTelemValue}"`);
|
||||||
|
const ladTableValue = await ladTableValuePromise.textContent();
|
||||||
|
|
||||||
|
expect(ladTableValue).toBe(subscribeTelemValue);
|
||||||
|
});
|
||||||
|
test('telemetry value exactly matches latest telemetry value received in fixed time', async ({ page }) => {
|
||||||
|
// Create LAD table
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'LAD Table',
|
||||||
|
name: "Test LAD Table"
|
||||||
|
});
|
||||||
|
// Edit LAD table
|
||||||
|
await page.locator('[title="Edit"]').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('text=Test Sine Wave Generator', '.c-lad-table-wrapper');
|
||||||
|
await page.locator('button[title="Save"]').click();
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
// Subscribe to the Sine Wave Generator data
|
||||||
|
const getTelemValuePromise = await subscribeToTelemetry(page, sineWaveObject.uuid);
|
||||||
|
// Set an offset of 1 minute and then change the time mode to fixed to set a 1 minute historical window
|
||||||
|
await setStartOffset(page, { mins: '1' });
|
||||||
|
await setFixedTimeMode(page);
|
||||||
|
|
||||||
|
// On getting data, check if the value found in the LAD table is the most recent value
|
||||||
|
// from the Sine Wave Generator
|
||||||
|
const subscribeTelemValue = await getTelemValuePromise;
|
||||||
|
const ladTableValuePromise = await page.waitForSelector(`text="${subscribeTelemValue}"`);
|
||||||
|
const ladTableValue = await ladTableValuePromise.textContent();
|
||||||
|
|
||||||
|
expect(ladTableValue).toBe(subscribeTelemValue);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Util for subscribing to a telemetry object by object identifier
|
||||||
|
* Limitations: Currently only works to return telemetry once to the node scope
|
||||||
|
* To Do: See if there's a way to await this multiple times to allow for multiple
|
||||||
|
* values to be returned over time
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} objectIdentifier identifier for object
|
||||||
|
* @returns {Promise<string>} the formatted sin telemetry value
|
||||||
|
*/
|
||||||
|
async function subscribeToTelemetry(page, objectIdentifier) {
|
||||||
|
const getTelemValuePromise = new Promise(resolve => page.exposeFunction('getTelemValue', resolve));
|
||||||
|
|
||||||
|
await page.evaluate(async (telemetryIdentifier) => {
|
||||||
|
const telemetryObject = await window.openmct.objects.get(telemetryIdentifier);
|
||||||
|
const metadata = window.openmct.telemetry.getMetadata(telemetryObject);
|
||||||
|
const formats = await window.openmct.telemetry.getFormatMap(metadata);
|
||||||
|
window.openmct.telemetry.subscribe(telemetryObject, (obj) => {
|
||||||
|
const sinVal = obj.sin;
|
||||||
|
const formattedSinVal = formats.sin.format(sinVal);
|
||||||
|
window.getTelemValue(formattedSinVal);
|
||||||
|
});
|
||||||
|
}, objectIdentifier);
|
||||||
|
|
||||||
|
return getTelemValuePromise;
|
||||||
|
}
|
335
e2e/tests/functional/plugins/notebook/notebook.e2e.spec.js
Normal file
335
e2e/tests/functional/plugins/notebook/notebook.e2e.spec.js
Normal file
@ -0,0 +1,335 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding Notebooks.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// FIXME: Remove this eslint exception once tests are implemented
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
const { expandTreePaneItemByName, createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
const nbUtils = require('../../../../helper/notebookUtils');
|
||||||
|
|
||||||
|
test.describe('Notebook CRUD Operations', () => {
|
||||||
|
test.fixme('Can create a Notebook Object', async ({ page }) => {
|
||||||
|
//Create domain object
|
||||||
|
//Newly created notebook should have one Section and one page, 'Unnamed Section'/'Unnamed Page'
|
||||||
|
});
|
||||||
|
test.fixme('Can update a Notebook Object', async ({ page }) => {});
|
||||||
|
test.fixme('Can view a perviously created Notebook Object', async ({ page }) => {});
|
||||||
|
test.fixme('Can Delete a Notebook Object', async ({ page }) => {
|
||||||
|
// Other than non-persistible objects
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Default Notebook', () => {
|
||||||
|
// General Default Notebook statements
|
||||||
|
// ## Useful commands:
|
||||||
|
// 1. - To check default notebook:
|
||||||
|
// `JSON.parse(localStorage.getItem('notebook-storage'));`
|
||||||
|
// 1. - Clear default notebook:
|
||||||
|
// `localStorage.setItem('notebook-storage', null);`
|
||||||
|
test.fixme('A newly created Notebook is automatically set as the default notebook if no other notebooks exist', async ({ page }) => {
|
||||||
|
//Create new notebook
|
||||||
|
//Verify Default Notebook Characteristics
|
||||||
|
});
|
||||||
|
test.fixme('A newly created Notebook is automatically set as the default notebook if at least one other notebook exists', async ({ page }) => {
|
||||||
|
//Create new notebook A
|
||||||
|
//Create second notebook B
|
||||||
|
//Verify Non-Default Notebook A Characteristics
|
||||||
|
//Verify Default Notebook B Characteristics
|
||||||
|
});
|
||||||
|
test.fixme('If a default notebook is deleted, the second most recent notebook becomes the default', async ({ page }) => {
|
||||||
|
//Create new notebook A
|
||||||
|
//Create second notebook B
|
||||||
|
//Delete Notebook B
|
||||||
|
//Verify Default Notebook A Characteristics
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Notebook section tests', () => {
|
||||||
|
//The following test cases are associated with Notebook Sections
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
//Navigate to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create Notebook
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Notebook',
|
||||||
|
name: "Test Notebook"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test('Default and new sections are automatically named Unnamed Section with Unnamed Page', async ({ page }) => {
|
||||||
|
// Check that the default section and page are created and the name matches the defaults
|
||||||
|
const defaultSectionName = await page.locator('.c-notebook__sections .c-list__item__name').textContent();
|
||||||
|
expect(defaultSectionName).toBe('Unnamed Section');
|
||||||
|
const defaultPageName = await page.locator('.c-notebook__pages .c-list__item__name').textContent();
|
||||||
|
expect(defaultPageName).toBe('Unnamed Page');
|
||||||
|
|
||||||
|
// Expand sidebar and add a section
|
||||||
|
await page.locator('.c-notebook__toggle-nav-button').click();
|
||||||
|
await page.locator('.js-sidebar-sections .c-icon-button.icon-plus').click();
|
||||||
|
|
||||||
|
// Check that new section and page within the new section match the defaults
|
||||||
|
const newSectionName = await page.locator('.c-notebook__sections .c-list__item__name').nth(1).textContent();
|
||||||
|
expect(newSectionName).toBe('Unnamed Section');
|
||||||
|
const newPageName = await page.locator('.c-notebook__pages .c-list__item__name').textContent();
|
||||||
|
expect(newPageName).toBe('Unnamed Page');
|
||||||
|
});
|
||||||
|
test.fixme('Section selection operations and associated behavior', async ({ page }) => {
|
||||||
|
//Create new notebook A
|
||||||
|
//Add Sections until 6 total with no default section/page
|
||||||
|
//Select 3rd section
|
||||||
|
//Delete 4th section
|
||||||
|
//3rd section is still selected
|
||||||
|
//Delete 3rd section
|
||||||
|
//1st section is selected
|
||||||
|
//Set 3rd section as default
|
||||||
|
//Delete 2nd section
|
||||||
|
//3rd section is still default
|
||||||
|
//Delete 3rd section
|
||||||
|
//1st is selected and there is no default notebook
|
||||||
|
});
|
||||||
|
test.fixme('Section rename operations', async ({ page }) => {
|
||||||
|
// Create a new notebook
|
||||||
|
// Add a section
|
||||||
|
// Rename the section but do not confirm
|
||||||
|
// Keyboard press 'Escape'
|
||||||
|
// Verify that the section name reverts to the default name
|
||||||
|
// Rename the section but do not confirm
|
||||||
|
// Keyboard press 'Enter'
|
||||||
|
// Verify that the section name is updated
|
||||||
|
// Rename the section to "" (empty string)
|
||||||
|
// Keyboard press 'Enter' to confirm
|
||||||
|
// Verify that the section name reverts to the default name
|
||||||
|
// Rename the section to something long that overflows the text box
|
||||||
|
// Verify that the section name is not truncated while input is active
|
||||||
|
// Confirm the section name edit
|
||||||
|
// Verify that the section name is truncated now that input is not active
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Notebook page tests', () => {
|
||||||
|
//The following test cases are associated with Notebook Pages
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
//Navigate to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create Notebook
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Notebook',
|
||||||
|
name: "Test Notebook"
|
||||||
|
});
|
||||||
|
});
|
||||||
|
//Test will need to be implemented after a refactor in #5713
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip('Delete page popup is removed properly on clicking dropdown again', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5713'
|
||||||
|
});
|
||||||
|
// Expand sidebar and add a second page
|
||||||
|
await page.locator('.c-notebook__toggle-nav-button').click();
|
||||||
|
await page.locator('text=Page Add >> button').click();
|
||||||
|
|
||||||
|
// Click on the 2nd page dropdown button and expect the Delete Page option to appear
|
||||||
|
await page.locator('button[title="Open context menu"]').nth(2).click();
|
||||||
|
await expect(page.locator('text=Delete Page')).toBeEnabled();
|
||||||
|
// Clicking on the same page a second time causes the same Delete Page option to recreate
|
||||||
|
await page.locator('button[title="Open context menu"]').nth(2).click();
|
||||||
|
await expect(page.locator('text=Delete Page')).toBeEnabled();
|
||||||
|
// Clicking on the first page causes the first delete button to detach and recreate on the first page
|
||||||
|
await page.locator('button[title="Open context menu"]').nth(1).click();
|
||||||
|
const numOfDeletePagePopups = await page.locator('li[title="Delete Page"]').count();
|
||||||
|
expect(numOfDeletePagePopups).toBe(1);
|
||||||
|
});
|
||||||
|
test.fixme('Page selection operations and associated behavior', async ({ page }) => {
|
||||||
|
//Create new notebook A
|
||||||
|
//Delete existing Page
|
||||||
|
//New 'Unnamed Page' automatically created
|
||||||
|
//Create 6 total Pages without a default page
|
||||||
|
//Select 3rd
|
||||||
|
//Delete 3rd
|
||||||
|
//First is now selected
|
||||||
|
//Set 3rd as default
|
||||||
|
//Select 2nd page
|
||||||
|
//Delete 2nd page
|
||||||
|
//3rd (default) is now selected
|
||||||
|
//Set 3rd as default page
|
||||||
|
//Select 3rd (default) page
|
||||||
|
//Delete 3rd page
|
||||||
|
//First is now selected and there is no default notebook
|
||||||
|
});
|
||||||
|
test.fixme('Page rename operations', async ({ page }) => {
|
||||||
|
// Create a new notebook
|
||||||
|
// Add a page
|
||||||
|
// Rename the page but do not confirm
|
||||||
|
// Keyboard press 'Escape'
|
||||||
|
// Verify that the page name reverts to the default name
|
||||||
|
// Rename the page but do not confirm
|
||||||
|
// Keyboard press 'Enter'
|
||||||
|
// Verify that the page name is updated
|
||||||
|
// Rename the page to "" (empty string)
|
||||||
|
// Keyboard press 'Enter' to confirm
|
||||||
|
// Verify that the page name reverts to the default name
|
||||||
|
// Rename the page to something long that overflows the text box
|
||||||
|
// Verify that the page name is not truncated while input is active
|
||||||
|
// Confirm the page name edit
|
||||||
|
// Verify that the page name is truncated now that input is not active
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Notebook search tests', () => {
|
||||||
|
test.fixme('Can search for a single result', async ({ page }) => {});
|
||||||
|
test.fixme('Can search for many results', async ({ page }) => {});
|
||||||
|
test.fixme('Can search for new and recently modified entries', async ({ page }) => {});
|
||||||
|
test.fixme('Can search for section text', async ({ page }) => {});
|
||||||
|
test.fixme('Can search for page text', async ({ page }) => {});
|
||||||
|
test.fixme('Can search for entry text', async ({ page }) => {});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Notebook entry tests', () => {
|
||||||
|
test.fixme('When a new entry is created, it should be focused', async ({ page }) => {});
|
||||||
|
test('When an object is dropped into a notebook, a new entry is created and it should be focused @unstable', async ({ page }) => {
|
||||||
|
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create Notebook
|
||||||
|
const notebook = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Notebook',
|
||||||
|
name: "Embed Test Notebook"
|
||||||
|
});
|
||||||
|
// Create Overlay Plot
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Overlay Plot',
|
||||||
|
name: "Dropped Overlay Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page, 'My Items');
|
||||||
|
|
||||||
|
await page.goto(notebook.url);
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', '.c-notebook__drag-area');
|
||||||
|
|
||||||
|
const embed = page.locator('.c-ne__embed__link');
|
||||||
|
const embedName = await embed.textContent();
|
||||||
|
|
||||||
|
await expect(embed).toHaveClass(/icon-plot-overlay/);
|
||||||
|
expect(embedName).toBe('Dropped Overlay Plot');
|
||||||
|
});
|
||||||
|
test('When an object is dropped into a notebooks existing entry, it should be focused @unstable', async ({ page }) => {
|
||||||
|
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create Notebook
|
||||||
|
const notebook = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Notebook',
|
||||||
|
name: "Embed Test Notebook"
|
||||||
|
});
|
||||||
|
// Create Overlay Plot
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Overlay Plot',
|
||||||
|
name: "Dropped Overlay Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page, 'My Items');
|
||||||
|
|
||||||
|
await page.goto(notebook.url);
|
||||||
|
|
||||||
|
await nbUtils.enterTextEntry(page, 'Entry to drop into');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', 'text=Entry to drop into');
|
||||||
|
|
||||||
|
const existingEntry = page.locator('.c-ne__content', { has: page.locator('text="Entry to drop into"') });
|
||||||
|
const embed = existingEntry.locator('.c-ne__embed__link');
|
||||||
|
const embedName = await embed.textContent();
|
||||||
|
|
||||||
|
await expect(embed).toHaveClass(/icon-plot-overlay/);
|
||||||
|
expect(embedName).toBe('Dropped Overlay Plot');
|
||||||
|
});
|
||||||
|
test.fixme('new entries persist through navigation events without save', async ({ page }) => {});
|
||||||
|
test.fixme('previous and new entries can be deleted', async ({ page }) => {});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Snapshot Menu tests', () => {
|
||||||
|
test.fixme('When no default notebook is selected, Snapshot Menu dropdown should only have a single option', async ({ page }) => {
|
||||||
|
// There should be no default notebook
|
||||||
|
// Clear default notebook if exists using `localStorage.setItem('notebook-storage', null);`
|
||||||
|
// refresh page
|
||||||
|
// Click on 'Notebook Snaphot Menu'
|
||||||
|
// 'save to Notebook Snapshots' should be only option there
|
||||||
|
});
|
||||||
|
test.fixme('When default notebook is updated selected, Snapshot Menu dropdown should list it as the newest option', async ({ page }) => {
|
||||||
|
// Create 2a notebooks
|
||||||
|
// Set Notebook A as Default
|
||||||
|
// Open Snapshot Menu and note that Notebook A is listed
|
||||||
|
// Close Snapshot Menu
|
||||||
|
// Set Default Notebook to Notebook B
|
||||||
|
// Open Snapshot Notebook and note that Notebook B is listed
|
||||||
|
// Select Default Notebook Option and verify that Snapshot is added to Notebook B
|
||||||
|
});
|
||||||
|
test.fixme('Can add Snapshots via Snapshot Menu and details are correct', async ({ page }) => {
|
||||||
|
//Note this should be a visual test, too
|
||||||
|
// Create Telemetry object
|
||||||
|
// Create A notebook with many pages and sections.
|
||||||
|
// Set page and section defaults to be between first and last of many. i.e. 3 of 5
|
||||||
|
// Navigate to Telemetry object
|
||||||
|
// Select Default Notebook Option and verify that Snapshot is added to Notebook A
|
||||||
|
// Verify Snapshot Details appear correctly
|
||||||
|
});
|
||||||
|
test.fixme('Snapshots adjust time conductor', async ({ page }) => {
|
||||||
|
// Create Telemetry object
|
||||||
|
// Set Telemetry object's timeconductor to Fixed time with Start and Endtimes are recorded
|
||||||
|
// Embed Telemetry object into notebook
|
||||||
|
// Set Time Conductor to Local clock
|
||||||
|
// Click into embedded telemetry object and verify object appears with same fixed time from record
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Snapshot Container tests', () => {
|
||||||
|
test.fixme('5 Snapshots can be added to a container', async ({ page }) => {});
|
||||||
|
test.fixme('5 Snapshots can be added to a container and Deleted with Delete All action', async ({ page }) => {});
|
||||||
|
test.fixme('A snapshot can be Deleted from Container', async ({ page }) => {});
|
||||||
|
test.fixme('A snapshot can be Previewed from Container', async ({ page }) => {});
|
||||||
|
test.fixme('A snapshot Container can be open and closed', async ({ page }) => {});
|
||||||
|
test.fixme('Can add object to Snapshot container and pull into notebook and create a new entry', async ({ page }) => {
|
||||||
|
//Create Notebook
|
||||||
|
//Create Telemetry Object
|
||||||
|
//From Telemetry Object, use 'save to Notebook Snapshots'
|
||||||
|
//Snapshots indicator should blink, click on it to view snapshots
|
||||||
|
//Navigate to Notebook
|
||||||
|
//Drag and Drop onto droppable area for new entry
|
||||||
|
//New Entry created with given snapshot added
|
||||||
|
//Snapshot removed from container?
|
||||||
|
});
|
||||||
|
test.fixme('Can add object to Snapshot container and pull into notebook and existing entry', async ({ page }) => {
|
||||||
|
//Create Notebook
|
||||||
|
//Create Telemetry Object
|
||||||
|
//From Telemetry Object, use 'save to Notebook Snapshots'
|
||||||
|
//Snapshots indicator should blink, click on it to view snapshots
|
||||||
|
//Navigate to Notebook
|
||||||
|
//Drag and Drop into exiting entry
|
||||||
|
//Existing Entry updated with given snapshot
|
||||||
|
//Snapshot removed from container?
|
||||||
|
});
|
||||||
|
test.fixme('Verify Embedded options for PNG, JPG, and Annotate work correctly', async ({ page }) => {
|
||||||
|
//Add snapshot to container
|
||||||
|
//Verify PNG, JPG, and Annotate buttons work correctly
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,193 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { openObjectTreeContextMenu, createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
const path = require('path');
|
||||||
|
const nbUtils = require('../../../../helper/notebookUtils');
|
||||||
|
|
||||||
|
const TEST_TEXT = 'Testing text for entries.';
|
||||||
|
const TEST_TEXT_NAME = 'Test Page';
|
||||||
|
const CUSTOM_NAME = 'CUSTOM_NAME';
|
||||||
|
|
||||||
|
test.describe('Restricted Notebook', () => {
|
||||||
|
let notebook;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
notebook = await startAndAddRestrictedNotebookObject(page);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can be renamed @addInit', async ({ page }) => {
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText(`Unnamed ${CUSTOM_NAME}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can be deleted if there are no locked pages @addInit', async ({ page, openmctConfig }) => {
|
||||||
|
await openObjectTreeContextMenu(page, notebook.url);
|
||||||
|
|
||||||
|
const menuOptions = page.locator('.c-menu ul');
|
||||||
|
await expect.soft(menuOptions).toContainText('Remove');
|
||||||
|
|
||||||
|
const restrictedNotebookTreeObject = page.locator(`a:has-text("Unnamed ${CUSTOM_NAME}")`);
|
||||||
|
|
||||||
|
// notebook tree object exists
|
||||||
|
expect.soft(await restrictedNotebookTreeObject.count()).toEqual(1);
|
||||||
|
|
||||||
|
// Click Remove Text
|
||||||
|
await page.locator('text=Remove').click();
|
||||||
|
|
||||||
|
// Click 'OK' on confirmation window and wait for save banner to appear
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=OK').click(),
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// has been deleted
|
||||||
|
expect(await restrictedNotebookTreeObject.count()).toEqual(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can be locked if at least one page has one entry @addInit', async ({ page }) => {
|
||||||
|
|
||||||
|
await nbUtils.enterTextEntry(page, TEST_TEXT);
|
||||||
|
|
||||||
|
const commitButton = page.locator('button:has-text("Commit Entries")');
|
||||||
|
expect(await commitButton.count()).toEqual(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Restricted Notebook with at least one entry and with the page locked @addInit', () => {
|
||||||
|
let notebook;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
notebook = await startAndAddRestrictedNotebookObject(page);
|
||||||
|
await nbUtils.enterTextEntry(page, TEST_TEXT);
|
||||||
|
await lockPage(page);
|
||||||
|
|
||||||
|
// open sidebar
|
||||||
|
await page.locator('button.c-notebook__toggle-nav-button').click();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Locked page should now be in a locked state @addInit @unstable', async ({ page }, testInfo) => {
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip(testInfo.project === 'chrome-beta', "Test is unreliable on chrome-beta");
|
||||||
|
// main lock message on page
|
||||||
|
const lockMessage = page.locator('text=This page has been committed and cannot be modified or removed');
|
||||||
|
expect.soft(await lockMessage.count()).toEqual(1);
|
||||||
|
|
||||||
|
// lock icon on page in sidebar
|
||||||
|
const pageLockIcon = page.locator('ul.c-notebook__pages li div.icon-lock');
|
||||||
|
expect.soft(await pageLockIcon.count()).toEqual(1);
|
||||||
|
|
||||||
|
// no way to remove a restricted notebook with a locked page
|
||||||
|
await openObjectTreeContextMenu(page, notebook.url);
|
||||||
|
const menuOptions = page.locator('.c-menu ul');
|
||||||
|
|
||||||
|
await expect(menuOptions).not.toContainText('Remove');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can still: add page, rename, add entry, delete unlocked pages @addInit', async ({ page }) => {
|
||||||
|
// Click text=Page Add >> button
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Page Add >> button').click()
|
||||||
|
]);
|
||||||
|
// Click text=Unnamed Page >> nth=1
|
||||||
|
await page.locator('text=Unnamed Page').nth(1).click();
|
||||||
|
// Press a with modifiers
|
||||||
|
await page.locator('text=Unnamed Page').nth(1).fill(TEST_TEXT_NAME);
|
||||||
|
|
||||||
|
// expect to be able to rename unlocked pages
|
||||||
|
const newPageElement = page.locator(`text=${TEST_TEXT_NAME}`);
|
||||||
|
const newPageCount = await newPageElement.count();
|
||||||
|
await newPageElement.press('Enter'); // exit contenteditable state
|
||||||
|
expect.soft(newPageCount).toEqual(1);
|
||||||
|
|
||||||
|
// enter test text
|
||||||
|
await nbUtils.enterTextEntry(page, TEST_TEXT);
|
||||||
|
|
||||||
|
// expect new page to be lockable
|
||||||
|
const commitButton = page.locator('BUTTON:HAS-TEXT("COMMIT ENTRIES")');
|
||||||
|
expect.soft(await commitButton.count()).toEqual(1);
|
||||||
|
|
||||||
|
// Click text=Unnamed PageTest Page >> button
|
||||||
|
await page.locator('text=Unnamed PageTest Page >> button').click();
|
||||||
|
// Click text=Delete Page
|
||||||
|
await page.locator('text=Delete Page').click();
|
||||||
|
// Click text=Ok
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Ok').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
// deleted page, should no longer exist
|
||||||
|
const deletedPageElement = page.locator(`text=${TEST_TEXT_NAME}`);
|
||||||
|
expect(await deletedPageElement.count()).toEqual(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Restricted Notebook with a page locked and with an embed @addInit', () => {
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
await startAndAddRestrictedNotebookObject(page);
|
||||||
|
await nbUtils.dragAndDropEmbed(page, myItemsFolderName);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Allows embeds to be deleted if page unlocked @addInit', async ({ page }) => {
|
||||||
|
// Click .c-ne__embed__name .c-popup-menu-button
|
||||||
|
await page.locator('.c-ne__embed__name .c-popup-menu-button').click(); // embed popup menu
|
||||||
|
|
||||||
|
const embedMenu = page.locator('body >> .c-menu');
|
||||||
|
await expect(embedMenu).toContainText('Remove This Embed');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Disallows embeds to be deleted if page locked @addInit', async ({ page }) => {
|
||||||
|
await lockPage(page);
|
||||||
|
// Click .c-ne__embed__name .c-popup-menu-button
|
||||||
|
await page.locator('.c-ne__embed__name .c-popup-menu-button').click(); // embed popup menu
|
||||||
|
|
||||||
|
const embedMenu = page.locator('body >> .c-menu');
|
||||||
|
await expect(embedMenu).not.toContainText('Remove This Embed');
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function startAndAddRestrictedNotebookObject(page) {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, '../../../../helper/', 'addInitRestrictedNotebook.js') });
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
return createDomainObjectWithDefaults(page, { type: CUSTOM_NAME });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function lockPage(page) {
|
||||||
|
const commitButton = page.locator('button:has-text("Commit Entries")');
|
||||||
|
await commitButton.click();
|
||||||
|
|
||||||
|
//Wait until Lock Banner is visible
|
||||||
|
await page.locator('text=Lock Page').click();
|
||||||
|
}
|
212
e2e/tests/functional/plugins/notebook/tags.e2e.spec.js
Normal file
212
e2e/tests/functional/plugins/notebook/tags.e2e.spec.js
Normal file
@ -0,0 +1,212 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 form functionality.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a notebook object and adds an entry.
|
||||||
|
* @param {import('@playwright/test').Page} - page to load
|
||||||
|
* @param {number} [iterations = 1] - the number of entries to create
|
||||||
|
*/
|
||||||
|
async function createNotebookAndEntry(page, iterations = 1) {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
createDomainObjectWithDefaults(page, { type: 'Notebook' });
|
||||||
|
|
||||||
|
for (let iteration = 0; iteration < iterations; iteration++) {
|
||||||
|
// Click text=To start a new entry, click here or drag and drop any object
|
||||||
|
await page.locator('text=To start a new entry, click here or drag and drop any object').click();
|
||||||
|
const entryLocator = `[aria-label="Notebook Entry Input"] >> nth = ${iteration}`;
|
||||||
|
await page.locator(entryLocator).click();
|
||||||
|
await page.locator(entryLocator).fill(`Entry ${iteration}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a notebook object, adds an entry, and adds a tag.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {number} [iterations = 1] - the number of entries (and tags) to create
|
||||||
|
*/
|
||||||
|
async function createNotebookEntryAndTags(page, iterations = 1) {
|
||||||
|
await createNotebookAndEntry(page, iterations);
|
||||||
|
|
||||||
|
for (let iteration = 0; iteration < iterations; iteration++) {
|
||||||
|
// Hover and click "Add Tag" button
|
||||||
|
// Hover is needed here to "slow down" the actions while running in headless mode
|
||||||
|
await page.hover(`button:has-text("Add Tag") >> nth = ${iteration}`);
|
||||||
|
await page.locator(`button:has-text("Add Tag") >> nth = ${iteration}`).click();
|
||||||
|
|
||||||
|
// Click inside the tag search input
|
||||||
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
|
// Select the "Driving" tag
|
||||||
|
await page.locator('[aria-label="Autocomplete Options"] >> text=Driving').click();
|
||||||
|
|
||||||
|
// Hover and click "Add Tag" button
|
||||||
|
// Hover is needed here to "slow down" the actions while running in headless mode
|
||||||
|
await page.hover(`button:has-text("Add Tag") >> nth = ${iteration}`);
|
||||||
|
await page.locator(`button:has-text("Add Tag") >> nth = ${iteration}`).click();
|
||||||
|
// Click inside the tag search input
|
||||||
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
|
// Select the "Science" tag
|
||||||
|
await page.locator('[aria-label="Autocomplete Options"] >> text=Science').click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
test.describe('Tagging in Notebooks @addInit', () => {
|
||||||
|
test('Can load tags', async ({ page }) => {
|
||||||
|
|
||||||
|
await createNotebookAndEntry(page);
|
||||||
|
// Click text=To start a new entry, click here or drag and drop any object
|
||||||
|
await page.locator('button:has-text("Add Tag")').click();
|
||||||
|
|
||||||
|
// Click [placeholder="Type to select tag"]
|
||||||
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
|
|
||||||
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).toContainText("Drilling");
|
||||||
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).toContainText("Driving");
|
||||||
|
});
|
||||||
|
test('Can add tags', async ({ page }) => {
|
||||||
|
await createNotebookEntryAndTags(page);
|
||||||
|
|
||||||
|
await expect(page.locator('[aria-label="Notebook Entry"]')).toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Notebook Entry"]')).toContainText("Driving");
|
||||||
|
|
||||||
|
// Click button:has-text("Add Tag")
|
||||||
|
await page.locator('button:has-text("Add Tag")').click();
|
||||||
|
// Click [placeholder="Type to select tag"]
|
||||||
|
await page.locator('[placeholder="Type to select tag"]').click();
|
||||||
|
|
||||||
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).not.toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).not.toContainText("Driving");
|
||||||
|
await expect(page.locator('[aria-label="Autocomplete Options"]')).toContainText("Drilling");
|
||||||
|
});
|
||||||
|
test('Can search for tags', async ({ page }) => {
|
||||||
|
await createNotebookEntryAndTags(page);
|
||||||
|
// Click [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sc');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toContainText("Driving");
|
||||||
|
|
||||||
|
// Click [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Sc');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toContainText("Driving");
|
||||||
|
|
||||||
|
// Click [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Xq');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toBeHidden();
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toBeHidden();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can delete tags', async ({ page }) => {
|
||||||
|
await createNotebookEntryAndTags(page);
|
||||||
|
await page.locator('[aria-label="Notebook Entries"]').click();
|
||||||
|
// Delete Driving
|
||||||
|
await page.hover('.c-tag__label:has-text("Driving")');
|
||||||
|
await page.locator('.c-tag__label:has-text("Driving") ~ .c-completed-tag-deletion').click();
|
||||||
|
|
||||||
|
await expect(page.locator('[aria-label="Notebook Entry"]')).toContainText("Science");
|
||||||
|
await expect(page.locator('[aria-label="Notebook Entry"]')).not.toContainText("Driving");
|
||||||
|
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sc');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).not.toContainText("Driving");
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can delete objects with tags and neither return in search', async ({ page }) => {
|
||||||
|
await createNotebookEntryAndTags(page);
|
||||||
|
// Delete Notebook
|
||||||
|
await page.locator('button[title="More options"]').click();
|
||||||
|
await page.locator('li[title="Remove this object from its containing object."]').click();
|
||||||
|
await page.locator('button:has-text("OK")').click();
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Unnamed');
|
||||||
|
await expect(page.locator('text=No results found')).toBeVisible();
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('sci');
|
||||||
|
await expect(page.locator('text=No results found')).toBeVisible();
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('dri');
|
||||||
|
await expect(page.locator('text=No results found')).toBeVisible();
|
||||||
|
});
|
||||||
|
test('Tags persist across reload', async ({ page }) => {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, { type: 'Clock' });
|
||||||
|
|
||||||
|
const ITERATIONS = 4;
|
||||||
|
await createNotebookEntryAndTags(page, ITERATIONS);
|
||||||
|
|
||||||
|
for (let iteration = 0; iteration < ITERATIONS; iteration++) {
|
||||||
|
const entryLocator = `[aria-label="Notebook Entry"] >> nth = ${iteration}`;
|
||||||
|
await expect(page.locator(entryLocator)).toContainText("Science");
|
||||||
|
await expect(page.locator(entryLocator)).toContainText("Driving");
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.goto('./#/browse/mine?hideTree=false'),
|
||||||
|
page.click('.c-disclosure-triangle')
|
||||||
|
]);
|
||||||
|
// Click Unnamed Clock
|
||||||
|
await page.click('text="Unnamed Clock"');
|
||||||
|
|
||||||
|
// Click Unnamed Notebook
|
||||||
|
await page.click('text="Unnamed Notebook"');
|
||||||
|
|
||||||
|
for (let iteration = 0; iteration < ITERATIONS; iteration++) {
|
||||||
|
const entryLocator = `[aria-label="Notebook Entry"] >> nth = ${iteration}`;
|
||||||
|
await expect(page.locator(entryLocator)).toContainText("Science");
|
||||||
|
await expect(page.locator(entryLocator)).toContainText("Driving");
|
||||||
|
}
|
||||||
|
|
||||||
|
//Reload Page
|
||||||
|
await Promise.all([
|
||||||
|
page.reload(),
|
||||||
|
page.waitForLoadState('networkidle')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// Click Unnamed Notebook
|
||||||
|
await page.click('text="Unnamed Notebook"');
|
||||||
|
|
||||||
|
for (let iteration = 0; iteration < ITERATIONS; iteration++) {
|
||||||
|
const entryLocator = `[aria-label="Notebook Entry"] >> nth = ${iteration}`;
|
||||||
|
await expect(page.locator(entryLocator)).toContainText("Science");
|
||||||
|
await expect(page.locator(entryLocator)).toContainText("Driving");
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
@ -24,21 +24,7 @@
|
|||||||
Testsuite for plot autoscale.
|
Testsuite for plot autoscale.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test: _test, expect } = require('@playwright/test');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
|
||||||
// create a new `test` API that will not append platform details to snapshot
|
|
||||||
// file names, only for the tests in this file, so that the same snapshots will
|
|
||||||
// be used for all platforms.
|
|
||||||
const test = _test.extend({
|
|
||||||
_autoSnapshotSuffix: [
|
|
||||||
async ({}, use, testInfo) => {
|
|
||||||
testInfo.snapshotSuffix = '';
|
|
||||||
await use();
|
|
||||||
},
|
|
||||||
{ auto: true }
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
test.use({
|
test.use({
|
||||||
viewport: {
|
viewport: {
|
||||||
width: 1280,
|
width: 1280,
|
||||||
@ -47,27 +33,32 @@ test.use({
|
|||||||
});
|
});
|
||||||
|
|
||||||
test.describe('ExportAsJSON', () => {
|
test.describe('ExportAsJSON', () => {
|
||||||
test.slow('User can set autoscale with a valid range @snapshot', async ({ page }) => {
|
test('User can set autoscale with a valid range @snapshot', async ({ page, openmctConfig }) => {
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
//This is necessary due to the size of the test suite.
|
||||||
|
test.slow();
|
||||||
|
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
await setTimeRange(page);
|
await setTimeRange(page);
|
||||||
|
|
||||||
await createSinewaveOverlayPlot(page);
|
await createSinewaveOverlayPlot(page, myItemsFolderName);
|
||||||
|
|
||||||
await testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']);
|
await testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']);
|
||||||
|
|
||||||
await turnOffAutoscale(page);
|
await turnOffAutoscale(page);
|
||||||
|
|
||||||
|
// Make sure that after turning off autoscale, the user selected range values start at the same values the plot had prior.
|
||||||
|
await testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']);
|
||||||
|
|
||||||
const canvas = page.locator('canvas').nth(1);
|
const canvas = page.locator('canvas').nth(1);
|
||||||
|
|
||||||
// Make sure that after turning off autoscale, the user selected range values start at the same values the plot had prior.
|
await canvas.hover({trial: true});
|
||||||
await Promise.all([
|
|
||||||
testYTicks(page, ['-1.00', '-0.50', '0.00', '0.50', '1.00']),
|
|
||||||
new Promise(r => setTimeout(r, 100))
|
|
||||||
.then(() => canvas.screenshot())
|
|
||||||
.then(shot => expect(shot).toMatchSnapshot('autoscale-canvas-prepan.png', { maxDiffPixels: 40 }))
|
|
||||||
]);
|
|
||||||
|
|
||||||
|
expect(await canvas.screenshot()).toMatchSnapshot('autoscale-canvas-prepan.png', { animations: 'disabled' });
|
||||||
|
|
||||||
|
//Alt Drag Start
|
||||||
await page.keyboard.down('Alt');
|
await page.keyboard.down('Alt');
|
||||||
|
|
||||||
await canvas.dragTo(canvas, {
|
await canvas.dragTo(canvas, {
|
||||||
@ -81,15 +72,15 @@ test.describe('ExportAsJSON', () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
//Alt Drag End
|
||||||
await page.keyboard.up('Alt');
|
await page.keyboard.up('Alt');
|
||||||
|
|
||||||
// Ensure the drag worked.
|
// Ensure the drag worked.
|
||||||
await Promise.all([
|
await testYTicks(page, ['0.00', '0.50', '1.00', '1.50', '2.00']);
|
||||||
testYTicks(page, ['0.00', '0.50', '1.00', '1.50', '2.00']),
|
|
||||||
new Promise(r => setTimeout(r, 100))
|
await canvas.hover({trial: true});
|
||||||
.then(() => canvas.screenshot())
|
|
||||||
.then(shot => expect(shot).toMatchSnapshot('autoscale-canvas-panned.png', { maxDiffPixels: 40 }))
|
expect(await canvas.screenshot()).toMatchSnapshot('autoscale-canvas-panned.png', { animations: 'disabled' });
|
||||||
]);
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -112,8 +103,9 @@ async function setTimeRange(page, start = '2022-03-29 22:00:00.000Z', end = '202
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} myItemsFolderName
|
||||||
*/
|
*/
|
||||||
async function createSinewaveOverlayPlot(page) {
|
async function createSinewaveOverlayPlot(page, myItemsFolderName) {
|
||||||
// click create button
|
// click create button
|
||||||
await page.locator('button:has-text("Create")').click();
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
|
||||||
@ -149,7 +141,7 @@ async function createSinewaveOverlayPlot(page) {
|
|||||||
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
||||||
|
|
||||||
// focus the overlay plot
|
// focus the overlay plot
|
||||||
await page.locator('text=Open MCT My Items >> span').nth(3).click();
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
page.waitForNavigation(),
|
page.waitForNavigation(),
|
||||||
page.locator('text=Unnamed Overlay Plot').first().click()
|
page.locator('text=Unnamed Overlay Plot').first().click()
|
Binary file not shown.
After Width: | Height: | Size: 16 KiB |
Binary file not shown.
After Width: | Height: | Size: 15 KiB |
Binary file not shown.
After Width: | Height: | Size: 18 KiB |
Binary file not shown.
After Width: | Height: | Size: 18 KiB |
@ -25,11 +25,15 @@ Tests to verify log plot functionality. Note this test suite if very much under
|
|||||||
necessarily be used for reference when writing new tests in this area.
|
necessarily be used for reference when writing new tests in this area.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
|
||||||
test.describe('Log plot tests', () => {
|
test.describe('Log plot tests', () => {
|
||||||
test.slow('Log Plot ticks are functionally correct in regular and log mode and after refresh', async ({ page }) => {
|
test('Log Plot ticks are functionally correct in regular and log mode and after refresh', async ({ page, openmctConfig }) => {
|
||||||
await makeOverlayPlot(page);
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
//Test.slow decorator is currently broken. Needs to be fixed in https://github.com/nasa/openmct/issues/5374
|
||||||
|
test.slow();
|
||||||
|
|
||||||
|
await makeOverlayPlot(page, myItemsFolderName);
|
||||||
await testRegularTicks(page);
|
await testRegularTicks(page);
|
||||||
await enableEditMode(page);
|
await enableEditMode(page);
|
||||||
await enableLogMode(page);
|
await enableLogMode(page);
|
||||||
@ -40,21 +44,14 @@ test.describe('Log plot tests', () => {
|
|||||||
await testLogTicks(page);
|
await testLogTicks(page);
|
||||||
await saveOverlayPlot(page);
|
await saveOverlayPlot(page);
|
||||||
await testLogTicks(page);
|
await testLogTicks(page);
|
||||||
//await testLogPlotPixels(page);
|
|
||||||
|
|
||||||
// refresh page and wait for charts and ticks to load
|
|
||||||
await page.waitForTimeout(1 * 1000);
|
|
||||||
await page.reload({ waitUntil: 'networkidle'});
|
|
||||||
await page.waitForSelector('.gl-plot-chart-area');
|
|
||||||
await page.waitForSelector('.gl-plot-y-tick-label');
|
|
||||||
|
|
||||||
// test log ticks hold up after refresh
|
|
||||||
await testLogTicks(page);
|
|
||||||
//await testLogPlotPixels(page);
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test.skip('Verify that log mode option is reflected in import/export JSON', async ({ page }) => {
|
// Leaving test as 'TODO' for now.
|
||||||
await makeOverlayPlot(page);
|
// NOTE: Not eligible for community contributions.
|
||||||
|
test.fixme('Verify that log mode option is reflected in import/export JSON', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
await makeOverlayPlot(page, myItemsFolderName);
|
||||||
await enableEditMode(page);
|
await enableEditMode(page);
|
||||||
await enableLogMode(page);
|
await enableLogMode(page);
|
||||||
await saveOverlayPlot(page);
|
await saveOverlayPlot(page);
|
||||||
@ -72,10 +69,11 @@ test.describe('Log plot tests', () => {
|
|||||||
/**
|
/**
|
||||||
* Makes an overlay plot with a sine wave generator and clicks on the overlay plot in the sidebar so it is the active thing displayed.
|
* Makes an overlay plot with a sine wave generator and clicks on the overlay plot in the sidebar so it is the active thing displayed.
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} myItemsFolderName
|
||||||
*/
|
*/
|
||||||
async function makeOverlayPlot(page) {
|
async function makeOverlayPlot(page, myItemsFolderName) {
|
||||||
// fresh page with time range from 2022-03-29 22:00:00.000Z to 2022-03-29 22:00:30.000Z
|
// fresh page with time range from 2022-03-29 22:00:00.000Z to 2022-03-29 22:00:30.000Z
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
// Set a specific time range for consistency, otherwise it will change
|
// Set a specific time range for consistency, otherwise it will change
|
||||||
// on every test to a range based on the current time.
|
// on every test to a range based on the current time.
|
||||||
@ -112,14 +110,14 @@ async function makeOverlayPlot(page) {
|
|||||||
|
|
||||||
// set amplitude to 6, offset 4, period 2
|
// set amplitude to 6, offset 4, period 2
|
||||||
|
|
||||||
await page.locator('div:nth-child(5) .form-row .c-form-row__controls .form-control .field input').click();
|
await page.locator('div:nth-child(5) .c-form-row__controls .form-control .field input').click();
|
||||||
await page.locator('div:nth-child(5) .form-row .c-form-row__controls .form-control .field input').fill('6');
|
await page.locator('div:nth-child(5) .c-form-row__controls .form-control .field input').fill('6');
|
||||||
|
|
||||||
await page.locator('div:nth-child(6) .form-row .c-form-row__controls .form-control .field input').click();
|
await page.locator('div:nth-child(6) .c-form-row__controls .form-control .field input').click();
|
||||||
await page.locator('div:nth-child(6) .form-row .c-form-row__controls .form-control .field input').fill('4');
|
await page.locator('div:nth-child(6) .c-form-row__controls .form-control .field input').fill('4');
|
||||||
|
|
||||||
await page.locator('div:nth-child(7) .form-row .c-form-row__controls .form-control .field input').click();
|
await page.locator('div:nth-child(7) .c-form-row__controls .form-control .field input').click();
|
||||||
await page.locator('div:nth-child(7) .form-row .c-form-row__controls .form-control .field input').fill('2');
|
await page.locator('div:nth-child(7) .c-form-row__controls .form-control .field input').fill('2');
|
||||||
|
|
||||||
// Click OK to make generator
|
// Click OK to make generator
|
||||||
|
|
||||||
@ -135,7 +133,7 @@ async function makeOverlayPlot(page) {
|
|||||||
|
|
||||||
// click on overlay plot
|
// click on overlay plot
|
||||||
|
|
||||||
await page.locator('text=Open MCT My Items >> span').nth(3).click();
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
page.waitForNavigation(),
|
page.waitForNavigation(),
|
||||||
page.locator('text=Unnamed Overlay Plot').first().click()
|
page.locator('text=Unnamed Overlay Plot').first().click()
|
||||||
@ -238,6 +236,8 @@ async function saveOverlayPlot(page) {
|
|||||||
/**
|
/**
|
||||||
* @param {import('@playwright/test').Page} page
|
* @param {import('@playwright/test').Page} page
|
||||||
*/
|
*/
|
||||||
|
// FIXME: Remove this eslint exception once implemented
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
async function testLogPlotPixels(page) {
|
async function testLogPlotPixels(page) {
|
||||||
const pixelsMatch = await page.evaluate(async () => {
|
const pixelsMatch = await page.evaluate(async () => {
|
||||||
// TODO get canvas pixels at a few locations to make sure they're the correct color, to test that the plot comes out as expected.
|
// TODO get canvas pixels at a few locations to make sure they're the correct color, to test that the plot comes out as expected.
|
157
e2e/tests/functional/plugins/plot/missingPlotObj.e2e.spec.js
Normal file
157
e2e/tests/functional/plugins/plot/missingPlotObj.e2e.spec.js
Normal file
@ -0,0 +1,157 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Tests to verify log plot functionality when objects are missing
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
|
||||||
|
test.describe('Handle missing object for plots', () => {
|
||||||
|
test('Displays empty div for missing stacked plot item @unstable', async ({ page, browserName, openmctConfig }) => {
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.skip(browserName === 'firefox', 'Firefox failing due to console events being missed');
|
||||||
|
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
const errorLogs = [];
|
||||||
|
|
||||||
|
page.on("console", (message) => {
|
||||||
|
if (message.type() === 'warning' && message.text().includes('Missing domain object')) {
|
||||||
|
errorLogs.push(message.text());
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
//Make stacked plot
|
||||||
|
await makeStackedPlot(page, myItemsFolderName);
|
||||||
|
|
||||||
|
//Gets local storage and deletes the last sine wave generator in the stacked plot
|
||||||
|
const localStorage = await page.evaluate(() => window.localStorage);
|
||||||
|
const parsedData = JSON.parse(localStorage.mct);
|
||||||
|
const keys = Object.keys(parsedData);
|
||||||
|
const lastKey = keys[keys.length - 1];
|
||||||
|
|
||||||
|
delete parsedData[lastKey];
|
||||||
|
|
||||||
|
//Sets local storage with missing object
|
||||||
|
await page.evaluate(
|
||||||
|
`window.localStorage.setItem('mct', '${JSON.stringify(parsedData)}')`
|
||||||
|
);
|
||||||
|
|
||||||
|
//Reloads page and clicks on stacked plot
|
||||||
|
await Promise.all([
|
||||||
|
page.reload(),
|
||||||
|
page.waitForLoadState('networkidle')
|
||||||
|
]);
|
||||||
|
|
||||||
|
//Verify Main section is there on load
|
||||||
|
await expect.soft(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Stacked Plot');
|
||||||
|
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Unnamed Stacked Plot').first().click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
//Check that there is only one stacked item plot with a plot, the missing one will be empty
|
||||||
|
await expect(page.locator(".c-plot--stacked-container:has(.gl-plot)")).toHaveCount(1);
|
||||||
|
//Verify that console.warn is thrown
|
||||||
|
expect(errorLogs).toHaveLength(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is used the create a stacked plot object
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
async function makeStackedPlot(page, myItemsFolderName) {
|
||||||
|
// fresh page with time range from 2022-03-29 22:00:00.000Z to 2022-03-29 22:00:30.000Z
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// create stacked plot
|
||||||
|
await page.locator('button.c-create-button').click();
|
||||||
|
await page.locator('li:has-text("Stacked Plot")').click();
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({ waitUntil: 'networkidle'}),
|
||||||
|
page.locator('text=OK').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
|
||||||
|
// save the stacked plot
|
||||||
|
await saveStackedPlot(page);
|
||||||
|
|
||||||
|
// create a sinewave generator
|
||||||
|
await createSineWaveGenerator(page);
|
||||||
|
|
||||||
|
// click on stacked plot
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Unnamed Stacked Plot').first().click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
// create a second sinewave generator
|
||||||
|
await createSineWaveGenerator(page);
|
||||||
|
|
||||||
|
// click on stacked plot
|
||||||
|
await page.locator(`text=Open MCT ${myItemsFolderName} >> span`).nth(3).click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Unnamed Stacked Plot').first().click()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is used to save a stacked plot object
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
async function saveStackedPlot(page) {
|
||||||
|
// save stacked plot
|
||||||
|
await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click();
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.locator('text=Save and Finish Editing').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
//Wait until Save Banner is gone
|
||||||
|
await page.locator('.c-message-banner__close-button').click();
|
||||||
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached' });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is used to create a sine wave generator object
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
async function createSineWaveGenerator(page) {
|
||||||
|
//Create sine wave generator
|
||||||
|
await page.locator('button.c-create-button').click();
|
||||||
|
await page.locator('li:has-text("Sine Wave Generator")').click();
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({ waitUntil: 'networkidle'}),
|
||||||
|
page.locator('text=OK').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
}
|
110
e2e/tests/functional/plugins/plot/plotLegendSwatch.e2e.spec.js
Normal file
110
e2e/tests/functional/plugins/plot/plotLegendSwatch.e2e.spec.js
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Tests to verify log plot functionality. Note this test suite if very much under active development and should not
|
||||||
|
necessarily be used for reference when writing new tests in this area.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
|
||||||
|
test.describe('Legend color in sync with plot color', () => {
|
||||||
|
test('Testing', async ({ page }) => {
|
||||||
|
await makeOverlayPlot(page);
|
||||||
|
|
||||||
|
// navigate to plot series color palette
|
||||||
|
await page.click('.l-browse-bar__actions__edit');
|
||||||
|
await page.locator('li.c-tree__item.menus-to-left .c-disclosure-triangle').click();
|
||||||
|
await page.locator('.c-click-swatch--menu').click();
|
||||||
|
await page.locator('.c-palette__item[style="background: rgb(255, 166, 61);"]').click();
|
||||||
|
|
||||||
|
// gets color for swatch located in legend
|
||||||
|
const element = await page.waitForSelector('.plot-series-color-swatch');
|
||||||
|
const color = await element.evaluate((el) => {
|
||||||
|
return window.getComputedStyle(el).getPropertyValue('background-color');
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(color).toBe('rgb(255, 166, 61)');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
async function saveOverlayPlot(page) {
|
||||||
|
// save overlay plot
|
||||||
|
await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click();
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.locator('text=Save and Finish Editing').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
//Wait until Save Banner is gone
|
||||||
|
await page.locator('.c-message-banner__close-button').click();
|
||||||
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached' });
|
||||||
|
}
|
||||||
|
|
||||||
|
async function makeOverlayPlot(page) {
|
||||||
|
// fresh page with time range from 2022-03-29 22:00:00.000Z to 2022-03-29 22:00:30.000Z
|
||||||
|
await page.goto('/', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// create overlay plot
|
||||||
|
|
||||||
|
await page.locator('button.c-create-button').click();
|
||||||
|
await page.locator('li:has-text("Overlay Plot")').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({ waitUntil: 'networkidle'}),
|
||||||
|
page.locator('text=OK').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
//Wait until Save Banner is gone
|
||||||
|
await page.locator('.c-message-banner__close-button').click();
|
||||||
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
||||||
|
|
||||||
|
// save the overlay plot
|
||||||
|
|
||||||
|
await saveOverlayPlot(page);
|
||||||
|
|
||||||
|
// create a sinewave generator
|
||||||
|
|
||||||
|
await page.locator('button.c-create-button').click();
|
||||||
|
await page.locator('li:has-text("Sine Wave Generator")').click();
|
||||||
|
|
||||||
|
// Click OK to make generator
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation({ waitUntil: 'networkidle'}),
|
||||||
|
page.locator('text=OK').click(),
|
||||||
|
//Wait for Save Banner to appear
|
||||||
|
page.waitForSelector('.c-message-banner__message')
|
||||||
|
]);
|
||||||
|
//Wait until Save Banner is gone
|
||||||
|
await page.locator('.c-message-banner__close-button').click();
|
||||||
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
||||||
|
|
||||||
|
// click on overlay plot
|
||||||
|
|
||||||
|
await page.locator('text=Open MCT My Items >> span').nth(3).click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Unnamed Overlay Plot').first().click()
|
||||||
|
]);
|
||||||
|
}
|
@ -0,0 +1,75 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
|
||||||
|
test.describe('Telemetry Table', () => {
|
||||||
|
test('unpauses and filters data when paused by button and user changes bounds', async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5113'
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
const table = await createDomainObjectWithDefaults(page, { type: 'Telemetry Table' });
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Sine Wave Generator',
|
||||||
|
parent: table.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
// focus the Telemetry Table
|
||||||
|
page.goto(table.url);
|
||||||
|
|
||||||
|
// Click pause button
|
||||||
|
const pauseButton = page.locator('button.c-button.icon-pause');
|
||||||
|
await pauseButton.click();
|
||||||
|
|
||||||
|
const tableWrapper = page.locator('div.c-table-wrapper');
|
||||||
|
await expect(tableWrapper).toHaveClass(/is-paused/);
|
||||||
|
|
||||||
|
// Subtract 5 minutes from the current end bound datetime and set it
|
||||||
|
const endTimeInput = page.locator('input[type="text"].c-input--datetime').nth(1);
|
||||||
|
await endTimeInput.click();
|
||||||
|
|
||||||
|
let endDate = await endTimeInput.inputValue();
|
||||||
|
endDate = new Date(endDate);
|
||||||
|
|
||||||
|
endDate.setUTCMinutes(endDate.getUTCMinutes() - 5);
|
||||||
|
endDate = endDate.toISOString().replace(/T/, ' ');
|
||||||
|
|
||||||
|
await endTimeInput.fill('');
|
||||||
|
await endTimeInput.fill(endDate);
|
||||||
|
await page.keyboard.press('Enter');
|
||||||
|
|
||||||
|
await expect(tableWrapper).not.toHaveClass(/is-paused/);
|
||||||
|
|
||||||
|
// Get the most recent telemetry date
|
||||||
|
const latestTelemetryDate = await page.locator('table.c-telemetry-table__body > tbody > tr').last().locator('td').nth(1).getAttribute('title');
|
||||||
|
|
||||||
|
// Verify that it is <= our new end bound
|
||||||
|
const latestMilliseconds = Date.parse(latestTelemetryDate);
|
||||||
|
const endBoundMilliseconds = Date.parse(endDate);
|
||||||
|
expect(latestMilliseconds).toBeLessThanOrEqual(endBoundMilliseconds);
|
||||||
|
});
|
||||||
|
});
|
@ -0,0 +1,170 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../baseFixtures');
|
||||||
|
const { setFixedTimeMode, setRealTimeMode, setStartOffset, setEndOffset } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Time conductor operations', () => {
|
||||||
|
test('validate start time does not exceeds end time', async ({ page }) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
const year = new Date().getFullYear();
|
||||||
|
|
||||||
|
let startDate = 'xxxx-01-01 01:00:00.000Z';
|
||||||
|
startDate = year + startDate.substring(4);
|
||||||
|
|
||||||
|
let endDate = 'xxxx-01-01 02:00:00.000Z';
|
||||||
|
endDate = year + endDate.substring(4);
|
||||||
|
|
||||||
|
const startTimeLocator = page.locator('input[type="text"]').first();
|
||||||
|
const endTimeLocator = page.locator('input[type="text"]').nth(1);
|
||||||
|
|
||||||
|
// Click start time
|
||||||
|
await startTimeLocator.click();
|
||||||
|
|
||||||
|
// Click end time
|
||||||
|
await endTimeLocator.click();
|
||||||
|
|
||||||
|
await endTimeLocator.fill(endDate.toString());
|
||||||
|
await startTimeLocator.fill(startDate.toString());
|
||||||
|
|
||||||
|
// invalid start date
|
||||||
|
startDate = (year + 1) + startDate.substring(4);
|
||||||
|
await startTimeLocator.fill(startDate.toString());
|
||||||
|
await endTimeLocator.click();
|
||||||
|
|
||||||
|
const startDateValidityStatus = await startTimeLocator.evaluate((element) => element.checkValidity());
|
||||||
|
expect(startDateValidityStatus).not.toBeTruthy();
|
||||||
|
|
||||||
|
// fix to valid start date
|
||||||
|
startDate = (year - 1) + startDate.substring(4);
|
||||||
|
await startTimeLocator.fill(startDate.toString());
|
||||||
|
|
||||||
|
// invalid end date
|
||||||
|
endDate = (year - 2) + endDate.substring(4);
|
||||||
|
await endTimeLocator.fill(endDate.toString());
|
||||||
|
await startTimeLocator.click();
|
||||||
|
|
||||||
|
const endDateValidityStatus = await endTimeLocator.evaluate((element) => element.checkValidity());
|
||||||
|
expect(endDateValidityStatus).not.toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Testing instructions:
|
||||||
|
// Try to change the realtime offsets when in realtime (local clock) mode.
|
||||||
|
test.describe('Time conductor input fields real-time mode', () => {
|
||||||
|
test('validate input fields in real-time mode', async ({ page }) => {
|
||||||
|
const startOffset = {
|
||||||
|
secs: '23'
|
||||||
|
};
|
||||||
|
|
||||||
|
const endOffset = {
|
||||||
|
secs: '31'
|
||||||
|
};
|
||||||
|
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Switch to real-time mode
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// Set start time offset
|
||||||
|
await setStartOffset(page, startOffset);
|
||||||
|
|
||||||
|
// Verify time was updated on time offset button
|
||||||
|
await expect(page.locator('data-testid=conductor-start-offset-button')).toContainText('00:30:23');
|
||||||
|
|
||||||
|
// Set end time offset
|
||||||
|
await setEndOffset(page, endOffset);
|
||||||
|
|
||||||
|
// Verify time was updated on preceding time offset button
|
||||||
|
await expect(page.locator('data-testid=conductor-end-offset-button')).toContainText('00:00:31');
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verify that offsets and url params are preserved when switching
|
||||||
|
* between fixed timespan and real-time mode.
|
||||||
|
*/
|
||||||
|
test('preserve offsets and url params when switching between fixed and real-time mode', async ({ page }) => {
|
||||||
|
const startOffset = {
|
||||||
|
mins: '30',
|
||||||
|
secs: '23'
|
||||||
|
};
|
||||||
|
|
||||||
|
const endOffset = {
|
||||||
|
secs: '01'
|
||||||
|
};
|
||||||
|
|
||||||
|
// Convert offsets to milliseconds
|
||||||
|
const startDelta = (30 * 60 * 1000) + (23 * 1000);
|
||||||
|
const endDelta = (1 * 1000);
|
||||||
|
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Switch to real-time mode
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// Set start time offset
|
||||||
|
await setStartOffset(page, startOffset);
|
||||||
|
|
||||||
|
// Set end time offset
|
||||||
|
await setEndOffset(page, endOffset);
|
||||||
|
|
||||||
|
// Switch to fixed timespan mode
|
||||||
|
await setFixedTimeMode(page);
|
||||||
|
|
||||||
|
// Switch back to real-time mode
|
||||||
|
await setRealTimeMode(page);
|
||||||
|
|
||||||
|
// Verify updated start time offset persists after mode switch
|
||||||
|
await expect(page.locator('data-testid=conductor-start-offset-button')).toContainText('00:30:23');
|
||||||
|
|
||||||
|
// Verify updated end time offset persists after mode switch
|
||||||
|
await expect(page.locator('data-testid=conductor-end-offset-button')).toContainText('00:00:01');
|
||||||
|
|
||||||
|
// Verify url parameters persist after mode switch
|
||||||
|
await page.waitForNavigation({ waitUntil: 'networkidle' });
|
||||||
|
expect(page.url()).toContain(`startDelta=${startDelta}`);
|
||||||
|
expect(page.url()).toContain(`endDelta=${endDelta}`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test.fixme('time conductor history in fixed time mode will track changing start and end times', async ({ page }) => {
|
||||||
|
// change start time, verify it's tracked in history
|
||||||
|
// change end time, verify it's tracked in history
|
||||||
|
});
|
||||||
|
|
||||||
|
test.fixme('time conductor history in realtime mode will track changing start and end times', async ({ page }) => {
|
||||||
|
// change start offset, verify it's tracked in history
|
||||||
|
// change end offset, verify it's tracked in history
|
||||||
|
});
|
||||||
|
|
||||||
|
test.fixme('time conductor history allows you to set a historical timeframe', async ({ page }) => {
|
||||||
|
// make sure there are historical history options
|
||||||
|
// select an option and make sure the time conductor start and end bounds are updated correctly
|
||||||
|
});
|
||||||
|
|
||||||
|
test.fixme('time conductor history allows you to set a realtime offsets', async ({ page }) => {
|
||||||
|
// make sure there are realtime history options
|
||||||
|
// select an option and verify the offsets are updated correctly
|
||||||
|
});
|
||||||
|
});
|
156
e2e/tests/functional/plugins/timer/timer.e2e.spec.js
Normal file
156
e2e/tests/functional/plugins/timer/timer.e2e.spec.js
Normal file
@ -0,0 +1,156 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../../../pluginFixtures');
|
||||||
|
const { openObjectTreeContextMenu, createDomainObjectWithDefaults } = require('../../../../appActions');
|
||||||
|
|
||||||
|
test.describe('Timer', () => {
|
||||||
|
let timer;
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
timer = await createDomainObjectWithDefaults(page, { type: 'timer' });
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Can perform actions on the Timer', async ({ page, openmctConfig }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/4313'
|
||||||
|
});
|
||||||
|
|
||||||
|
const timerUrl = timer.url;
|
||||||
|
|
||||||
|
await test.step("From the tree context menu", async () => {
|
||||||
|
await triggerTimerContextMenuAction(page, timerUrl, 'Start');
|
||||||
|
await triggerTimerContextMenuAction(page, timerUrl, 'Pause');
|
||||||
|
await triggerTimerContextMenuAction(page, timerUrl, 'Restart at 0');
|
||||||
|
await triggerTimerContextMenuAction(page, timerUrl, 'Stop');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step("From the 3dot menu", async () => {
|
||||||
|
await triggerTimer3dotMenuAction(page, 'Start');
|
||||||
|
await triggerTimer3dotMenuAction(page, 'Pause');
|
||||||
|
await triggerTimer3dotMenuAction(page, 'Restart at 0');
|
||||||
|
await triggerTimer3dotMenuAction(page, 'Stop');
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step("From the object view", async () => {
|
||||||
|
await triggerTimerViewAction(page, 'Start');
|
||||||
|
await triggerTimerViewAction(page, 'Pause');
|
||||||
|
await triggerTimerViewAction(page, 'Restart at 0');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Actions that can be performed on a timer from context menus.
|
||||||
|
* @typedef {'Start' | 'Stop' | 'Pause' | 'Restart at 0'} TimerAction
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Actions that can be performed on a timer from the object view.
|
||||||
|
* @typedef {'Start' | 'Pause' | 'Restart at 0'} TimerViewAction
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trigger a timer action from the tree context menu
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {TimerAction} action
|
||||||
|
*/
|
||||||
|
async function triggerTimerContextMenuAction(page, timerUrl, action) {
|
||||||
|
const menuAction = `.c-menu ul li >> text="${action}"`;
|
||||||
|
await openObjectTreeContextMenu(page, timerUrl);
|
||||||
|
await page.locator(menuAction).click();
|
||||||
|
assertTimerStateAfterAction(page, action);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trigger a timer action from the 3dot menu
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {TimerAction} action
|
||||||
|
*/
|
||||||
|
async function triggerTimer3dotMenuAction(page, action) {
|
||||||
|
const menuAction = `.c-menu ul li >> text="${action}"`;
|
||||||
|
const threeDotMenuButton = 'button[title="More options"]';
|
||||||
|
let isActionAvailable = false;
|
||||||
|
let iterations = 0;
|
||||||
|
// Dismiss/open the 3dot menu until the action is available
|
||||||
|
// or a maximum number of iterations is reached
|
||||||
|
while (!isActionAvailable && iterations <= 20) {
|
||||||
|
await page.click('.c-object-view');
|
||||||
|
await page.click(threeDotMenuButton);
|
||||||
|
isActionAvailable = await page.locator(menuAction).isVisible();
|
||||||
|
iterations++;
|
||||||
|
}
|
||||||
|
|
||||||
|
await page.locator(menuAction).click();
|
||||||
|
assertTimerStateAfterAction(page, action);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Trigger a timer action from the object view
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {TimerViewAction} action
|
||||||
|
*/
|
||||||
|
async function triggerTimerViewAction(page, action) {
|
||||||
|
await page.locator('.c-timer').hover({trial: true});
|
||||||
|
const buttonTitle = buttonTitleFromAction(action);
|
||||||
|
await page.click(`button[title="${buttonTitle}"]`);
|
||||||
|
assertTimerStateAfterAction(page, action);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Takes in a TimerViewAction and returns the button title
|
||||||
|
* @param {TimerViewAction} action
|
||||||
|
*/
|
||||||
|
function buttonTitleFromAction(action) {
|
||||||
|
switch (action) {
|
||||||
|
case 'Start':
|
||||||
|
return 'Start';
|
||||||
|
case 'Pause':
|
||||||
|
return 'Pause';
|
||||||
|
case 'Restart at 0':
|
||||||
|
return 'Reset';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verify the timer state after a timer action has been performed.
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {TimerAction} action
|
||||||
|
*/
|
||||||
|
async function assertTimerStateAfterAction(page, action) {
|
||||||
|
let timerStateClass;
|
||||||
|
switch (action) {
|
||||||
|
case 'Start':
|
||||||
|
case 'Restart at 0':
|
||||||
|
timerStateClass = "is-started";
|
||||||
|
break;
|
||||||
|
case 'Stop':
|
||||||
|
timerStateClass = 'is-stopped';
|
||||||
|
break;
|
||||||
|
case 'Pause':
|
||||||
|
timerStateClass = 'is-paused';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
await expect.soft(page.locator('.c-timer')).toHaveClass(new RegExp(timerStateClass));
|
||||||
|
}
|
271
e2e/tests/functional/search.e2e.spec.js
Normal file
271
e2e/tests/functional/search.e2e.spec.js
Normal file
@ -0,0 +1,271 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 search functionalities.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
|
const { v4: uuid } = require('uuid');
|
||||||
|
|
||||||
|
test.describe('Grand Search', () => {
|
||||||
|
test('Can search for objects, and subsequent search dropdown behaves properly', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
|
||||||
|
await createObjectsForSearch(page, myItemsFolderName);
|
||||||
|
|
||||||
|
// Click [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Cl');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText(`Clock A ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=1')).toContainText(`Clock B ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=2')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=3')).toContainText(`Clock D ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
// Click text=Elements >> nth=0
|
||||||
|
await page.locator('text=Elements').first().click();
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click();
|
||||||
|
await page.locator('[aria-label="Clock A clock result"] >> text=Clock A').click();
|
||||||
|
await expect(page.locator('.js-preview-window')).toBeVisible();
|
||||||
|
|
||||||
|
// Click [aria-label="Close"]
|
||||||
|
await page.locator('[aria-label="Close"]').click();
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeVisible();
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText(`Clock A ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
|
||||||
|
// Click [aria-label="OpenMCT Search"] a >> nth=0
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] a').first().click();
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
||||||
|
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('foo');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toBeHidden();
|
||||||
|
|
||||||
|
// Click text=Snapshot Save and Finish Editing Save and Continue Editing >> button >> nth=1
|
||||||
|
await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click();
|
||||||
|
// Click text=Save and Finish Editing
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
// Click [aria-label="OpenMCT Search"] [aria-label="Search Input"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click();
|
||||||
|
// Fill [aria-label="OpenMCT Search"] [aria-label="Search Input"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').fill('Cl');
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Clock A').click()
|
||||||
|
]);
|
||||||
|
await expect(page.locator('.is-object-type-clock')).toBeVisible();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').fill('Disp');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText('Unnamed Display Layout');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).not.toContainText('Folder');
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Clock C');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Cloc');
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=0')).toContainText(`Clock A ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=1')).toContainText(`Clock B ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=2')).toContainText(`Clock C ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"] >> nth=3')).toContainText(`Clock D ${myItemsFolderName} Red Folder Blue Folder`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe("Search Tests @unstable", () => {
|
||||||
|
const searchResultSelector = '.c-gsearch-result__title';
|
||||||
|
|
||||||
|
test('Validate empty search result', async ({ page }) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto("./", { waitUntil: "networkidle" });
|
||||||
|
|
||||||
|
// Invalid search for objects
|
||||||
|
await page.type("input[type=search]", 'not found');
|
||||||
|
|
||||||
|
// Wait for search to complete
|
||||||
|
await waitForSearchCompletion(page);
|
||||||
|
|
||||||
|
// Get the search results
|
||||||
|
const searchResults = await page.locator(searchResultSelector);
|
||||||
|
|
||||||
|
// Verify that no results are found
|
||||||
|
expect(await searchResults.count()).toBe(0);
|
||||||
|
|
||||||
|
// Verify proper message appears
|
||||||
|
await expect(page.locator('text=No results found')).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Validate single object in search result @couchdb', async ({ page }) => {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto("./", { waitUntil: "networkidle" });
|
||||||
|
|
||||||
|
// Create a folder object
|
||||||
|
const folderName = uuid();
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'folder',
|
||||||
|
name: folderName
|
||||||
|
});
|
||||||
|
|
||||||
|
// Full search for object
|
||||||
|
await page.type("input[type=search]", folderName);
|
||||||
|
|
||||||
|
// Wait for search to complete
|
||||||
|
await waitForSearchCompletion(page);
|
||||||
|
|
||||||
|
// Get the search results
|
||||||
|
const searchResults = page.locator(searchResultSelector);
|
||||||
|
|
||||||
|
// Verify that one result is found
|
||||||
|
expect(await searchResults.count()).toBe(1);
|
||||||
|
await expect(searchResults).toHaveText(folderName);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Validate multiple objects in search results return partial matches", async ({ page }) => {
|
||||||
|
test.info().annotations.push({
|
||||||
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/4667'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto("/", { waitUntil: "networkidle" });
|
||||||
|
|
||||||
|
// Create folder objects
|
||||||
|
const folderName = "e928a26e-e924-4ea0";
|
||||||
|
const folderName2 = "e928a26e-e924-4001";
|
||||||
|
|
||||||
|
await createFolderObject(page, folderName);
|
||||||
|
await createFolderObject(page, folderName2);
|
||||||
|
|
||||||
|
// Partial search for objects
|
||||||
|
await page.type("input[type=search]", 'e928a26e');
|
||||||
|
|
||||||
|
// Wait for search to finish
|
||||||
|
await waitForSearchCompletion(page);
|
||||||
|
|
||||||
|
// Get the search results
|
||||||
|
const searchResults = await page.locator(searchResultSelector);
|
||||||
|
|
||||||
|
// Verify that the search result/s correctly match the search query
|
||||||
|
expect(await searchResults.count()).toBe(2);
|
||||||
|
await expect(await searchResults.first()).toHaveText(folderName);
|
||||||
|
await expect(await searchResults.last()).toHaveText(folderName2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
async function createFolderObject(page, folderName) {
|
||||||
|
// Open Create menu
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
|
||||||
|
// Select Folder object
|
||||||
|
await page.locator('text=Folder').nth(1).click();
|
||||||
|
|
||||||
|
// Click folder title to enter edit mode
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
||||||
|
|
||||||
|
// Enter folder name
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(folderName);
|
||||||
|
|
||||||
|
// Create folder object
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function waitForSearchCompletion(page) {
|
||||||
|
// Wait loading spinner to disappear
|
||||||
|
await page.waitForSelector('.c-tree-and-search__loading', { state: 'detached' });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates some domain objects for searching
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
*/
|
||||||
|
async function createObjectsForSearch(page, myItemsFolderName) {
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li:has-text("Folder") >> nth=1').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').fill('Red Folder'),
|
||||||
|
await page.locator(`text=Save In Open MCT ${myItemsFolderName} >> span`).nth(3).click(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li:has-text("Folder") >> nth=2').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"]').fill('Blue Folder'),
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Red Folder').click(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li[title="A digital clock that uses system time and supports a variety of display formats and timezones."]').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"] >> nth=0').fill('Clock A'),
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Blue Folder').click(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li[title="A digital clock that uses system time and supports a variety of display formats and timezones."]').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"] >> nth=0').fill('Clock B'),
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Blue Folder').click(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li[title="A digital clock that uses system time and supports a variety of display formats and timezones."]').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"] >> nth=0').fill('Clock C'),
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Blue Folder').click(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
await page.locator('li[title="A digital clock that uses system time and supports a variety of display formats and timezones."]').click();
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
await page.locator('text=Properties Title Notes >> input[type="text"] >> nth=0').fill('Clock D'),
|
||||||
|
await page.locator('form[name="mctForm"] >> text=Blue Folder').click(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator(`a:has-text("${myItemsFolderName}") >> nth=0`).click()
|
||||||
|
]);
|
||||||
|
// Click button:has-text("Create")
|
||||||
|
await page.locator('button:has-text("Create")').click();
|
||||||
|
// Click li:has-text("Notebook")
|
||||||
|
await page.locator('li:has-text("Display Layout")').click();
|
||||||
|
// Click button:has-text("OK")
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('button:has-text("OK")').click()
|
||||||
|
]);
|
||||||
|
}
|
@ -33,17 +33,27 @@ comfortable running this test during a live mission?" Avoid creating or deleting
|
|||||||
Make no assumptions about the order that elements appear in the DOM.
|
Make no assumptions about the order that elements appear in the DOM.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
|
||||||
test('Verify that the create button appears and that the Folder Domain Object is available for selection', async ({ page }) => {
|
test('Verify that the create button appears and that the Folder Domain Object is available for selection', async ({ page }) => {
|
||||||
|
|
||||||
//Go to baseURL
|
//Go to baseURL
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
// Verify that Create Folder appears in the dropdown
|
// Verify that Create Folder appears in the dropdown
|
||||||
const locator = page.locator(':nth-match(:text("Folder"), 2)');
|
await expect(page.locator(':nth-match(:text("Folder"), 2)')).toBeEnabled();
|
||||||
await expect(locator).toBeEnabled();
|
});
|
||||||
|
|
||||||
|
test('Verify that My Items Tree appears @ipad', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
//Test.slow annotation is currently broken. Needs to be fixed in https://github.com/nasa/openmct/issues/5374
|
||||||
|
test.slow();
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./');
|
||||||
|
|
||||||
|
//My Items to be visible
|
||||||
|
await expect(page.locator(`a:has-text("${myItemsFolderName}")`)).toBeEnabled();
|
||||||
});
|
});
|
138
e2e/tests/functional/tree.e2e.spec.js
Normal file
138
e2e/tests/functional/tree.e2e.spec.js
Normal file
@ -0,0 +1,138 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../pluginFixtures.js');
|
||||||
|
const {
|
||||||
|
createDomainObjectWithDefaults,
|
||||||
|
openObjectTreeContextMenu
|
||||||
|
} = require('../../appActions.js');
|
||||||
|
|
||||||
|
test.describe('Tree operations', () => {
|
||||||
|
test('Renaming an object reorders the tree @unstable', async ({ page, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Foo'
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Bar'
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: 'Baz'
|
||||||
|
});
|
||||||
|
|
||||||
|
const clock1 = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Clock',
|
||||||
|
name: 'aaa'
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Clock',
|
||||||
|
name: 'www'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Expand the root folder
|
||||||
|
await expandTreePaneItemByName(page, myItemsFolderName);
|
||||||
|
|
||||||
|
await test.step("Reorders objects with the same tree depth", async () => {
|
||||||
|
await getAndAssertTreeItems(page, ['aaa', 'Bar', 'Baz', 'Foo', 'www']);
|
||||||
|
await renameObjectFromContextMenu(page, clock1.url, 'zzz');
|
||||||
|
await getAndAssertTreeItems(page, ['Bar', 'Baz', 'Foo', 'www', 'zzz']);
|
||||||
|
});
|
||||||
|
|
||||||
|
await test.step("Reorders links to objects as well as original objects", async () => {
|
||||||
|
await page.click('role=treeitem[name=/Bar/]');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/www/]', '.c-object-view');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/zzz/]', '.c-object-view');
|
||||||
|
await page.click('role=treeitem[name=/Baz/]');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/www/]', '.c-object-view');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/zzz/]', '.c-object-view');
|
||||||
|
await page.click('role=treeitem[name=/Foo/]');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/www/]', '.c-object-view');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/zzz/]', '.c-object-view');
|
||||||
|
// Expand the unopened folders
|
||||||
|
await expandTreePaneItemByName(page, 'Bar');
|
||||||
|
await expandTreePaneItemByName(page, 'Baz');
|
||||||
|
await expandTreePaneItemByName(page, 'Foo');
|
||||||
|
|
||||||
|
await renameObjectFromContextMenu(page, clock1.url, '___');
|
||||||
|
await getAndAssertTreeItems(page,
|
||||||
|
[
|
||||||
|
"___",
|
||||||
|
"Bar",
|
||||||
|
"___",
|
||||||
|
"www",
|
||||||
|
"Baz",
|
||||||
|
"___",
|
||||||
|
"www",
|
||||||
|
"Foo",
|
||||||
|
"___",
|
||||||
|
"www",
|
||||||
|
"www"
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {Array<string>} expected
|
||||||
|
*/
|
||||||
|
async function getAndAssertTreeItems(page, expected) {
|
||||||
|
const treeItems = page.locator('[role="treeitem"]');
|
||||||
|
const allTexts = await treeItems.allInnerTexts();
|
||||||
|
// Get rid of root folder ('My Items') as its position will not change
|
||||||
|
allTexts.shift();
|
||||||
|
expect(allTexts).toEqual(expected);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} name
|
||||||
|
*/
|
||||||
|
async function expandTreePaneItemByName(page, name) {
|
||||||
|
const treePane = page.locator('#tree-pane');
|
||||||
|
const treeItem = treePane.locator(`role=treeitem[expanded=false][name=/${name}/]`);
|
||||||
|
const expandTriangle = treeItem.locator('.c-disclosure-triangle');
|
||||||
|
await expandTriangle.click();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} myItemsFolderName
|
||||||
|
* @param {string} url
|
||||||
|
* @param {string} newName
|
||||||
|
*/
|
||||||
|
async function renameObjectFromContextMenu(page, url, newName) {
|
||||||
|
await openObjectTreeContextMenu(page, url);
|
||||||
|
await page.click('li:text("Edit Properties")');
|
||||||
|
const nameInput = page.locator('form[name="mctForm"] .first input[type="text"]');
|
||||||
|
await nameInput.fill("");
|
||||||
|
await nameInput.fill(newName);
|
||||||
|
await page.click('[aria-label="Save"]');
|
||||||
|
}
|
@ -1,141 +0,0 @@
|
|||||||
/*****************************************************************************
|
|
||||||
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding moving objects.
|
|
||||||
*/
|
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
|
||||||
|
|
||||||
test.describe('Move item tests', () => {
|
|
||||||
test('Create a basic object and verify that it can be moved to another folder', async ({ page }) => {
|
|
||||||
// Go to Open MCT
|
|
||||||
await page.goto('/');
|
|
||||||
|
|
||||||
// Create a new folder in the root my items folder
|
|
||||||
let folder1 = "Folder1";
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
|
||||||
await page.locator('li.icon-folder').click();
|
|
||||||
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(folder1);
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.locator('text=OK').click(),
|
|
||||||
page.waitForSelector('.c-message-banner__message')
|
|
||||||
]);
|
|
||||||
//Wait until Save Banner is gone
|
|
||||||
await page.locator('.c-message-banner__close-button').click();
|
|
||||||
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
|
||||||
|
|
||||||
// Create another folder with a new name at default location, which is currently inside Folder 1
|
|
||||||
let folder2 = "Folder2";
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
|
||||||
await page.locator('li.icon-folder').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(folder2);
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.locator('text=OK').click(),
|
|
||||||
page.waitForSelector('.c-message-banner__message')
|
|
||||||
]);
|
|
||||||
//Wait until Save Banner is gone
|
|
||||||
await page.locator('.c-message-banner__close-button').click();
|
|
||||||
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
|
||||||
|
|
||||||
// Move Folder 2 from Folder 1 to My Items
|
|
||||||
await page.locator('text=Open MCT My Items >> span').nth(3).click();
|
|
||||||
await page.locator('.c-tree__scrollable div div:nth-child(2) .c-tree__item .c-tree__item__view-control').click();
|
|
||||||
|
|
||||||
await page.locator(`a:has-text("${folder2}")`).click({
|
|
||||||
button: 'right'
|
|
||||||
});
|
|
||||||
await page.locator('li.icon-move').click();
|
|
||||||
await page.locator('form[name="mctForm"] >> text=My Items').click();
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.locator('text=OK').click()
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Expect that Folder 2 is in My Items, the root folder
|
|
||||||
expect(page.locator(`text=My Items >> nth=0:has(text=${folder2})`)).toBeTruthy();
|
|
||||||
});
|
|
||||||
test('Create a basic object and verify that it cannot be moved to telemetry object without Composition Provider', async ({ page }) => {
|
|
||||||
// Go to Open MCT
|
|
||||||
await page.goto('/');
|
|
||||||
|
|
||||||
// Create Telemetry Table
|
|
||||||
let telemetryTable = 'Test Telemetry Table';
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
|
||||||
await page.locator('li:has-text("Telemetry Table")').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(telemetryTable);
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.locator('text=OK').click()
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Finish editing and save Telemetry Table
|
|
||||||
await page.locator('.c-button--menu.c-button--major.icon-save').click();
|
|
||||||
await page.locator('text=Save and Finish Editing').click();
|
|
||||||
|
|
||||||
// Create New Folder Basic Domain Object
|
|
||||||
let folder = 'Test Folder';
|
|
||||||
await page.locator('button:has-text("Create")').click();
|
|
||||||
await page.locator('li:has-text("Folder")').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').click();
|
|
||||||
await page.locator('text=Properties Title Notes >> input[type="text"]').fill(folder);
|
|
||||||
|
|
||||||
// See if it's possible to put the folder in the Telemetry object during creation (Soft Assert)
|
|
||||||
await page.locator(`form[name="mctForm"] >> text=${telemetryTable}`).click();
|
|
||||||
let okButton = await page.locator('button.c-button.c-button--major:has-text("OK")');
|
|
||||||
let okButtonStateDisabled = await okButton.isDisabled();
|
|
||||||
expect.soft(okButtonStateDisabled).toBeTruthy();
|
|
||||||
|
|
||||||
// Continue test regardless of assertion and create it in My Items
|
|
||||||
await page.locator('form[name="mctForm"] >> text=My Items').click();
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.locator('text=OK').click()
|
|
||||||
]);
|
|
||||||
|
|
||||||
// Open My Items
|
|
||||||
await page.locator('text=Open MCT My Items >> span').nth(3).click();
|
|
||||||
|
|
||||||
// Select Folder Object and select Move from context menu
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation(),
|
|
||||||
page.locator(`a:has-text("${folder}")`).click()
|
|
||||||
]);
|
|
||||||
await page.locator('.c-tree__item.is-navigated-object .c-tree__item__label .c-tree__item__type-icon').click({
|
|
||||||
button: 'right'
|
|
||||||
});
|
|
||||||
await page.locator('li.icon-move').click();
|
|
||||||
|
|
||||||
// See if it's possible to put the folder in the Telemetry object after creation
|
|
||||||
await page.locator('text=Location Open MCT My Items >> span').nth(3).click();
|
|
||||||
await page.locator(`form[name="mctForm"] >> text=${telemetryTable}`).click();
|
|
||||||
let okButton2 = await page.locator('button.c-button.c-button--major:has-text("OK")');
|
|
||||||
let okButtonStateDisabled2 = await okButton2.isDisabled();
|
|
||||||
expect(okButtonStateDisabled2).toBeTruthy();
|
|
||||||
});
|
|
||||||
});
|
|
177
e2e/tests/performance/imagery.perf.spec.js
Normal file
177
e2e/tests/performance/imagery.perf.spec.js
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 performance tests to ensure that testability of performance
|
||||||
|
is not broken upstream on Open MCT. Any assumptions made downstream will be tested here
|
||||||
|
|
||||||
|
TODO:
|
||||||
|
- Update resolution of performance config
|
||||||
|
- Add Performance Observer on init to push all performance marks
|
||||||
|
- Move client CDP connection to before or to a fixture
|
||||||
|
-
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('@playwright/test');
|
||||||
|
|
||||||
|
const filePath = 'e2e/test-data/PerformanceDisplayLayout.json';
|
||||||
|
|
||||||
|
test.describe('Performance tests', () => {
|
||||||
|
test.beforeEach(async ({ page, browser }, testInfo) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Click a:has-text("My Items")
|
||||||
|
await page.locator('a:has-text("My Items")').click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Click text=Import from JSON
|
||||||
|
await page.locator('text=Import from JSON').click();
|
||||||
|
|
||||||
|
// Upload Performance Display Layout.json
|
||||||
|
await page.setInputFiles('#fileElem', filePath);
|
||||||
|
|
||||||
|
// Click text=OK
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
await expect(page.locator('a:has-text("Performance Display Layout Display Layout")')).toBeVisible();
|
||||||
|
|
||||||
|
//Create a Chrome Performance Timeline trace to store as a test artifact
|
||||||
|
console.log("\n==== Devtools: startTracing ====\n");
|
||||||
|
await browser.startTracing(page, {
|
||||||
|
path: `${testInfo.outputPath()}-trace.json`,
|
||||||
|
screenshots: true
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test.afterEach(async ({ page, browser}) => {
|
||||||
|
console.log("\n==== Devtools: stopTracing ====\n");
|
||||||
|
await browser.stopTracing();
|
||||||
|
|
||||||
|
/* Measurement Section
|
||||||
|
/ The following section includes a block of performance measurements.
|
||||||
|
*/
|
||||||
|
//Get time difference between viewlarge actionability and evaluate time
|
||||||
|
await page.evaluate(() => (window.performance.measure("machine-time-difference", "viewlarge.start", "viewLarge.start.test")));
|
||||||
|
|
||||||
|
//Get StartTime
|
||||||
|
const startTime = await page.evaluate(() => window.performance.timing.navigationStart);
|
||||||
|
console.log('window.performance.timing.navigationStart', startTime);
|
||||||
|
|
||||||
|
//Get All Performance Marks
|
||||||
|
const getAllMarksJson = await page.evaluate(() => JSON.stringify(window.performance.getEntriesByType("mark")));
|
||||||
|
const getAllMarks = JSON.parse(getAllMarksJson);
|
||||||
|
console.log('window.performance.getEntriesByType("mark")', getAllMarks);
|
||||||
|
|
||||||
|
//Get All Performance Measures
|
||||||
|
const getAllMeasuresJson = await page.evaluate(() => JSON.stringify(window.performance.getEntriesByType("measure")));
|
||||||
|
const getAllMeasures = JSON.parse(getAllMeasuresJson);
|
||||||
|
console.log('window.performance.getEntriesByType("measure")', getAllMeasures);
|
||||||
|
|
||||||
|
});
|
||||||
|
/* The following test will navigate to a previously created Performance Display Layout and measure the
|
||||||
|
/ following metrics:
|
||||||
|
/ - ElementResourceTiming
|
||||||
|
/ - Interaction Timing
|
||||||
|
*/
|
||||||
|
test('Embedded View Large for Imagery is performant in Fixed Time', async ({ page, browser }) => {
|
||||||
|
const client = await page.context().newCDPSession(page);
|
||||||
|
// Tell the DevTools session to record performance metrics
|
||||||
|
// https://chromedevtools.github.io/devtools-protocol/tot/Performance/#method-getMetrics
|
||||||
|
await client.send('Performance.enable');
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./');
|
||||||
|
|
||||||
|
// Search Available after Launch
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
await page.evaluate(() => window.performance.mark("search-available"));
|
||||||
|
// Fill Search input
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Performance Display Layout');
|
||||||
|
await page.evaluate(() => window.performance.mark("search-entered"));
|
||||||
|
//Search Result Appears and is clicked
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('a:has-text("Performance Display Layout")').first().click(),
|
||||||
|
page.evaluate(() => window.performance.mark("click-search-result"))
|
||||||
|
]);
|
||||||
|
|
||||||
|
//Time to Example Imagery Frame loads within Display Layout
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__bg', { state: 'visible'});
|
||||||
|
//Time to Example Imagery object loads
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__background-image', { state: 'visible'});
|
||||||
|
|
||||||
|
//Get background-image url from background-image css prop
|
||||||
|
const backgroundImage = await page.locator('.c-imagery__main-image__background-image');
|
||||||
|
let backgroundImageUrl = await backgroundImage.evaluate((el) => {
|
||||||
|
return window.getComputedStyle(el).getPropertyValue('background-image').match(/url\(([^)]+)\)/)[1];
|
||||||
|
});
|
||||||
|
backgroundImageUrl = backgroundImageUrl.slice(1, -1); //forgive me, padre
|
||||||
|
console.log('backgroundImageurl ' + backgroundImageUrl);
|
||||||
|
|
||||||
|
//Get ResourceTiming of background-image jpg
|
||||||
|
const resourceTimingJson = await page.evaluate((bgImageUrl) =>
|
||||||
|
JSON.stringify(window.performance.getEntriesByName(bgImageUrl).pop()),
|
||||||
|
backgroundImageUrl
|
||||||
|
);
|
||||||
|
console.log('resourceTimingJson ' + resourceTimingJson);
|
||||||
|
|
||||||
|
//Open Large view
|
||||||
|
await page.locator('button:has-text("Large View")').click(); //This action includes the performance.mark named 'viewLarge.start'
|
||||||
|
await page.evaluate(() => window.performance.mark("viewLarge.start.test")); //This is a mark only to compare evaluate timing
|
||||||
|
|
||||||
|
//Time to Imagery Rendered in Large Frame
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__bg', { state: 'visible'});
|
||||||
|
await page.evaluate(() => window.performance.mark("background-image-frame"));
|
||||||
|
|
||||||
|
//Time to Example Imagery object loads
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__background-image', { state: 'visible'});
|
||||||
|
await page.evaluate(() => window.performance.mark("background-image-visible"));
|
||||||
|
|
||||||
|
// Get Current number of images in thumbstrip
|
||||||
|
await page.waitForSelector('.c-imagery__thumb');
|
||||||
|
const thumbCount = await page.locator('.c-imagery__thumb').count();
|
||||||
|
console.log('number of thumbs rendered ' + thumbCount);
|
||||||
|
await page.locator('.c-imagery__thumb').last().click();
|
||||||
|
|
||||||
|
//Get ResourceTiming of all jpg resources
|
||||||
|
const resourceTimingJson2 = await page.evaluate(() =>
|
||||||
|
JSON.stringify(window.performance.getEntriesByType('resource'))
|
||||||
|
);
|
||||||
|
const resourceTiming = JSON.parse(resourceTimingJson2);
|
||||||
|
const jpgResourceTiming = resourceTiming.find((element) =>
|
||||||
|
element.name.includes('.jpg')
|
||||||
|
);
|
||||||
|
console.log('jpgResourceTiming ' + JSON.stringify(jpgResourceTiming));
|
||||||
|
|
||||||
|
// Click Close Icon
|
||||||
|
await page.locator('[aria-label="Close"]').click();
|
||||||
|
await page.evaluate(() => window.performance.mark("view-large-close-button"));
|
||||||
|
|
||||||
|
//await client.send('HeapProfiler.enable');
|
||||||
|
await client.send('HeapProfiler.collectGarbage');
|
||||||
|
|
||||||
|
let performanceMetrics = await client.send('Performance.getMetrics');
|
||||||
|
console.log(performanceMetrics.metrics);
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
119
e2e/tests/performance/memleak-imagery.perf.spec.js
Normal file
119
e2e/tests/performance/memleak-imagery.perf.spec.js
Normal file
@ -0,0 +1,119 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 an initial example for memory leak testing using performance. This configuration and execution must
|
||||||
|
be kept separate from the traditional performance measurements to avoid any "observer" effects associated with tracing
|
||||||
|
or profiling playwright and/or the browser.
|
||||||
|
|
||||||
|
Based on a pattern identified in https://github.com/trentmwillis/devtools-protocol-demos/blob/master/testing-demos/memory-leak-by-heap.js
|
||||||
|
and https://github.com/paulirish/automated-chrome-profiling/issues/3
|
||||||
|
|
||||||
|
Best path forward: https://github.com/cowchimp/headless-devtools/blob/master/src/Memory/example.js
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('@playwright/test');
|
||||||
|
|
||||||
|
const filePath = 'e2e/test-data/PerformanceDisplayLayout.json';
|
||||||
|
|
||||||
|
// eslint-disable-next-line playwright/no-skipped-test
|
||||||
|
test.describe.skip('Memory Performance tests', () => {
|
||||||
|
test.beforeEach(async ({ page, browser }, testInfo) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Click a:has-text("My Items")
|
||||||
|
await page.locator('a:has-text("My Items")').click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Click text=Import from JSON
|
||||||
|
await page.locator('text=Import from JSON').click();
|
||||||
|
|
||||||
|
// Upload Performance Display Layout.json
|
||||||
|
await page.setInputFiles('#fileElem', filePath);
|
||||||
|
|
||||||
|
// Click text=OK
|
||||||
|
await page.locator('text=OK').click();
|
||||||
|
|
||||||
|
await expect(page.locator('a:has-text("Performance Display Layout Display Layout")')).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Embedded View Large for Imagery is performant in Fixed Time', async ({ page, browser }) => {
|
||||||
|
|
||||||
|
await page.goto('./', {waitUntil: 'networkidle'});
|
||||||
|
|
||||||
|
// To to Search Available after Launch
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
// Fill Search input
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Performance Display Layout');
|
||||||
|
//Search Result Appears and is clicked
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('a:has-text("Performance Display Layout")').first().click()
|
||||||
|
]);
|
||||||
|
|
||||||
|
//Time to Example Imagery Frame loads within Display Layout
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__bg', { state: 'visible'});
|
||||||
|
//Time to Example Imagery object loads
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__background-image', { state: 'visible'});
|
||||||
|
|
||||||
|
const client = await page.context().newCDPSession(page);
|
||||||
|
await client.send('HeapProfiler.enable');
|
||||||
|
await client.send('HeapProfiler.startSampling');
|
||||||
|
// await client.send('HeapProfiler.collectGarbage');
|
||||||
|
await client.send('Performance.enable');
|
||||||
|
|
||||||
|
let performanceMetricsBefore = await client.send('Performance.getMetrics');
|
||||||
|
console.log(performanceMetricsBefore.metrics);
|
||||||
|
|
||||||
|
//await client.send('Performance.disable');
|
||||||
|
|
||||||
|
//Open Large view
|
||||||
|
await page.locator('button:has-text("Large View")').click();
|
||||||
|
await client.send('HeapProfiler.takeHeapSnapshot');
|
||||||
|
|
||||||
|
//Time to Imagery Rendered in Large Frame
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__bg', { state: 'visible'});
|
||||||
|
|
||||||
|
//Time to Example Imagery object loads
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__background-image', { state: 'visible'});
|
||||||
|
|
||||||
|
// Click Close Icon
|
||||||
|
await page.locator('.c-click-icon').click();
|
||||||
|
|
||||||
|
//Time to Example Imagery Frame loads within Display Layout
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__bg', { state: 'visible'});
|
||||||
|
//Time to Example Imagery object loads
|
||||||
|
await page.waitForSelector('.c-imagery__main-image__background-image', { state: 'visible'});
|
||||||
|
|
||||||
|
await client.send('HeapProfiler.collectGarbage');
|
||||||
|
//await client.send('Performance.enable');
|
||||||
|
|
||||||
|
let performanceMetricsAfter = await client.send('Performance.getMetrics');
|
||||||
|
console.log(performanceMetricsAfter.metrics);
|
||||||
|
|
||||||
|
//await client.send('Performance.disable');
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
158
e2e/tests/performance/notebook.perf.spec.js
Normal file
158
e2e/tests/performance/notebook.perf.spec.js
Normal file
@ -0,0 +1,158 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 performance tests to ensure that testability of performance
|
||||||
|
is not broken upstream on Open MCT. Any assumptions made downstream will be tested here.
|
||||||
|
|
||||||
|
TODO:
|
||||||
|
- Update resolution of performance config
|
||||||
|
- Add Performance Observer on init to push all performance marks
|
||||||
|
- Move client CDP connection to before or to a fixture
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('@playwright/test');
|
||||||
|
|
||||||
|
const notebookFilePath = 'e2e/test-data/PerformanceNotebook.json';
|
||||||
|
|
||||||
|
test.describe('Performance tests', () => {
|
||||||
|
test.beforeEach(async ({ page, browser }, testInfo) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Click a:has-text("My Items")
|
||||||
|
await page.locator('a:has-text("My Items")').click({
|
||||||
|
button: 'right'
|
||||||
|
});
|
||||||
|
|
||||||
|
// Click text=Import from JSON
|
||||||
|
await page.locator('text=Import from JSON').click();
|
||||||
|
|
||||||
|
// Upload Performance Display Layout.json
|
||||||
|
await page.setInputFiles('#fileElem', notebookFilePath);
|
||||||
|
|
||||||
|
// TODO Fix this
|
||||||
|
await page.locator('text=OK >> nth=1').click();
|
||||||
|
|
||||||
|
await expect(page.locator('a:has-text("Performance Notebook")')).toBeVisible();
|
||||||
|
|
||||||
|
//Create a Chrome Performance Timeline trace to store as a test artifact
|
||||||
|
console.log("\n==== Devtools: startTracing ====\n");
|
||||||
|
await browser.startTracing(page, {
|
||||||
|
path: `${testInfo.outputPath()}-trace.json`,
|
||||||
|
screenshots: true
|
||||||
|
});
|
||||||
|
});
|
||||||
|
test.afterEach(async ({ page, browser}) => {
|
||||||
|
console.log("\n==== Devtools: stopTracing ====\n");
|
||||||
|
await browser.stopTracing();
|
||||||
|
|
||||||
|
/* Measurement Section
|
||||||
|
/ The following section includes a block of performance measurements.
|
||||||
|
*/
|
||||||
|
const startTime = await page.evaluate(() => window.performance.timing.navigationStart);
|
||||||
|
console.log('window.performance.timing.navigationStart', startTime);
|
||||||
|
|
||||||
|
//Get All Performance Marks
|
||||||
|
const getAllMarksJson = await page.evaluate(() => JSON.stringify(window.performance.getEntriesByType("mark")));
|
||||||
|
const getAllMarks = JSON.parse(getAllMarksJson);
|
||||||
|
console.log('window.performance.getEntriesByType("mark")', getAllMarks);
|
||||||
|
|
||||||
|
//Get All Performance Measures
|
||||||
|
const getAllMeasuresJson = await page.evaluate(() => JSON.stringify(window.performance.getEntriesByType("measure")));
|
||||||
|
const getAllMeasures = JSON.parse(getAllMeasuresJson);
|
||||||
|
console.log('window.performance.getEntriesByType("measure")', getAllMeasures);
|
||||||
|
|
||||||
|
});
|
||||||
|
/* The following test will navigate to a previously created Performance Display Layout and measure the
|
||||||
|
/ following metrics:
|
||||||
|
/ - ElementResourceTiming
|
||||||
|
/ - Interaction Timing
|
||||||
|
*/
|
||||||
|
test('Notebook Search, Add Entry, Update Entry are performant', async ({ page, browser }) => {
|
||||||
|
const client = await page.context().newCDPSession(page);
|
||||||
|
// Tell the DevTools session to record performance metrics
|
||||||
|
// https://chromedevtools.github.io/devtools-protocol/tot/Performance/#method-getMetrics
|
||||||
|
await client.send('Performance.enable');
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./');
|
||||||
|
|
||||||
|
// To to Search Available after Launch
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
await page.evaluate(() => window.performance.mark("search-available"));
|
||||||
|
// Fill Search input
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill('Performance Notebook');
|
||||||
|
await page.evaluate(() => window.performance.mark("search-entered"));
|
||||||
|
//Search Result Appears and is clicked
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('a:has-text("Performance Notebook")').first().click(),
|
||||||
|
page.evaluate(() => window.performance.mark("click-search-result"))
|
||||||
|
]);
|
||||||
|
|
||||||
|
await page.waitForSelector('.c-tree__item c-tree-and-search__loading loading', {state: 'hidden'});
|
||||||
|
await page.evaluate(() => window.performance.mark("search-spinner-gone"));
|
||||||
|
|
||||||
|
await page.waitForSelector('.l-browse-bar__object-name', { state: 'visible'});
|
||||||
|
await page.evaluate(() => window.performance.mark("object-title-appears"));
|
||||||
|
|
||||||
|
await page.waitForSelector('.c-notebook__entry >> nth=0', { state: 'visible'});
|
||||||
|
await page.evaluate(() => window.performance.mark("notebook-entry-appears"));
|
||||||
|
|
||||||
|
// Click Add new Notebook Entry
|
||||||
|
await page.locator('.c-notebook__drag-area').click();
|
||||||
|
await page.evaluate(() => window.performance.mark("new-notebook-entry-created"));
|
||||||
|
|
||||||
|
// Enter Notebook Entry text
|
||||||
|
await page.locator('div.c-ne__text').last().fill('New Entry');
|
||||||
|
await page.keyboard.press('Enter');
|
||||||
|
await page.evaluate(() => window.performance.mark("new-notebook-entry-filled"));
|
||||||
|
|
||||||
|
//Individual Notebook Entry Search
|
||||||
|
await page.evaluate(() => window.performance.mark("notebook-search-start"));
|
||||||
|
await page.locator('.c-notebook__search >> input').fill('Existing Entry');
|
||||||
|
await page.evaluate(() => window.performance.mark("notebook-search-filled"));
|
||||||
|
await page.waitForSelector('text=Search Results (3)', { state: 'visible'});
|
||||||
|
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
||||||
|
await page.waitForSelector('.c-notebook__entry >> nth=2', { state: 'visible'});
|
||||||
|
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
||||||
|
|
||||||
|
//Clear Search
|
||||||
|
await page.locator('.c-search.c-notebook__search .c-search__clear-input').click();
|
||||||
|
await page.evaluate(() => window.performance.mark("notebook-search-processed"));
|
||||||
|
|
||||||
|
// Hover on Last
|
||||||
|
await page.evaluate(() => window.performance.mark("new-notebook-entry-delete"));
|
||||||
|
await page.locator('div.c-ne__time-and-content').last().hover();
|
||||||
|
await page.locator('button[title="Delete this entry"]').last().click();
|
||||||
|
await page.locator('button:has-text("Ok")').click();
|
||||||
|
await page.waitForSelector('.c-notebook__entry >> nth=3', { state: 'detached'});
|
||||||
|
await page.evaluate(() => window.performance.mark("new-notebook-entry-deleted"));
|
||||||
|
|
||||||
|
//await client.send('HeapProfiler.enable');
|
||||||
|
await client.send('HeapProfiler.collectGarbage');
|
||||||
|
|
||||||
|
let performanceMetrics = await client.send('Performance.getMetrics');
|
||||||
|
console.log(performanceMetrics.metrics);
|
||||||
|
});
|
||||||
|
});
|
@ -1,238 +0,0 @@
|
|||||||
/*****************************************************************************
|
|
||||||
* Open MCT, Copyright (c) 2014-2022, 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 the basic operations surrounding imagery,
|
|
||||||
but only assume that example imagery is present.
|
|
||||||
*/
|
|
||||||
/* globals process */
|
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
|
||||||
|
|
||||||
test.describe('Example Imagery', () => {
|
|
||||||
|
|
||||||
test.beforeEach(async ({ page }) => {
|
|
||||||
page.on('console', msg => console.log(msg.text()));
|
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
//Click the Create button
|
|
||||||
await page.click('button:has-text("Create")');
|
|
||||||
|
|
||||||
// Click text=Example Imagery
|
|
||||||
await page.click('text=Example Imagery');
|
|
||||||
|
|
||||||
// Click text=OK
|
|
||||||
await Promise.all([
|
|
||||||
page.waitForNavigation({waitUntil: 'networkidle'}),
|
|
||||||
page.click('text=OK'),
|
|
||||||
//Wait for Save Banner to appear
|
|
||||||
page.waitForSelector('.c-message-banner__message')
|
|
||||||
]);
|
|
||||||
//Wait until Save Banner is gone
|
|
||||||
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
|
||||||
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Example Imagery');
|
|
||||||
});
|
|
||||||
|
|
||||||
const backgroundImageSelector = '.c-imagery__main-image__background-image';
|
|
||||||
test('Can use Mouse Wheel to zoom in and out of latest image', async ({ page }) => {
|
|
||||||
const bgImageLocator = await page.locator(backgroundImageSelector);
|
|
||||||
const deltaYStep = 100; //equivalent to 1x zoom
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const originalImageDimensions = await page.locator(backgroundImageSelector).boundingBox();
|
|
||||||
// zoom in
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
await page.mouse.wheel(0, deltaYStep * 2);
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const imageMouseZoomedIn = await page.locator(backgroundImageSelector).boundingBox();
|
|
||||||
// zoom out
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
await page.mouse.wheel(0, -deltaYStep);
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const imageMouseZoomedOut = await page.locator(backgroundImageSelector).boundingBox();
|
|
||||||
|
|
||||||
expect(imageMouseZoomedIn.height).toBeGreaterThan(originalImageDimensions.height);
|
|
||||||
expect(imageMouseZoomedIn.width).toBeGreaterThan(originalImageDimensions.width);
|
|
||||||
expect(imageMouseZoomedOut.height).toBeLessThan(imageMouseZoomedIn.height);
|
|
||||||
expect(imageMouseZoomedOut.width).toBeLessThan(imageMouseZoomedIn.width);
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
test('Can use alt+drag to move around image once zoomed in', async ({ page }) => {
|
|
||||||
const deltaYStep = 100; //equivalent to 1x zoom
|
|
||||||
const panHotkey = process.platform === 'linux' ? ['Control', 'Alt'] : ['Alt'];
|
|
||||||
|
|
||||||
const bgImageLocator = await page.locator(backgroundImageSelector);
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
|
|
||||||
// zoom in
|
|
||||||
await page.mouse.wheel(0, deltaYStep * 2);
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const zoomedBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
const imageCenterX = zoomedBoundingBox.x + zoomedBoundingBox.width / 2;
|
|
||||||
const imageCenterY = zoomedBoundingBox.y + zoomedBoundingBox.height / 2;
|
|
||||||
// move to the right
|
|
||||||
|
|
||||||
// center the mouse pointer
|
|
||||||
await page.mouse.move(imageCenterX, imageCenterY);
|
|
||||||
|
|
||||||
//Get Diagnostic info about process environment
|
|
||||||
console.log('process.platform is ' + process.platform);
|
|
||||||
const getUA = await page.evaluate(() => navigator.userAgent);
|
|
||||||
console.log('navigator.userAgent ' + getUA);
|
|
||||||
// Pan Imagery Hints
|
|
||||||
const expectedAltText = process.platform === 'linux' ? 'Ctrl+Alt drag to pan' : 'Alt drag to pan';
|
|
||||||
const imageryHintsText = await page.locator('.c-imagery__hints').innerText();
|
|
||||||
expect(expectedAltText).toEqual(imageryHintsText);
|
|
||||||
|
|
||||||
// pan right
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
|
||||||
await page.mouse.down();
|
|
||||||
await page.mouse.move(imageCenterX - 200, imageCenterY, 10);
|
|
||||||
await page.mouse.up();
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
|
||||||
const afterRightPanBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect(zoomedBoundingBox.x).toBeGreaterThan(afterRightPanBoundingBox.x);
|
|
||||||
|
|
||||||
// pan left
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
|
||||||
await page.mouse.down();
|
|
||||||
await page.mouse.move(imageCenterX, imageCenterY, 10);
|
|
||||||
await page.mouse.up();
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
|
||||||
const afterLeftPanBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect(afterRightPanBoundingBox.x).toBeLessThan(afterLeftPanBoundingBox.x);
|
|
||||||
|
|
||||||
// pan up
|
|
||||||
await page.mouse.move(imageCenterX, imageCenterY);
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
|
||||||
await page.mouse.down();
|
|
||||||
await page.mouse.move(imageCenterX, imageCenterY + 200, 10);
|
|
||||||
await page.mouse.up();
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
|
||||||
const afterUpPanBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect(afterUpPanBoundingBox.y).toBeGreaterThan(afterLeftPanBoundingBox.y);
|
|
||||||
|
|
||||||
// pan down
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.down(x)));
|
|
||||||
await page.mouse.down();
|
|
||||||
await page.mouse.move(imageCenterX, imageCenterY - 200, 10);
|
|
||||||
await page.mouse.up();
|
|
||||||
await Promise.all(panHotkey.map(x => page.keyboard.up(x)));
|
|
||||||
const afterDownPanBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect(afterDownPanBoundingBox.y).toBeLessThan(afterUpPanBoundingBox.y);
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
test('Can use + - buttons to zoom on the image', async ({ page }) => {
|
|
||||||
const bgImageLocator = await page.locator(backgroundImageSelector);
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const zoomInBtn = await page.locator('.t-btn-zoom-in');
|
|
||||||
const zoomOutBtn = await page.locator('.t-btn-zoom-out');
|
|
||||||
const initialBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
|
|
||||||
await zoomInBtn.click();
|
|
||||||
await zoomInBtn.click();
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const zoomedInBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect(zoomedInBoundingBox.height).toBeGreaterThan(initialBoundingBox.height);
|
|
||||||
expect(zoomedInBoundingBox.width).toBeGreaterThan(initialBoundingBox.width);
|
|
||||||
|
|
||||||
await zoomOutBtn.click();
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
const zoomedOutBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect(zoomedOutBoundingBox.height).toBeLessThan(zoomedInBoundingBox.height);
|
|
||||||
expect(zoomedOutBoundingBox.width).toBeLessThan(zoomedInBoundingBox.width);
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
test('Can use the reset button to reset the image', async ({ page }) => {
|
|
||||||
const bgImageLocator = await page.locator(backgroundImageSelector);
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
|
|
||||||
const zoomInBtn = await page.locator('.t-btn-zoom-in');
|
|
||||||
const zoomResetBtn = await page.locator('.t-btn-zoom-reset');
|
|
||||||
const initialBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
|
|
||||||
await zoomInBtn.click();
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
await zoomInBtn.click();
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
|
|
||||||
const zoomedInBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect.soft(zoomedInBoundingBox.height).toBeGreaterThan(initialBoundingBox.height);
|
|
||||||
expect.soft(zoomedInBoundingBox.width).toBeGreaterThan(initialBoundingBox.width);
|
|
||||||
|
|
||||||
await zoomResetBtn.click();
|
|
||||||
// wait for zoom animation to finish
|
|
||||||
await bgImageLocator.hover();
|
|
||||||
|
|
||||||
const resetBoundingBox = await bgImageLocator.boundingBox();
|
|
||||||
expect.soft(resetBoundingBox.height).toBeLessThan(zoomedInBoundingBox.height);
|
|
||||||
expect.soft(resetBoundingBox.width).toBeLessThan(zoomedInBoundingBox.width);
|
|
||||||
|
|
||||||
expect.soft(resetBoundingBox.height).toEqual(initialBoundingBox.height);
|
|
||||||
expect(resetBoundingBox.width).toEqual(initialBoundingBox.width);
|
|
||||||
});
|
|
||||||
|
|
||||||
//test.fixme('Can use Mouse Wheel to zoom in and out of previous image');
|
|
||||||
//test.fixme('Can zoom into the latest image and the real-time/fixed-time imagery will pause');
|
|
||||||
//test.fixme('Can zoom into a previous image from thumbstrip in real-time or fixed-time');
|
|
||||||
//test.fixme('Clicking on the left arrow should pause the imagery and go to previous image');
|
|
||||||
//test.fixme('If the imagery view is in pause mode, it should not be updated when new images come in');
|
|
||||||
//test.fixme('If the imagery view is not in pause mode, it should be updated when new images come in');
|
|
||||||
});
|
|
||||||
|
|
||||||
test.describe('Example Imagery in Display layout', () => {
|
|
||||||
test.fixme('Can use Mouse Wheel to zoom in and out of previous image');
|
|
||||||
test.fixme('Can use alt+drag to move around image once zoomed in');
|
|
||||||
test.fixme('Can zoom into the latest image and the real-time/fixed-time imagery will pause');
|
|
||||||
test.fixme('Clicking on the left arrow should pause the imagery and go to previous image');
|
|
||||||
test.fixme('If the imagery view is in pause mode, it should not be updated when new images come in');
|
|
||||||
test.fixme('If the imagery view is not in pause mode, it should be updated when new images come in');
|
|
||||||
});
|
|
||||||
|
|
||||||
test.describe('Example Imagery in Flexible layout', () => {
|
|
||||||
test.fixme('Can use Mouse Wheel to zoom in and out of previous image');
|
|
||||||
test.fixme('Can use alt+drag to move around image once zoomed in');
|
|
||||||
test.fixme('Can zoom into the latest image and the real-time/fixed-time imagery will pause');
|
|
||||||
test.fixme('Clicking on the left arrow should pause the imagery and go to previous image');
|
|
||||||
test.fixme('If the imagery view is in pause mode, it should not be updated when new images come in');
|
|
||||||
test.fixme('If the imagery view is not in pause mode, it should be updated when new images come in');
|
|
||||||
});
|
|
||||||
|
|
||||||
test.describe('Example Imagery in Tabs view', () => {
|
|
||||||
test.fixme('Can use Mouse Wheel to zoom in and out of previous image');
|
|
||||||
test.fixme('Can use alt+drag to move around image once zoomed in');
|
|
||||||
test.fixme('Can zoom into the latest image and the real-time/fixed-time imagery will pause');
|
|
||||||
test.fixme('Can zoom into a previous image from thumbstrip in real-time or fixed-time');
|
|
||||||
test.fixme('Clicking on the left arrow should pause the imagery and go to previous image');
|
|
||||||
test.fixme('If the imagery view is in pause mode, it should not be updated when new images come in');
|
|
||||||
test.fixme('If the imagery view is not in pause mode, it should be updated when new images come in');
|
|
||||||
});
|
|
Binary file not shown.
Before Width: | Height: | Size: 15 KiB |
Binary file not shown.
Before Width: | Height: | Size: 18 KiB |
@ -1,112 +0,0 @@
|
|||||||
/*****************************************************************************
|
|
||||||
* Open MCT, Copyright (c) 2014-2022, 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.
|
|
||||||
*****************************************************************************/
|
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
|
||||||
|
|
||||||
test.describe('Time counductor operations', () => {
|
|
||||||
test('validate start time does not exceeds end time', async ({ page }) => {
|
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
const year = new Date().getFullYear();
|
|
||||||
|
|
||||||
let startDate = 'xxxx-01-01 01:00:00.000Z';
|
|
||||||
startDate = year + startDate.substring(4);
|
|
||||||
|
|
||||||
let endDate = 'xxxx-01-01 02:00:00.000Z';
|
|
||||||
endDate = year + endDate.substring(4);
|
|
||||||
|
|
||||||
const startTimeLocator = page.locator('input[type="text"]').first();
|
|
||||||
const endTimeLocator = page.locator('input[type="text"]').nth(1);
|
|
||||||
|
|
||||||
// Click start time
|
|
||||||
await startTimeLocator.click();
|
|
||||||
|
|
||||||
// Click end time
|
|
||||||
await endTimeLocator.click();
|
|
||||||
|
|
||||||
await endTimeLocator.fill(endDate.toString());
|
|
||||||
await startTimeLocator.fill(startDate.toString());
|
|
||||||
|
|
||||||
// invalid start date
|
|
||||||
startDate = (year + 1) + startDate.substring(4);
|
|
||||||
await startTimeLocator.fill(startDate.toString());
|
|
||||||
await endTimeLocator.click();
|
|
||||||
|
|
||||||
const startDateValidityStatus = await startTimeLocator.evaluate((element) => element.checkValidity());
|
|
||||||
expect(startDateValidityStatus).not.toBeTruthy();
|
|
||||||
|
|
||||||
// fix to valid start date
|
|
||||||
startDate = (year - 1) + startDate.substring(4);
|
|
||||||
await startTimeLocator.fill(startDate.toString());
|
|
||||||
|
|
||||||
// invalid end date
|
|
||||||
endDate = (year - 2) + endDate.substring(4);
|
|
||||||
await endTimeLocator.fill(endDate.toString());
|
|
||||||
await startTimeLocator.click();
|
|
||||||
|
|
||||||
const endDateValidityStatus = await endTimeLocator.evaluate((element) => element.checkValidity());
|
|
||||||
expect(endDateValidityStatus).not.toBeTruthy();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
// Testing instructions:
|
|
||||||
// Try to change the realtime offsets when in realtime (local clock) mode.
|
|
||||||
test.describe('Time conductor input fields real-time mode', () => {
|
|
||||||
test('validate input fields in real-time mode', async ({ page }) => {
|
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
// Set realtime "local clock" mode offsets
|
|
||||||
const timeInputs = page.locator('input.c-input--datetime');
|
|
||||||
|
|
||||||
// Click fixed timespan button
|
|
||||||
await page.locator('.c-button__label >> text=Fixed Timespan').click();
|
|
||||||
|
|
||||||
// Click local clock
|
|
||||||
await page.locator('.icon-clock >> text=Local Clock').click();
|
|
||||||
|
|
||||||
// Click time offset button
|
|
||||||
await page.locator('.c-conductor__delta-button >> text=00:30:00').click();
|
|
||||||
|
|
||||||
// Input start time offset
|
|
||||||
await page.fill('.pr-time-controls__secs', '23');
|
|
||||||
|
|
||||||
// Click the check button
|
|
||||||
await page.locator('.icon-check').click();
|
|
||||||
|
|
||||||
// Verify time was updated on time offset button
|
|
||||||
await expect(page.locator('.c-conductor__delta-button').first()).toContainText('00:30:23');
|
|
||||||
|
|
||||||
// Click time offset set preceding now button
|
|
||||||
await page.locator('.c-conductor__delta-button >> text=00:00:30').click();
|
|
||||||
|
|
||||||
// Input preceding time offset
|
|
||||||
await page.fill('.pr-time-controls__secs', '31')
|
|
||||||
|
|
||||||
// Click the check buttons
|
|
||||||
await page.locator('.icon-check').click();
|
|
||||||
|
|
||||||
// Verify time was updated on preceding time offset button
|
|
||||||
await expect(page.locator('.c-conductor__delta-button').nth(1)).toContainText('00:00:31');
|
|
||||||
});
|
|
||||||
});
|
|
@ -1,22 +0,0 @@
|
|||||||
{
|
|
||||||
"cookies": [],
|
|
||||||
"origins": [
|
|
||||||
{
|
|
||||||
"origin": "http://localhost:8080",
|
|
||||||
"localStorage": [
|
|
||||||
{
|
|
||||||
"name": "tcHistory",
|
|
||||||
"value": "{\"utc\":[{\"start\":1651513945533,\"end\":1651515745533}]}"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "mct",
|
|
||||||
"value": "{\"mine\":{\"identifier\":{\"key\":\"mine\",\"namespace\":\"\"},\"name\":\"My Items\",\"type\":\"folder\",\"composition\":[{\"key\":\"c0f99e39-85e7-4ef7-99b1-ef52d4ed69b2\",\"namespace\":\"\"}],\"location\":\"ROOT\",\"persisted\":1651515746374,\"modified\":1651515746374},\"c0f99e39-85e7-4ef7-99b1-ef52d4ed69b2\":{\"name\":\"Unnamed Condition Set\",\"type\":\"conditionSet\",\"identifier\":{\"key\":\"c0f99e39-85e7-4ef7-99b1-ef52d4ed69b2\",\"namespace\":\"\"},\"configuration\":{\"conditionTestData\":[],\"conditionCollection\":[{\"isDefault\":true,\"id\":\"e35a066b-eb0e-4b05-a4c9-cc31dc202572\",\"configuration\":{\"name\":\"Default\",\"output\":\"Default\",\"trigger\":\"all\",\"criteria\":[]},\"summary\":\"Default condition\"}]},\"composition\":[],\"telemetry\":{},\"modified\":1651515746373,\"location\":\"mine\",\"persisted\":1651515746373}}"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "mct-tree-expanded",
|
|
||||||
"value": "[]"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
62
e2e/tests/visual/addInit.visual.spec.js
Normal file
62
e2e/tests/visual/addInit.visual.spec.js
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
/* eslint-disable no-undef */
|
||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Collection of Visual Tests set to run with modified init scripts to inject plugins not otherwise available in the default contexts.
|
||||||
|
|
||||||
|
These should only use functional expect statements to verify assumptions about the state
|
||||||
|
in a test and not for functional verification of correctness. Visual tests are not supposed
|
||||||
|
to "fail" on assertions. Instead, they should be used to detect changes between builds or branches.
|
||||||
|
|
||||||
|
Note: Larger testsuite sizes are OK due to the setup time associated with these tests.
|
||||||
|
*/
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
const path = require('path');
|
||||||
|
|
||||||
|
const CUSTOM_NAME = 'CUSTOM_NAME';
|
||||||
|
|
||||||
|
test.describe('Visual - addInit', () => {
|
||||||
|
test.use({
|
||||||
|
clockOptions: {
|
||||||
|
now: 0, //Set browser clock to UNIX Epoch
|
||||||
|
shouldAdvanceTime: false //Don't advance the clock
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Restricted Notebook is visually correct @addInit @unstable', async ({ page, theme }) => {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, '../../helper', './addInitRestrictedNotebook.js') });
|
||||||
|
//Go to baseURL
|
||||||
|
await page.goto('./#/browse/mine?hideTree=true', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, { type: CUSTOM_NAME });
|
||||||
|
|
||||||
|
// Take a snapshot of the newly created CUSTOM_NAME notebook
|
||||||
|
await percySnapshot(page, `Restricted Notebook with CUSTOM_NAME (theme: '${theme}')`);
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
101
e2e/tests/visual/components/tree.visual.spec.js
Normal file
101
e2e/tests/visual/components/tree.visual.spec.js
Normal file
@ -0,0 +1,101 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test } = require('../../../pluginFixtures.js');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../../appActions.js');
|
||||||
|
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
|
||||||
|
test.describe('Visual - Tree Pane', () => {
|
||||||
|
test('Tree pane in various states @unstable', async ({ page, theme, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
const foo = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: "Foo Folder"
|
||||||
|
});
|
||||||
|
|
||||||
|
const bar = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: "Bar Folder",
|
||||||
|
parent: foo.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
const baz = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: "Baz Folder",
|
||||||
|
parent: bar.uuid
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Clock',
|
||||||
|
name: 'A Clock'
|
||||||
|
});
|
||||||
|
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Clock',
|
||||||
|
name: 'Z Clock'
|
||||||
|
});
|
||||||
|
|
||||||
|
const treePane = "#tree-pane";
|
||||||
|
|
||||||
|
await percySnapshot(page, `Tree Pane w/ collapsed tree (theme: ${theme})`, {
|
||||||
|
scope: treePane
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page, myItemsFolderName);
|
||||||
|
|
||||||
|
await page.goto(foo.url);
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/A Clock/]', '.c-object-view');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/Z Clock/]', '.c-object-view');
|
||||||
|
await page.goto(bar.url);
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/A Clock/]', '.c-object-view');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/Z Clock/]', '.c-object-view');
|
||||||
|
await page.goto(baz.url);
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/A Clock/]', '.c-object-view');
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/Z Clock/]', '.c-object-view');
|
||||||
|
|
||||||
|
await percySnapshot(page, `Tree Pane w/ single level expanded (theme: ${theme})`, {
|
||||||
|
scope: treePane
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page, foo.name);
|
||||||
|
await expandTreePaneItemByName(page, bar.name);
|
||||||
|
await expandTreePaneItemByName(page, baz.name);
|
||||||
|
|
||||||
|
await percySnapshot(page, `Tree Pane w/ multiple levels expanded (theme: ${theme})`, {
|
||||||
|
scope: treePane
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {import('@playwright/test').Page} page
|
||||||
|
* @param {string} name
|
||||||
|
*/
|
||||||
|
async function expandTreePaneItemByName(page, name) {
|
||||||
|
const treePane = page.locator('#tree-pane');
|
||||||
|
const treeItem = treePane.locator(`role=treeitem[expanded=false][name=/${name}/]`);
|
||||||
|
const expandTriangle = treeItem.locator('.c-disclosure-triangle');
|
||||||
|
await expandTriangle.click();
|
||||||
|
}
|
56
e2e/tests/visual/controlledClock.visual.spec.js
Normal file
56
e2e/tests/visual/controlledClock.visual.spec.js
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Collection of Visual Tests set to run in a default context. The tests within this suite
|
||||||
|
are only meant to run against openmct's app.js started by `npm run start` within the
|
||||||
|
`./e2e/playwright-visual.config.js` file.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
|
||||||
|
test.describe('Visual - Controlled Clock @localStorage', () => {
|
||||||
|
test.use({
|
||||||
|
storageState: './e2e/test-data/VisualTestData_storage.json',
|
||||||
|
clockOptions: {
|
||||||
|
now: 0, //Set browser clock to UNIX Epoch
|
||||||
|
shouldAdvanceTime: false //Don't advance the clock
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Overlay Plot Loading Indicator @localStorage', async ({ page, theme }) => {
|
||||||
|
// Go to baseURL
|
||||||
|
await page.goto('./#/browse/mine?hideTree=true', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await page.locator('a:has-text("Unnamed Overlay Plot Overlay Plot")').click();
|
||||||
|
//Ensure that we're on the Unnamed Overlay Plot object
|
||||||
|
await expect(page.locator('.l-browse-bar__object-name')).toContainText('Unnamed Overlay Plot');
|
||||||
|
|
||||||
|
//Wait for canvas to be rendered and stop animating
|
||||||
|
await page.locator('canvas >> nth=1').hover({trial: true});
|
||||||
|
|
||||||
|
//Take snapshot of Sine Wave Generator within Overlay Plot
|
||||||
|
await percySnapshot(page, `SineWaveInOverlayPlot (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
});
|
@ -32,89 +32,62 @@ to "fail" on assertions. Instead, they should be used to detect changes between
|
|||||||
Note: Larger testsuite sizes are OK due to the setup time associated with these tests.
|
Note: Larger testsuite sizes are OK due to the setup time associated with these tests.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const { test, expect } = require('@playwright/test');
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
const percySnapshot = require('@percy/playwright');
|
const percySnapshot = require('@percy/playwright');
|
||||||
const path = require('path');
|
const { createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
const sinon = require('sinon');
|
|
||||||
|
|
||||||
const VISUAL_GRACE_PERIOD = 5 * 1000; //Lets the application "simmer" before the snapshot is taken
|
test.describe('Visual - Default', () => {
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
// Snippet from https://github.com/microsoft/playwright/issues/6347#issuecomment-965887758
|
//Go to baseURL and Hide Tree
|
||||||
// Will replace with cy.clock() equivalent
|
await page.goto('./#/browse/mine?hideTree=true', { waitUntil: 'networkidle' });
|
||||||
test.beforeEach(async ({ context }) => {
|
|
||||||
await context.addInitScript({
|
|
||||||
// eslint-disable-next-line no-undef
|
|
||||||
path: path.join(__dirname, '../../..', './node_modules/sinon/pkg/sinon.js')
|
|
||||||
});
|
});
|
||||||
await context.addInitScript(() => {
|
test.use({
|
||||||
window.__clock = sinon.useFakeTimers(); //Set browser clock to UNIX Epoch
|
clockOptions: {
|
||||||
|
now: 0, //Set browser clock to UNIX Epoch
|
||||||
|
shouldAdvanceTime: false //Don't advance the clock
|
||||||
|
}
|
||||||
});
|
});
|
||||||
});
|
|
||||||
|
|
||||||
test('Visual - Root and About', async ({ page }) => {
|
|
||||||
// Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
|
test('Visual - Root and About', async ({ page, theme }) => {
|
||||||
// Verify that Create button is actionable
|
// Verify that Create button is actionable
|
||||||
const createButtonLocator = page.locator('button:has-text("Create")');
|
await expect(page.locator('button:has-text("Create")')).toBeEnabled();
|
||||||
await expect(createButtonLocator).toBeEnabled();
|
|
||||||
|
|
||||||
// Take a snapshot of the Dashboard
|
// Take a snapshot of the Dashboard
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `Root (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'Root');
|
|
||||||
|
|
||||||
// Click About button
|
// Click About button
|
||||||
await page.click('.l-shell__app-logo');
|
await page.click('.l-shell__app-logo');
|
||||||
|
|
||||||
// Modify the Build information in 'about' to be consistent run-over-run
|
// Modify the Build information in 'about' to be consistent run-over-run
|
||||||
const versionInformationLocator = page.locator('ul.t-info.l-info.s-info');
|
const versionInformationLocator = page.locator('ul.t-info.l-info.s-info').first();
|
||||||
await expect(versionInformationLocator).toBeEnabled();
|
await expect(versionInformationLocator).toBeEnabled();
|
||||||
await versionInformationLocator.evaluate(node => node.innerHTML = '<li>Version: visual-snapshot</li> <li>Build Date: Mon Nov 15 2021 08:07:51 GMT-0800 (Pacific Standard Time)</li> <li>Revision: 93049cdbc6c047697ca204893db9603b864b8c9f</li> <li>Branch: master</li>');
|
await versionInformationLocator.evaluate(node => node.innerHTML = '<li>Version: visual-snapshot</li> <li>Build Date: Mon Nov 15 2021 08:07:51 GMT-0800 (Pacific Standard Time)</li> <li>Revision: 93049cdbc6c047697ca204893db9603b864b8c9f</li> <li>Branch: master</li>');
|
||||||
|
|
||||||
// Take a snapshot of the About modal
|
// Take a snapshot of the About modal
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `About (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'About');
|
});
|
||||||
});
|
|
||||||
|
|
||||||
test('Visual - Default Condition Set', async ({ page }) => {
|
test('Visual - Default Condition Set @unstable', async ({ page, theme }) => {
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
//Click the Create button
|
await createDomainObjectWithDefaults(page, { type: 'Condition Set' });
|
||||||
await page.click('button:has-text("Create")');
|
|
||||||
|
|
||||||
// Click text=Condition Set
|
|
||||||
await page.click('text=Condition Set');
|
|
||||||
|
|
||||||
// Click text=OK
|
|
||||||
await page.click('text=OK');
|
|
||||||
|
|
||||||
// Take a snapshot of the newly created Condition Set object
|
// Take a snapshot of the newly created Condition Set object
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `Default Condition Set (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'Default Condition Set');
|
});
|
||||||
});
|
|
||||||
|
|
||||||
test('Visual - Default Condition Widget', async ({ page }) => {
|
test('Visual - Default Condition Widget @unstable', async ({ page, theme }) => {
|
||||||
//Go to baseURL
|
test.info().annotations.push({
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
type: 'issue',
|
||||||
|
description: 'https://github.com/nasa/openmct/issues/5349'
|
||||||
|
});
|
||||||
|
|
||||||
//Click the Create button
|
await createDomainObjectWithDefaults(page, { type: 'Condition Widget' });
|
||||||
await page.click('button:has-text("Create")');
|
|
||||||
|
|
||||||
// Click text=Condition Widget
|
|
||||||
await page.click('text=Condition Widget');
|
|
||||||
|
|
||||||
// Click text=OK
|
|
||||||
await page.click('text=OK');
|
|
||||||
|
|
||||||
// Take a snapshot of the newly created Condition Widget object
|
// Take a snapshot of the newly created Condition Widget object
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `Default Condition Widget (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'Default Condition Widget');
|
});
|
||||||
});
|
|
||||||
|
|
||||||
test('Visual - Time Conductor start time is less than end time', async ({ page }) => {
|
test('Visual - Time Conductor start time is less than end time', async ({ page, theme }) => {
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
const year = new Date().getFullYear();
|
const year = new Date().getFullYear();
|
||||||
|
|
||||||
let startDate = 'xxxx-01-01 01:00:00.000Z';
|
let startDate = 'xxxx-01-01 01:00:00.000Z';
|
||||||
@ -127,16 +100,14 @@ test('Visual - Time Conductor start time is less than end time', async ({ page }
|
|||||||
await page.locator('input[type="text"]').first().fill(startDate.toString());
|
await page.locator('input[type="text"]').first().fill(startDate.toString());
|
||||||
|
|
||||||
// verify no error msg
|
// verify no error msg
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `Default Time conductor (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'Default Time conductor');
|
|
||||||
|
|
||||||
startDate = (year + 1) + startDate.substring(4);
|
startDate = (year + 1) + startDate.substring(4);
|
||||||
await page.locator('input[type="text"]').first().fill(startDate.toString());
|
await page.locator('input[type="text"]').first().fill(startDate.toString());
|
||||||
await page.locator('input[type="text"]').nth(1).click();
|
await page.locator('input[type="text"]').nth(1).click();
|
||||||
|
|
||||||
// verify error msg for start time (unable to capture snapshot of popup)
|
// verify error msg for start time (unable to capture snapshot of popup)
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `Start time error (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'Start time error');
|
|
||||||
|
|
||||||
startDate = (year - 1) + startDate.substring(4);
|
startDate = (year - 1) + startDate.substring(4);
|
||||||
await page.locator('input[type="text"]').first().fill(startDate.toString());
|
await page.locator('input[type="text"]').first().fill(startDate.toString());
|
||||||
@ -147,27 +118,51 @@ test('Visual - Time Conductor start time is less than end time', async ({ page }
|
|||||||
await page.locator('input[type="text"]').first().click();
|
await page.locator('input[type="text"]').first().click();
|
||||||
|
|
||||||
// verify error msg for end time (unable to capture snapshot of popup)
|
// verify error msg for end time (unable to capture snapshot of popup)
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `End time error (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'End time error');
|
});
|
||||||
});
|
|
||||||
|
|
||||||
test('Visual - Sine Wave Generator Form', async ({ page }) => {
|
|
||||||
//Go to baseURL
|
|
||||||
await page.goto('/', { waitUntil: 'networkidle' });
|
|
||||||
|
|
||||||
|
test('Visual - Sine Wave Generator Form', async ({ page, theme }) => {
|
||||||
//Click the Create button
|
//Click the Create button
|
||||||
await page.click('button:has-text("Create")');
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
// Click text=Sine Wave Generator
|
// Click text=Sine Wave Generator
|
||||||
await page.click('text=Sine Wave Generator');
|
await page.click('text=Sine Wave Generator');
|
||||||
|
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `Default Sine Wave Generator Form (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'Default Sine Wave Generator Form');
|
|
||||||
|
|
||||||
await page.locator('.field.control.l-input-sm input').first().click();
|
await page.locator('.field.control.l-input-sm input').first().click();
|
||||||
await page.locator('.field.control.l-input-sm input').first().fill('');
|
await page.locator('.field.control.l-input-sm input').first().fill('');
|
||||||
|
|
||||||
// Validate red x mark
|
// Validate red x mark
|
||||||
await page.waitForTimeout(VISUAL_GRACE_PERIOD);
|
await percySnapshot(page, `removed amplitude property value (theme: '${theme}')`);
|
||||||
await percySnapshot(page, 'removed amplitude property value');
|
});
|
||||||
|
|
||||||
|
test('Visual - Save Successful Banner @unstable', async ({ page, theme }) => {
|
||||||
|
await createDomainObjectWithDefaults(page, { type: 'Timer' });
|
||||||
|
|
||||||
|
await page.locator('.c-message-banner__message').hover({ trial: true });
|
||||||
|
await percySnapshot(page, `Banner message shown (theme: '${theme}')`);
|
||||||
|
|
||||||
|
//Wait until Save Banner is gone
|
||||||
|
await page.locator('.c-message-banner__close-button').click();
|
||||||
|
await page.waitForSelector('.c-message-banner__message', { state: 'detached'});
|
||||||
|
await percySnapshot(page, `Banner message gone (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Visual - Display Layout Icon is correct', async ({ page, theme }) => {
|
||||||
|
//Click the Create button
|
||||||
|
await page.click('button:has-text("Create")');
|
||||||
|
|
||||||
|
//Hover on Display Layout option.
|
||||||
|
await page.locator('text=Display Layout').hover();
|
||||||
|
await percySnapshot(page, `Display Layout Create Menu (theme: '${theme}')`);
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
test('Visual - Default Gauge is correct @unstable', async ({ page, theme }) => {
|
||||||
|
await createDomainObjectWithDefaults(page, { type: 'Gauge' });
|
||||||
|
|
||||||
|
// Take a snapshot of the newly created Gauge object
|
||||||
|
await percySnapshot(page, `Default Gauge (theme: '${theme}')`);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
78
e2e/tests/visual/faultManagement.visual.spec.js
Normal file
78
e2e/tests/visual/faultManagement.visual.spec.js
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const path = require('path');
|
||||||
|
const { test } = require('../../pluginFixtures');
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
|
||||||
|
const utils = require('../../helper/faultUtils');
|
||||||
|
|
||||||
|
test.describe('The Fault Management Plugin Visual Test', () => {
|
||||||
|
|
||||||
|
test('icon test', async ({ page, theme }) => {
|
||||||
|
// eslint-disable-next-line no-undef
|
||||||
|
await page.addInitScript({ path: path.join(__dirname, '../../helper/', 'addInitFaultManagementPlugin.js') });
|
||||||
|
await page.goto('./', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
await percySnapshot(page, `Fault Management icon appears in tree (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('fault list and acknowledged faults', async ({ page, theme }) => {
|
||||||
|
await utils.navigateToFaultManagementWithStaticExample(page);
|
||||||
|
|
||||||
|
await percySnapshot(page, `Shows a list of faults in the standard view (theme: '${theme}')`);
|
||||||
|
|
||||||
|
await utils.acknowledgeFault(page, 1);
|
||||||
|
await utils.changeViewTo(page, 'acknowledged');
|
||||||
|
|
||||||
|
await percySnapshot(page, `Acknowledged faults, have a checkmark on the fault icon and appear in the acknowldeged view (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('shelved faults', async ({ page, theme }) => {
|
||||||
|
await utils.navigateToFaultManagementWithStaticExample(page);
|
||||||
|
|
||||||
|
await utils.shelveFault(page, 1);
|
||||||
|
await utils.changeViewTo(page, 'shelved');
|
||||||
|
|
||||||
|
await percySnapshot(page, `Shelved faults appear in the shelved view (theme: '${theme}')`);
|
||||||
|
|
||||||
|
await utils.openFaultRowMenu(page, 1);
|
||||||
|
|
||||||
|
await percySnapshot(page, `Shelved faults have a 3-dot menu with Unshelve option enabled (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('3-dot menu for fault', async ({ page, theme }) => {
|
||||||
|
await utils.navigateToFaultManagementWithStaticExample(page);
|
||||||
|
|
||||||
|
await utils.openFaultRowMenu(page, 1);
|
||||||
|
|
||||||
|
await percySnapshot(page, `Faults have a 3-dot menu with Acknowledge, Shelve and Unshelve (Unshelve is disabled) options (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('ability to acknowledge or shelve', async ({ page, theme }) => {
|
||||||
|
await utils.navigateToFaultManagementWithStaticExample(page);
|
||||||
|
|
||||||
|
await utils.selectFaultItem(page, 1);
|
||||||
|
|
||||||
|
await percySnapshot(page, `Selected faults highlight the ability to Acknowledge or Shelve above the fault list (theme: '${theme}')`);
|
||||||
|
});
|
||||||
|
});
|
51
e2e/tests/visual/notebook.visual.spec.js
Normal file
51
e2e/tests/visual/notebook.visual.spec.js
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
const { test } = require('../../pluginFixtures');
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
const { expandTreePaneItemByName, createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
|
|
||||||
|
test.describe('Visual - Notebook', () => {
|
||||||
|
test('Accepts dropped objects as embeds @unstable', async ({ page, theme, openmctConfig }) => {
|
||||||
|
const { myItemsFolderName } = openmctConfig;
|
||||||
|
await page.goto('./#/browse/mine', { waitUntil: 'networkidle' });
|
||||||
|
|
||||||
|
// Create Notebook
|
||||||
|
const notebook = await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Notebook',
|
||||||
|
name: "Embed Test Notebook"
|
||||||
|
});
|
||||||
|
// Create Overlay Plot
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Overlay Plot',
|
||||||
|
name: "Dropped Overlay Plot"
|
||||||
|
});
|
||||||
|
|
||||||
|
await expandTreePaneItemByName(page, myItemsFolderName);
|
||||||
|
|
||||||
|
await page.goto(notebook.url);
|
||||||
|
await page.dragAndDrop('role=treeitem[name=/Dropped Overlay Plot/]', '.c-notebook__drag-area');
|
||||||
|
|
||||||
|
await percySnapshot(page, `Notebook w/ dropped embed (theme: ${theme})`);
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
84
e2e/tests/visual/search.visual.spec.js
Normal file
84
e2e/tests/visual/search.visual.spec.js
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 search functionality.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const { test, expect } = require('../../pluginFixtures');
|
||||||
|
const { createDomainObjectWithDefaults } = require('../../appActions');
|
||||||
|
|
||||||
|
const percySnapshot = require('@percy/playwright');
|
||||||
|
|
||||||
|
test.describe('Grand Search', () => {
|
||||||
|
test.beforeEach(async ({ page, theme }) => {
|
||||||
|
//Go to baseURL and Hide Tree
|
||||||
|
await page.goto('./#/browse/mine?hideTree=true', { waitUntil: 'networkidle' });
|
||||||
|
});
|
||||||
|
test.use({
|
||||||
|
clockOptions: {
|
||||||
|
now: 0, //Set browser clock to UNIX Epoch
|
||||||
|
shouldAdvanceTime: false //Don't advance the clock
|
||||||
|
}
|
||||||
|
});
|
||||||
|
//This needs to be rewritten to use a non clock or non display layout object
|
||||||
|
test('Can search for objects, and subsequent search dropdown behaves properly @unstable', async ({ page, theme }) => {
|
||||||
|
// await createDomainObjectWithDefaults(page, 'Display Layout');
|
||||||
|
// await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click();
|
||||||
|
// await page.locator('text=Save and Finish Editing').click();
|
||||||
|
const folder1 = 'Folder1';
|
||||||
|
await createDomainObjectWithDefaults(page, {
|
||||||
|
type: 'Folder',
|
||||||
|
name: folder1
|
||||||
|
});
|
||||||
|
|
||||||
|
// Click [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').click();
|
||||||
|
// Fill [aria-label="OpenMCT Search"] input[type="search"]
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] input[type="search"]').fill(folder1);
|
||||||
|
await expect(page.locator('[aria-label="Search Result"]')).toContainText(folder1);
|
||||||
|
await percySnapshot(page, 'Searching for Folder Object');
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click();
|
||||||
|
await page.locator('[aria-label="Unnamed Clock clock result"] >> text=Unnamed Clock').click();
|
||||||
|
await percySnapshot(page, 'Preview for clock should display when editing enabled and search item clicked');
|
||||||
|
|
||||||
|
await page.locator('[aria-label="Close"]').click();
|
||||||
|
await percySnapshot(page, 'Search should still be showing after preview closed');
|
||||||
|
|
||||||
|
await page.locator('text=Snapshot Save and Finish Editing Save and Continue Editing >> button').nth(1).click();
|
||||||
|
|
||||||
|
await page.locator('text=Save and Finish Editing').click();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').click();
|
||||||
|
|
||||||
|
await page.locator('[aria-label="OpenMCT Search"] [aria-label="Search Input"]').fill('Cl');
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
page.waitForNavigation(),
|
||||||
|
page.locator('text=Unnamed Clock').click()
|
||||||
|
]);
|
||||||
|
await percySnapshot(page, `Clicking on search results should navigate to them if not editing (theme: '${theme}')`);
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
33
example/exampleTags/plugin.js
Normal file
33
example/exampleTags/plugin.js
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* Open MCT, Copyright (c) 2014-2022, 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 availableTags from './tags.json';
|
||||||
|
/**
|
||||||
|
* @returns {function} The plugin install function
|
||||||
|
*/
|
||||||
|
export default function exampleTagsPlugin() {
|
||||||
|
return function install(openmct) {
|
||||||
|
Object.keys(availableTags.tags).forEach(tagKey => {
|
||||||
|
const tagDefinition = availableTags.tags[tagKey];
|
||||||
|
openmct.annotation.defineTag(tagKey, tagDefinition);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
}
|
19
example/exampleTags/tags.json
Normal file
19
example/exampleTags/tags.json
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"tags": {
|
||||||
|
"46a62ad1-bb86-4f88-9a17-2a029e12669d": {
|
||||||
|
"label": "Science",
|
||||||
|
"backgroundColor": "#cc0000",
|
||||||
|
"foregroundColor": "#ffffff"
|
||||||
|
},
|
||||||
|
"65f150ef-73b7-409a-b2e8-258cbd8b7323": {
|
||||||
|
"label": "Driving",
|
||||||
|
"backgroundColor": "#ffad32",
|
||||||
|
"foregroundColor": "#333333"
|
||||||
|
},
|
||||||
|
"f156b038-c605-46db-88a6-67cf2489a371": {
|
||||||
|
"label": "Drilling",
|
||||||
|
"backgroundColor": "#b0ac4e",
|
||||||
|
"foregroundColor": "#FFFFFF"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -21,19 +21,56 @@
|
|||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
import EventEmitter from 'EventEmitter';
|
import EventEmitter from 'EventEmitter';
|
||||||
import uuid from 'uuid';
|
import { v4 as uuid } from 'uuid';
|
||||||
import createExampleUser from './exampleUserCreator';
|
import createExampleUser from './exampleUserCreator';
|
||||||
|
|
||||||
|
const STATUSES = [{
|
||||||
|
key: "NO_STATUS",
|
||||||
|
label: "Not set",
|
||||||
|
iconClass: "icon-question-mark",
|
||||||
|
iconClassPoll: "icon-status-poll-question-mark"
|
||||||
|
}, {
|
||||||
|
key: "GO",
|
||||||
|
label: "Go",
|
||||||
|
iconClass: "icon-check",
|
||||||
|
iconClassPoll: "icon-status-poll-question-mark",
|
||||||
|
statusClass: "s-status-ok",
|
||||||
|
statusBgColor: "#33cc33",
|
||||||
|
statusFgColor: "#000"
|
||||||
|
}, {
|
||||||
|
key: "MAYBE",
|
||||||
|
label: "Maybe",
|
||||||
|
iconClass: "icon-alert-triangle",
|
||||||
|
iconClassPoll: "icon-status-poll-question-mark",
|
||||||
|
statusClass: "s-status-warning",
|
||||||
|
statusBgColor: "#ffb66c",
|
||||||
|
statusFgColor: "#000"
|
||||||
|
}, {
|
||||||
|
key: "NO_GO",
|
||||||
|
label: "No go",
|
||||||
|
iconClass: "icon-circle-slash",
|
||||||
|
iconClassPoll: "icon-status-poll-question-mark",
|
||||||
|
statusClass: "s-status-error",
|
||||||
|
statusBgColor: "#9900cc",
|
||||||
|
statusFgColor: "#fff"
|
||||||
|
}];
|
||||||
|
/**
|
||||||
|
* @implements {StatusUserProvider}
|
||||||
|
*/
|
||||||
export default class ExampleUserProvider extends EventEmitter {
|
export default class ExampleUserProvider extends EventEmitter {
|
||||||
constructor(openmct) {
|
constructor(openmct, {defaultStatusRole} = {defaultStatusRole: undefined}) {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
this.openmct = openmct;
|
this.openmct = openmct;
|
||||||
this.user = undefined;
|
this.user = undefined;
|
||||||
this.loggedIn = false;
|
this.loggedIn = false;
|
||||||
this.autoLoginUser = undefined;
|
this.autoLoginUser = undefined;
|
||||||
|
this.status = STATUSES[1];
|
||||||
|
this.pollQuestion = undefined;
|
||||||
|
this.defaultStatusRole = defaultStatusRole;
|
||||||
|
|
||||||
this.ExampleUser = createExampleUser(this.openmct.user.User);
|
this.ExampleUser = createExampleUser(this.openmct.user.User);
|
||||||
|
this.loginPromise = undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
isLoggedIn() {
|
isLoggedIn() {
|
||||||
@ -45,11 +82,19 @@ export default class ExampleUserProvider extends EventEmitter {
|
|||||||
}
|
}
|
||||||
|
|
||||||
getCurrentUser() {
|
getCurrentUser() {
|
||||||
if (this.loggedIn) {
|
if (!this.loginPromise) {
|
||||||
return Promise.resolve(this.user);
|
this.loginPromise = this._login().then(() => this.user);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this._login().then(() => this.user);
|
return this.loginPromise;
|
||||||
|
}
|
||||||
|
|
||||||
|
canProvideStatusForRole() {
|
||||||
|
return Promise.resolve(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
canSetPollQuestion() {
|
||||||
|
return Promise.resolve(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
hasRole(roleId) {
|
hasRole(roleId) {
|
||||||
@ -60,6 +105,55 @@ export default class ExampleUserProvider extends EventEmitter {
|
|||||||
return Promise.resolve(this.user.getRoles().includes(roleId));
|
return Promise.resolve(this.user.getRoles().includes(roleId));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getStatusRoleForCurrentUser() {
|
||||||
|
return Promise.resolve(this.defaultStatusRole);
|
||||||
|
}
|
||||||
|
|
||||||
|
getAllStatusRoles() {
|
||||||
|
return Promise.resolve([this.defaultStatusRole]);
|
||||||
|
}
|
||||||
|
|
||||||
|
getStatusForRole(role) {
|
||||||
|
return Promise.resolve(this.status);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getDefaultStatusForRole(role) {
|
||||||
|
const allRoles = await this.getPossibleStatuses();
|
||||||
|
|
||||||
|
return allRoles?.[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
setStatusForRole(role, status) {
|
||||||
|
this.status = status;
|
||||||
|
this.emit('statusChange', {
|
||||||
|
role,
|
||||||
|
status
|
||||||
|
});
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
getPollQuestion() {
|
||||||
|
return Promise.resolve({
|
||||||
|
question: 'Set "GO" if your position is ready for a boarding action on the Klingon cruiser',
|
||||||
|
timestamp: Date.now()
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
setPollQuestion(pollQuestion) {
|
||||||
|
this.pollQuestion = {
|
||||||
|
question: pollQuestion,
|
||||||
|
timestamp: Date.now()
|
||||||
|
};
|
||||||
|
this.emit("pollQuestionChange", this.pollQuestion);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
getPossibleStatuses() {
|
||||||
|
return Promise.resolve(STATUSES);
|
||||||
|
}
|
||||||
|
|
||||||
_login() {
|
_login() {
|
||||||
const id = uuid();
|
const id = uuid();
|
||||||
|
|
||||||
@ -108,3 +202,6 @@ export default class ExampleUserProvider extends EventEmitter {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
/**
|
||||||
|
* @typedef {import('@/api/user/StatusUserProvider').default} StatusUserProvider
|
||||||
|
*/
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user