From 44eef9ce934a6b49dbae75a0cdd50c11c90bd9a1 Mon Sep 17 00:00:00 2001 From: Nisha Bagdwal Date: Fri, 19 Jul 2024 16:33:32 -0400 Subject: [PATCH] HPCC-31857 : Develop an automated ECL Watch Test Suite Developed a testing framework using Java, Selenium, and TestNG. The framework is initiated by `TestRunner.java`, which loads all test classes listed in `config/TestClasses.java` and executes them sequentially. Each test class contains at least one method annotated with `@Test`, which serves as the entry point for the tests. Test cases cover the Activities and ECL Workunit pages, including tests for text presence, link functionality, sorting order, workunits content, description, and checkbox functionality and testing tab clicks. Added loggers in separate files for error, exception, debug, and detail. Updated YML files to reflect jars and commands for the current testing framework. Additionally, provided comprehensive documentation with UML diagrams and detailed explanations of each method within the classes. Signed-off-by: Nisha Bagdwal --- .github/workflows/build-docker.yml | 6 +- .github/workflows/test-ui-gh_runner.yml | 77 +- esp/src/test-ui/tests/Activities.java | 2 +- esp/src/test-ui/tests/framework/README.md | 65 + .../test-ui/tests/framework/SetupForDev.md | 127 ++ .../test-ui/tests/framework/TestRunner.java | 101 ++ .../tests/framework/config/Config.java | 29 + .../tests/framework/config/TestClasses.java | 22 + .../tests/framework/config/URLConfig.java | 53 + .../documentation/ActivitiesTest.png | Bin 0 -> 14585 bytes .../documentation/BaseTableTest1.png | Bin 0 -> 40540 bytes .../documentation/BaseTableTest2.png | Bin 0 -> 17203 bytes .../tests/framework/documentation/Common1.png | Bin 0 -> 29898 bytes .../tests/framework/documentation/Common2.png | Bin 0 -> 3083 bytes .../tests/framework/documentation/Config.png | Bin 0 -> 37917 bytes .../documentation/CustomFormatter.png | Bin 0 -> 4449 bytes .../documentation/DFULogicalFile.png | Bin 0 -> 13225 bytes .../documentation/DFULogicalFiles.png | Bin 0 -> 3211 bytes .../documentation/DFUQueryResponse.png | Bin 0 -> 11782 bytes .../framework/documentation/DFUQueryRoot.png | Bin 0 -> 3013 bytes .../framework/documentation/DesignDocument.md | 1276 +++++++++++++++++ .../documentation/ECLWorkUnitsTest.png | Bin 0 -> 30354 bytes .../documentation/FilesLogicalFilesTest.png | Bin 0 -> 22657 bytes .../framework/documentation/LoggerHolder.png | Bin 0 -> 2861 bytes .../documentation/NavigationWebElement.png | Bin 0 -> 10506 bytes .../framework/documentation/Relationship.png | Bin 0 -> 41362 bytes .../framework/documentation/TestClass.png | Bin 0 -> 3296 bytes .../framework/documentation/TestClasses.png | Bin 0 -> 2324 bytes .../framework/documentation/TestInjector.png | Bin 0 -> 10050 bytes .../framework/documentation/TestRunner.png | Bin 0 -> 7223 bytes .../framework/documentation/TimeUtils.png | Bin 0 -> 6626 bytes .../framework/documentation/URLConfig1.png | Bin 0 -> 42371 bytes .../framework/documentation/URLConfig2.png | Bin 0 -> 9282 bytes .../framework/documentation/URLMapping.png | Bin 0 -> 9850 bytes .../documentation/WUApplicationValue.png | Bin 0 -> 2919 bytes .../documentation/WUApplicationValues.png | Bin 0 -> 2566 bytes .../framework/documentation/WUECLWorkunit.png | Bin 0 -> 21832 bytes .../documentation/WUQueryResponse.png | Bin 0 -> 12328 bytes .../framework/documentation/WUQueryRoot.png | Bin 0 -> 4085 bytes .../framework/documentation/WUWorkunits.png | Bin 0 -> 4176 bytes .../documentation/WebDriverHolder.png | Bin 0 -> 3135 bytes .../tests/framework/model/DFULogicalFile.java | 382 +++++ .../framework/model/DFULogicalFiles.java | 19 + .../framework/model/DFUQueryResponse.java | 95 ++ .../tests/framework/model/DFUQueryRoot.java | 17 + .../framework/model/NavigationWebElement.java | 14 + .../tests/framework/model/TestClass.java | 16 + .../tests/framework/model/URLMapping.java | 33 + .../framework/model/WUApplicationValue.java | 15 + .../framework/model/WUApplicationValues.java | 11 + .../tests/framework/model/WUECLWorkunit.java | 111 ++ .../framework/model/WUQueryResponse.java | 55 + .../tests/framework/model/WUQueryRoot.java | 12 + .../tests/framework/model/WUWorkunits.java | 15 + .../tests/framework/pages/ActivitiesTest.java | 123 ++ .../tests/framework/pages/BaseTableTest.java | 587 ++++++++ .../framework/pages/ECLWorkUnitsTest.java | 424 ++++++ .../pages/FilesLogicalFilesTest.java | 200 +++ .../tests/framework/utility/Common.java | 206 +++ .../framework/utility/CustomFormatter.java | 16 + .../tests/framework/utility/TimeUtils.java | 63 + 61 files changed, 4141 insertions(+), 31 deletions(-) create mode 100644 esp/src/test-ui/tests/framework/README.md create mode 100644 esp/src/test-ui/tests/framework/SetupForDev.md create mode 100644 esp/src/test-ui/tests/framework/TestRunner.java create mode 100644 esp/src/test-ui/tests/framework/config/Config.java create mode 100644 esp/src/test-ui/tests/framework/config/TestClasses.java create mode 100644 esp/src/test-ui/tests/framework/config/URLConfig.java create mode 100644 esp/src/test-ui/tests/framework/documentation/ActivitiesTest.png create mode 100644 esp/src/test-ui/tests/framework/documentation/BaseTableTest1.png create mode 100644 esp/src/test-ui/tests/framework/documentation/BaseTableTest2.png create mode 100644 esp/src/test-ui/tests/framework/documentation/Common1.png create mode 100644 esp/src/test-ui/tests/framework/documentation/Common2.png create mode 100644 esp/src/test-ui/tests/framework/documentation/Config.png create mode 100644 esp/src/test-ui/tests/framework/documentation/CustomFormatter.png create mode 100644 esp/src/test-ui/tests/framework/documentation/DFULogicalFile.png create mode 100644 esp/src/test-ui/tests/framework/documentation/DFULogicalFiles.png create mode 100644 esp/src/test-ui/tests/framework/documentation/DFUQueryResponse.png create mode 100644 esp/src/test-ui/tests/framework/documentation/DFUQueryRoot.png create mode 100644 esp/src/test-ui/tests/framework/documentation/DesignDocument.md create mode 100644 esp/src/test-ui/tests/framework/documentation/ECLWorkUnitsTest.png create mode 100644 esp/src/test-ui/tests/framework/documentation/FilesLogicalFilesTest.png create mode 100644 esp/src/test-ui/tests/framework/documentation/LoggerHolder.png create mode 100644 esp/src/test-ui/tests/framework/documentation/NavigationWebElement.png create mode 100644 esp/src/test-ui/tests/framework/documentation/Relationship.png create mode 100644 esp/src/test-ui/tests/framework/documentation/TestClass.png create mode 100644 esp/src/test-ui/tests/framework/documentation/TestClasses.png create mode 100644 esp/src/test-ui/tests/framework/documentation/TestInjector.png create mode 100644 esp/src/test-ui/tests/framework/documentation/TestRunner.png create mode 100644 esp/src/test-ui/tests/framework/documentation/TimeUtils.png create mode 100644 esp/src/test-ui/tests/framework/documentation/URLConfig1.png create mode 100644 esp/src/test-ui/tests/framework/documentation/URLConfig2.png create mode 100644 esp/src/test-ui/tests/framework/documentation/URLMapping.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WUApplicationValue.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WUApplicationValues.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WUECLWorkunit.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WUQueryResponse.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WUQueryRoot.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WUWorkunits.png create mode 100644 esp/src/test-ui/tests/framework/documentation/WebDriverHolder.png create mode 100644 esp/src/test-ui/tests/framework/model/DFULogicalFile.java create mode 100644 esp/src/test-ui/tests/framework/model/DFULogicalFiles.java create mode 100644 esp/src/test-ui/tests/framework/model/DFUQueryResponse.java create mode 100644 esp/src/test-ui/tests/framework/model/DFUQueryRoot.java create mode 100644 esp/src/test-ui/tests/framework/model/NavigationWebElement.java create mode 100644 esp/src/test-ui/tests/framework/model/TestClass.java create mode 100644 esp/src/test-ui/tests/framework/model/URLMapping.java create mode 100644 esp/src/test-ui/tests/framework/model/WUApplicationValue.java create mode 100644 esp/src/test-ui/tests/framework/model/WUApplicationValues.java create mode 100644 esp/src/test-ui/tests/framework/model/WUECLWorkunit.java create mode 100644 esp/src/test-ui/tests/framework/model/WUQueryResponse.java create mode 100644 esp/src/test-ui/tests/framework/model/WUQueryRoot.java create mode 100644 esp/src/test-ui/tests/framework/model/WUWorkunits.java create mode 100644 esp/src/test-ui/tests/framework/pages/ActivitiesTest.java create mode 100644 esp/src/test-ui/tests/framework/pages/BaseTableTest.java create mode 100644 esp/src/test-ui/tests/framework/pages/ECLWorkUnitsTest.java create mode 100644 esp/src/test-ui/tests/framework/pages/FilesLogicalFilesTest.java create mode 100644 esp/src/test-ui/tests/framework/utility/Common.java create mode 100644 esp/src/test-ui/tests/framework/utility/CustomFormatter.java create mode 100644 esp/src/test-ui/tests/framework/utility/TimeUtils.java diff --git a/.github/workflows/build-docker.yml b/.github/workflows/build-docker.yml index 7c5090750b6..f4c97ebc7a2 100644 --- a/.github/workflows/build-docker.yml +++ b/.github/workflows/build-docker.yml @@ -176,13 +176,13 @@ jobs: ${{ github.workspace }}/HPCC-Platform/.github/workflows/timeoutcmd if-no-files-found: error - - name: Upload UI Test Files + - name: Upload ECL Watch UI Test Files if: ${{ inputs.upload-package == true }} uses: actions/upload-artifact@v3 with: - name: ${{ inputs.asset-name }}-ui_test-files + name: ${{ inputs.asset-name }}-ecl_watch_ui_tests path: | - ${{ github.workspace }}/HPCC-Platform/esp/src/test-ui/**/* + ${{ github.workspace }}/HPCC-Platform/esp/src/test-ui/tests/**/* if-no-files-found: error - name: Upload Error Logs diff --git a/.github/workflows/test-ui-gh_runner.yml b/.github/workflows/test-ui-gh_runner.yml index ed1fad9bc21..ef58ffb92e5 100644 --- a/.github/workflows/test-ui-gh_runner.yml +++ b/.github/workflows/test-ui-gh_runner.yml @@ -17,7 +17,10 @@ on: type: string description: 'Dependencies' required: false - default: 'bison flex build-essential binutils-dev curl lsb-release libcppunit-dev python3-dev default-jdk r-base-dev r-cran-rcpp r-cran-rinside r-cran-inline libtool autotools-dev automake git cmake xmlstarlet' + default: 'bison flex build-essential binutils-dev curl lsb-release libcppunit-dev python3-dev r-base-dev r-cran-rcpp r-cran-rinside r-cran-inline libtool autotools-dev automake git cmake xmlstarlet' + +env: + uploadArtifact: false jobs: @@ -33,17 +36,17 @@ jobs: - name: Download UI Test Files uses: actions/download-artifact@v3 with: - name: ${{ inputs.asset-name }}-ui_test-files - path: ${{ inputs.asset-name }}-ui_test-files + name: ${{ inputs.asset-name }}-ecl_watch_ui_tests + path: ${{ inputs.asset-name }}-ecl_watch_ui_tests - name: Check ECLWatch UI Test Directory id: check run: | - if [[ ! -d ${{ inputs.asset-name }}-ui_test-files ]] + if [[ ! -d ${{ inputs.asset-name }}-ecl_watch_ui_tests ]] then - echo "ECLWatch UI ${{ inputs.asset-name }}-ui_test-files directory missing." + echo "ECLWatch UI ${{ inputs.asset-name }}-ecl_watch_ui_tests directory missing." else - javaFilesCount=$(find ${{ inputs.asset-name }}-ui_test-files/ -iname '*.java' -type f -print | wc -l ) + javaFilesCount=$(find ${{ inputs.asset-name }}-ecl_watch_ui_tests/ -iname '*.java' -type f -print | wc -l ) echo "Number of test java files is $javaFilesCount" if [[ ${javaFilesCount} -eq 0 ]] then @@ -66,7 +69,6 @@ jobs: unzip \ xvfb \ libxi6 \ - default-jdk \ gdb \ ${{ inputs.dependencies }} @@ -95,7 +97,7 @@ jobs: chmod +x ./${{ inputs.asset-name }}-support-files/* sudo cp ./${{ inputs.asset-name }}-support-files/* /opt/HPCCSystems/bin - chmod +x ./${{ inputs.asset-name }}-ui_test-files/* + chmod +x ./${{ inputs.asset-name }}-ecl_watch_ui_tests/* - name: Start HPCC-Platform shell: "bash" @@ -134,36 +136,57 @@ jobs: if: steps.check.outputs.runtests shell: "bash" run: | + sudo apt remove -y openjdk-11-jdk + sudo apt autoremove -y + sudo apt update + sudo apt install -y openjdk-21-jdk + sudo update-alternatives --install /usr/bin/java java /usr/lib/jvm/java-21-openjdk-amd64/bin/java 1 + sudo update-alternatives --install /usr/bin/javac javac /usr/lib/jvm/java-21-openjdk-amd64/bin/javac 1 + sudo update-alternatives --set java /usr/lib/jvm/java-21-openjdk-amd64/bin/java + sudo update-alternatives --set javac /usr/lib/jvm/java-21-openjdk-amd64/bin/javac + export JAVA_HOME=/usr/lib/jvm/java-21-openjdk-amd64/bin/java + export PATH=$PATH:$JAVA_HOME/bin wget https://dl.google.com/linux/direct/google-chrome-stable_current_amd64.deb sudo apt-get install -y ./google-chrome-stable_current_amd64.deb - wget https://chromedriver.storage.googleapis.com/2.41/chromedriver_linux64.zip - unzip chromedriver_linux64.zip - sudo mv chromedriver /usr/bin/chromedriver + wget https://storage.googleapis.com/chrome-for-testing-public/126.0.6478.126/linux64/chromedriver-linux64.zip + unzip chromedriver-linux64.zip -d chromedriver + sudo mv chromedriver/chromedriver-linux64/chromedriver /usr/bin/chromedriver sudo chown root:root /usr/bin/chromedriver sudo chmod +x /usr/bin/chromedriver - wget https://selenium-release.storage.googleapis.com/3.141/selenium-server-standalone-3.141.59.jar - wget http://www.java2s.com/Code/JarDownload/testng/testng-6.8.7.jar.zip - unzip testng-6.8.7.jar.zip - - - name: Run Tests - timeout-minutes: 10 # generous, current runtime is ~1min, this should be increased if new tests are added + wget https://repo1.maven.org/maven2/org/testng/testng/7.7.1/testng-7.7.1.jar + wget https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-annotations/2.17.0/jackson-annotations-2.17.0.jar + wget https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-core/2.17.0/jackson-core-2.17.0.jar + wget https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-databind/2.17.0/jackson-databind-2.17.0.jar + wget https://repo1.maven.org/maven2/com/beust/jcommander/1.82/jcommander-1.82.jar + wget https://github.com/SeleniumHQ/selenium/releases/download/selenium-4.22.0/selenium-java-4.22.0.zip + wget https://repo1.maven.org/maven2/org/slf4j/slf4j-api/1.7.30/slf4j-api-1.7.30.jar + wget https://repo1.maven.org/maven2/org/slf4j/slf4j-simple/1.7.30/slf4j-simple-1.7.30.jar + unzip selenium-java-4.22.0.zip -d selenium-libs + + - name: Run ECL Watch UI Tests + timeout-minutes: 80 # generous, current runtime is ~38 minutes, this should be increased if new tests are added if: steps.check.outputs.runtests shell: "bash" run: | - export CLASSPATH=".:${{ github.workspace }}/selenium-server-standalone-3.141.59.jar:${{ github.workspace }}/testng-6.8.7.jar" - pushd ${{ inputs.asset-name }}-ui_test-files - ./run.sh tests http://localhost:8010 > eclWatchUiTest.log 2>&1 - retCode=$? - echo "UI test done" - [[ $retCode -ne 0 ]] && exit 1 + export CLASSPATH=".:${{ inputs.asset-name }}-ecl_watch_ui_tests:${{ github.workspace }}/selenium-libs/*:${{ github.workspace }}/testng-7.7.1.jar:${{ github.workspace }}/jackson-annotations-2.17.0.jar:${{ github.workspace }}/jackson-core-2.17.0.jar:${{ github.workspace }}/jackson-databind-2.17.0.jar:${{ github.workspace }}/jcommander-1.82.jar:${{ github.workspace }}/slf4j-api-1.7.30.jar:${{ github.workspace }}/slf4j-simple-1.7.30.jar" + pushd ${{ inputs.asset-name }}-ecl_watch_ui_tests + find . -iname '*.java' -type f -print -exec javac -Xlint:none {} \; + java framework.TestRunner -l detail -p /home/runner/HPCCSystems-regression/log/ + echo "ECL Watch UI test done" + lines=$(wc -l < error_ecl_test.log) + [[ $lines -ne 0 ]] && exit 1 + if [[ -f debug_ecl_test.log || -f detail_ecl_test.log || -f exception_ecl_test.log ]] + then + echo "uploadArtifact=true" >> $GITHUB_ENV + fi popd - - name: eclwatch-ui-test-logs-artifact - if: ${{ failure() || cancelled() }} + - name: Upload ECL Watch UI Test Logs To Artifact + if: ${{ failure() || cancelled() || env.uploadArtifact == 'true' }} uses: actions/upload-artifact@v3 with: - name: ${{ inputs.asset-name }}-ui_test-logs + name: ${{ inputs.asset-name }}-ecl_watch_ui_tests path: | - ${{ inputs.asset-name }}-ui_test-files/eclWatchUiTest.log + ${{ inputs.asset-name }}-ecl_watch_ui_tests/*.log /home/runner/HPCCSystems-regression/log/*.json if-no-files-found: error diff --git a/esp/src/test-ui/tests/Activities.java b/esp/src/test-ui/tests/Activities.java index d160ae8f60c..35686098d3e 100644 --- a/esp/src/test-ui/tests/Activities.java +++ b/esp/src/test-ui/tests/Activities.java @@ -24,7 +24,7 @@ public static void main(String[] args) throws IOException, InterruptedException Capabilities caps = ((RemoteWebDriver) driver).getCapabilities(); String browserName = caps.getBrowserName(); - String browserVersion = caps.getVersion(); + //String browserVersion = caps.getVersion(); // System.out.println(browserName+" "+browserVersion); driver.get(args[0]); diff --git a/esp/src/test-ui/tests/framework/README.md b/esp/src/test-ui/tests/framework/README.md new file mode 100644 index 00000000000..c90ae71b7d6 --- /dev/null +++ b/esp/src/test-ui/tests/framework/README.md @@ -0,0 +1,65 @@ +### Project: An Automated ECL Watch Test Suite + +This project's code begins with the TestRunner.java file. The main method in this class loads all the Java classes +created for writing test cases for specific web pages of the ECL Watch UI and then runs the tests in those classes +sequentially. + +The names of the Java classes that the TestRunner class needs to load should be listed in the config/TestClasses.java +file. ActivitiesTest class should always be the first class to load in TestClasses.java, as it gets URLs for all other web pages. + +Each Java class created to write tests for specific web pages should have at least one method annotated with @Test. The +code for each class starts to run from this method. + +#### Important Note: ChromeDriver Version Compatibility + +If the Chrome browser version updates in the future, it's crucial to ensure that the corresponding ChromeDriver version is also updated. Failure to do so may cause tests to fail due to compatibility issues between the browser and driver. Always verify and update ChromeDriver to the latest version whenever running tests to maintain compatibility and ensure smooth test execution. + +#### CLI Arguments for TestRunner.java + +While running the test suite, you can pass arguments in this way -> "-l log_level -p path". +- "log_level" is of two types "debug" and "detail" +- "debug" means generate error and exception log file with a debug log file. +- "detail" means generate error and exception log file with a detailed debug file. +- If no -l and log_level is passed in the argument, only error and exception log will be generated +- "path" is the path of the folder where the json files are +- The code will log an error if the '-p' and 'path' arguments are not provided, as the JSON folder path is required for the test suite. +- -h in the CLI arguments prints the details of parameter usage to the console + +path could be something like: + +for GitHub Actions -> /home/runner/HPCCSystems-regression/log/ + +for local machine -> C:/Users/{your_working_directory_of_json_files}/ + +So an example of complete CLI arguments would look like this: + +-l detail -p /home/runner/HPCCSystems-regression/log/ + +#### Implementation Steps for URL Management + +- A HashMap (urlMap) is created to store URL mappings in config/URLConfig.java file. This map will use the page name as the key and a URLMapping object as the value. The URLMapping object contains the page name, its URL, and another HashMap for nested pages and tabs. +- A static block is used to initialize the urlMap with the initial URL mapping for the Activities navigation. The URL is retrieved using a method from the Common utility class, which handles the dynamic retrieval of the IP address based on the environment whether it is local or GitHub Actions. +- For each main navigation section, a URLMapping object is created. This object includes the page name and its corresponding URL. Additionally, it contains another HashMap to store URLs for nested tabs and pages. +- Each URLMapping object is stored in the urlMap with the main navigation name as the key. This initial setup in the Activities.java class ensures that each navigation section has its base URL stored and accessible. +- For instance, for any navigation page, each page has multiple tabs, and within those tabs, there are multiple pages and tabs. This structure facilitates easy access to the URL of a particular page. +- Starting from the Activities page, for each main navigation section, the code iterates over its associated tabs. For each tab, a new URLMapping object is created and added to the HashMap within the corresponding URLMapping object of the main navigation section. This creates a tree-like structure, allowing easy access to URLs for both navigation sections and their nested tabs. +- By following these implementation steps, the URLConfig class ensures that all URLs within the application are well-organized and easily accessible through a hierarchical structure. This setup simplifies navigation and URL management within the application, making it easier to handle complex page structures and dynamic URL retrievals. + + +Below are the dependencies used in the project: + +- https://repo1.maven.org/maven2/org/testng/testng/7.7.1/testng-7.7.1.jar +- https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-annotations/2.17.0/jackson-annotations-2.17.0.jar +- https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-core/2.17.0/jackson-core-2.17.0.jar +- https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-databind/2.17.0/jackson-databind-2.17.0.jar +- https://repo1.maven.org/maven2/com/beust/jcommander/1.82/jcommander-1.82.jar +- https://github.com/SeleniumHQ/selenium/releases/download/selenium-4.22.0/selenium-java-4.22.0.zip +- https://repo1.maven.org/maven2/org/slf4j/slf4j-api/1.7.30/slf4j-api-1.7.30.jar +- https://repo1.maven.org/maven2/org/slf4j/slf4j-simple/1.7.30/slf4j-simple-1.7.30.jar + +Notes: +1. Users need to run these tests with regression test suite only. +2. Code should be updated accordingly if selenium server jar updates. +3. ActivitiesTest class should always be the first class to load in TestClasses.java, as it gets URLs for all other pages. +4. For future testing developers, custom class names or attributes defined by UI developers can change frequently during updates or redesigns. However, standard attributes that are part of the HTML specifications (such as id, type, value, href, aria-sort, aria-disabled, etc.) are much more stable. Therefore, it is advisable to use only standard HTML attributes to access web elements. This approach ensures that test cases remain consistent and are less likely to break due to UI changes. +5. Ignore the compiler warnings/errors before the beginning of the test logs. They are because of guava-33.2.1-jre-sources.jar, it seems it is not fully compatible with JRE 21, that is installed on GH Actions. But that does not impact our code in any way, so it is better to just ignore it. \ No newline at end of file diff --git a/esp/src/test-ui/tests/framework/SetupForDev.md b/esp/src/test-ui/tests/framework/SetupForDev.md new file mode 100644 index 00000000000..0ca6b0de58b --- /dev/null +++ b/esp/src/test-ui/tests/framework/SetupForDev.md @@ -0,0 +1,127 @@ +This documentation provides a comprehensive guide to setting up an Ubuntu VM on Oracle VirtualBox, installing the HPCC-Platform, and preparing the environment for testing and development. + +#### Setting Up an Ubuntu VM on Oracle VirtualBox + +**Overview of the installation process on a Windows machine:** + +1. **Download and install Oracle VirtualBox:** + - Visit the Oracle VirtualBox [website](https://www.virtualbox.org/) and download the latest version for Windows. + - Follow the installation instructions to install VirtualBox on your system. + +2. **Download the Ubuntu 22.04 Desktop ISO file:** + - Go to the Ubuntu [downloads page](https://ubuntu.com/download/desktop) and download the Ubuntu 22.04 LTS ISO file. + +3. **Set up a new VM in VirtualBox:** + - Open VirtualBox and click on `New` to create a new virtual machine. + - Name the VM and select the type and version (Linux, Ubuntu 64-bit). + - Configure system settings such as memory size and hard disk (create a virtual hard disk now). + - Assign multiple CPUs to your VM if your host machine has a multi-core processor. A good starting point is to allocate 2-4 CPUs to the VM. + - Allocate memory based on the number of CPUs assigned. A good rule of thumb is to assign at least 1 GB of RAM per CPU. If you allocate 2 CPUs, assign at least 2 GB of RAM. For 4 CPUs, assign at least 4 GB of RAM, and so on. Depending on the tasks you'll be performing on the VM, you might need to allocate more memory. + - Link the ISO file by going to the `Settings` of the VM, navigating to `Storage`, and attaching the ISO file to the optical drive. + - Boot the VM and follow the installation wizard to complete the Ubuntu setup. + +#### Installing HPCC-Platform on the VM + +**After successfully installing the VM, proceed with installing the HPCC-Platform:** + +1. **Download the HPCC-Platform package:** + ```sh + wget https://cdn.hpccsystems.com/releases/CE-Candidate-9.8.2/bin/platform/hpccsystems-platform-community_9.8.2-1jammy_amd64_withsymbols.deb + ``` + +2. **Install the package:** + ```sh + sudo dpkg -i hpccsystems-platform-community_9.8.2-1jammy_amd64_withsymbols.deb + ``` + +3. **Fix missing dependencies:** + ```sh + sudo apt-get install -f + ``` + +4. **Check if the installation is successful:** + ```sh + sudo dpkg -l | grep 'hpccsystems-pl' + ``` +5. **Start HPCC-Platform:** + ```sh + sudo /etc/init.d/hpcc-init start + ``` + +6. **Verify access to ECL Watch:** + - Open a browser on your local machine and go to ```http://{ip_of_your_vm}:8010/``` to check if you can access ECL Watch. + +#### Cloning and Checking Out the HPCC-Platform Repository on VM + +1. **Clone the HPCC-Platform GitHub repository:** + ```sh + git clone https://github.com/hpcc-systems/HPCC-Platform.git + ``` + +2. **Navigate to the repository directory:** + ```sh + cd HPCC-Platform + ``` + +3. **Check out the specific version of HPCC-Platform:** + ```sh + git checkout candidate-9.8.x + ``` + +#### Running Regression Test Setup + +**Navigate to the testing directory and set up regression tests:** + ```sh + cd testing/regress + ./ecl-test setup --preAbort '/opt/HPCCSystems/bin/smoketest-preabort.sh' + ``` + +#### Running Spray Tests + +**Execute the spray tests:** + ```sh + ./ecl-test query --preAbort /opt/HPCCSystems/bin/smoketest-preabort.sh --excludeclass python2,embedded-r,embedded-js,3rdpartyservice,mongodb *spray* + ``` + +#### Generating JSON Files + +**Generate JSON files for workunits, files, and DFU workunits:** + ```sh + curl localhost:8010/WsWorkunits/WUQuery.json | python3 -m json.tool > workunits.json + curl localhost:8010/WsDfu/DFUQuery.json?PageSize=250 | python3 -m json.tool > files.json + curl localhost:8010/FileSpray/GetDFUWorkunits.json | python3 -m json.tool > dfu-workunits.json + ``` + +#### Transferring Files Using WinSCP + +Find your json files in the VM and use the WinSCP tool to transfer files from the VM to your local machine. + +#### Downloading Dependencies + +1. **Download the following dependencies:** + - [TestNG 7.7.1](https://repo1.maven.org/maven2/org/testng/testng/7.7.1/testng-7.7.1.jar) + - [Jackson Annotations 2.17.0](https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-annotations/2.17.0/jackson-annotations-2.17.0.jar) + - [Jackson Core 2.17.0](https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-core/2.17.0/jackson-core-2.17.0.jar) + - [Jackson Databind 2.17.0](https://repo1.maven.org/maven2/com/fasterxml/jackson/core/jackson-databind/2.17.0/jackson-databind-2.17.0.jar) + - [JCommander 1.82](https://repo1.maven.org/maven2/com/beust/jcommander/1.82/jcommander-1.82.jar) + - [Selenium Java 4.22.0](https://github.com/SeleniumHQ/selenium/releases/download/selenium-4.22.0/selenium-java-4.22.0.zip) + - [SLF4J API 1.7.30](https://repo1.maven.org/maven2/org/slf4j/slf4j-api/1.7.30/slf4j-api-1.7.30.jar) + - [SLF4J Simple 1.7.30](https://repo1.maven.org/maven2/org/slf4j/slf4j-simple/1.7.30/slf4j-simple-1.7.30.jar) + +#### Adding Dependencies to Your Java Code + +1. Include the downloaded dependencies in your Java project. +2. Specify the path to your ChromeDriver when creating a `ChromeDriver` object in your code. + +#### Writing Test Cases for ECL Watch + +You are now ready to start writing your test cases for ECL Watch running at ```http://{ip_of_your_vm}:8010/```. + +#### Uninstall HPCC-Platform on VM + +Use below commands + +```sh + cd /opt/HPCCSystems/sbin + sudo ./complete-uninstall.sh -p + ``` diff --git a/esp/src/test-ui/tests/framework/TestRunner.java b/esp/src/test-ui/tests/framework/TestRunner.java new file mode 100644 index 00000000000..b4513a02dc5 --- /dev/null +++ b/esp/src/test-ui/tests/framework/TestRunner.java @@ -0,0 +1,101 @@ +package framework; + +import framework.config.Config; +import framework.config.TestClasses; +import framework.model.TestClass; +import framework.utility.Common; +import org.testng.TestNG; + +import java.util.ArrayList; +import java.util.List; + + +public class TestRunner { + public static void main(String[] args) { + + try { + getCommandLineParameters(args); + Common.initializeLoggerAndDriver(); + + if (Common.driver != null) { + TestNG testng = new TestNG(); + testng.setTestClasses(loadClasses()); + testng.run(); + Common.driver.quit(); + } + + Common.printNumOfErrorsAndExceptions(); + + } catch (Exception e) { + Common.logException("Exception occurred in TestRunner class: " + e.getMessage(), e); + } + } + + // Parses the command-line arguments to set the log level and JSON folder path. + // The method checks for the presence of '-l' (log_level) and '-p' (path) arguments. + // Logs an error if the '-p' argument is not provided, as the JSON path is required. + // path is the path of the folder where the json files are + // log level is of two types "debug" and "detail" + // "debug" means generate error log file with a debug log file. + // "detail" means generate error log file with a detailed debug file. + // if no -l and log level is passed in the argument, only error log will be generated + // -h in the CLI arguments prints the details of parameter usage to the console + + public static void getCommandLineParameters(String[] args) { // -l -p + + String log_level = null; + String path = null; + boolean help = false; + + for (int i = 0; i < args.length; i++) { + if ("-l".equals(args[i]) && i + 1 < args.length) { + log_level = args[++i]; + } else if ("-p".equals(args[i]) && i + 1 < args.length) { + path = args[++i]; + } else if ("-h".equals(args[i])) { + help = true; + } + } + + if (log_level != null) { + Config.LOG_LEVEL = log_level; + } + + if (path != null) { + Config.PATH_FOLDER_JSON = path; + } else { + Common.logError("Error: JSON folder path is required. Use -p to specify the path."); + } + + if (help) { + printParameterUsage(); + } + } + + private static void printParameterUsage() { + System.out.println(""" + Requires CLI arguments: -l -p + + is of two types "debug" and "detail" + "debug" means generate error log file with a debug log file. + "detail" means generate error log file with a detailed debug file. + if no -l and log level is passed in the argument, only error log will be generated + + is the path of the folder where the json files are + -p is a mandatory argument, code logs an error if the '-p' argument is not provided, as the JSON path is required for tests."""); + } + + private static Class[] loadClasses() { + + List> classes = new ArrayList<>(); + for (TestClass testClass : TestClasses.testClassesList) { + try { + classes.add(Class.forName(testClass.getPath())); + } catch (Exception e) { + Common.logException("Failure: Error in loading classes: " + e.getMessage(), e); + } + } + + return classes.toArray(new Class[0]); + } +} diff --git a/esp/src/test-ui/tests/framework/config/Config.java b/esp/src/test-ui/tests/framework/config/Config.java new file mode 100644 index 00000000000..4967875c1cf --- /dev/null +++ b/esp/src/test-ui/tests/framework/config/Config.java @@ -0,0 +1,29 @@ +package framework.config; + +public class Config { + + public static final String LOG_FILE_ERROR = "error_ecl_test.log"; // name of the error log file generated after code run is completed + public static final String LOG_FILE_EXCEPTION = "exception_ecl_test.log"; // name of the exception log file generated after code run is completed + public static final String LOG_FILE_DEBUG = "debug_ecl_test.log"; // name of the debug log file generated after code run is completed + public static final String LOG_FILE_DETAIL = "detail_ecl_test.log"; // name of the detail log file generated after code run is completed + public static final String LOCAL_OS = "Windows"; // name of your local OS, it is used to identify whether the code is running on a local machine or on GitHub Actions. + public static final String LOCAL_USER_PROFILE = "C:\\Users\\{your_username}"; // name of your local user profile, it is used to identify whether the code is running on a local machine or on GitHub Actions. + public static final String PATH_LOCAL_CHROME_DRIVER = "C:/Users/{your_working_directory_for_chromedriver}/chromedriver.exe"; // path of chrome driver on your local machine + public static final String PATH_GH_ACTION_CHROME_DRIVER = "/usr/bin/chromedriver"; // path of chrome driver on GitHub Actions + public static final int MALFORMED_TIME_STRING = -1; // this integer is used to denote any malformed time string that we can get from the UI or JSON file. + public static final int WAIT_TIME_IN_SECONDS = 1; // this is the default wait time that code uses to load any web element on UI. + public static final int WAIT_TIME_THRESHOLD_IN_SECONDS = 20; // This time is used to stop the code from waiting infinitely. If it is unable to find a web element on the UI, the code stops the search after this time logs an error if the element is not found. + public static final String TEST_DESCRIPTION_TEXT = "Testing Description"; // This is the test description that is used to test the description textbox functionality + public static final boolean TEST_DETAIL_PAGE_FIELD_NAMES_ALL = true; // true means the tests for field names on details page will run for all items (whether it is workunits or logical files) and false means it will only run for the first item + public static final boolean TEST_WU_DETAIL_PAGE_DESCRIPTION_ALL = true; // true means the tests for checking the description textbox functionality on details page will run for all workunits and false means it will only run for the first workunit + public static final boolean TEST_WU_DETAIL_PAGE_PROTECTED_ALL = true; // true means the tests for checking the protected checkbox functionality on details page will run for all workunits and false means it will only run for the first workunit + public static final boolean TEST_DETAIL_PAGE_TAB_CLICK_ALL = true; // true means the tests for tab click validity on details page will run for all items (whether it is workunits or logical files) and false means it will only run for the first item + + // these values are set in the beginning in the TestRunner.java file + public static String PATH_FOLDER_JSON = ""; // path of the folder of JSON files, it is passed in the CLI arguments. + public static String LOG_LEVEL = ""; // log level is also passed in the CLI arguments, it could be "debug" or "detail" + + public static final String WORKUNITS_JSON_FILE_NAME = "workunits.json"; // name of the workunits JSON file stored in the above JSON folder path + public static final String DFU_WORKUNITS_JSON_FILE_NAME = "dfu-workunits.json"; // name of the dfu-workunits JSON file stored in the above JSON folder path + public static final String FILES_JSON_FILE_NAME = "files.json"; // name of the files JSON file stored in the above JSON folder path +} diff --git a/esp/src/test-ui/tests/framework/config/TestClasses.java b/esp/src/test-ui/tests/framework/config/TestClasses.java new file mode 100644 index 00000000000..9d24257d7fd --- /dev/null +++ b/esp/src/test-ui/tests/framework/config/TestClasses.java @@ -0,0 +1,22 @@ +package framework.config; + +import framework.model.TestClass; + +import java.util.List; + +public class TestClasses { + + // ActivitiesTest class should always be the first class to load, as it gets URLs for all other pages. + + public static final List testClassesList = List.of( + new TestClass("ActivitiesTest", "framework.pages.ActivitiesTest"), + new TestClass("ECLWorkUnitsTest", "framework.pages.ECLWorkUnitsTest") + // The test class for FilesLogicalFilesTest is commented out because of an existing bug. + // In ECL Watch UI on Logical Files tab, all items do not render on UI even after + // selecting a higher dropdown, and it shows + // empty rows. Because of this issue code is unable to get all the file items from UI + // to go ahead for further testing. Until that issue is fixed, tests cannot check the + // validity of the content. For testing this class, wait for the JIRA to be resolved: HPCC-32297 + //new TestClass("FilesLogicalFilesTest", "framework.pages.FilesLogicalFilesTest") + ); +} diff --git a/esp/src/test-ui/tests/framework/config/URLConfig.java b/esp/src/test-ui/tests/framework/config/URLConfig.java new file mode 100644 index 00000000000..e526b0d65c4 --- /dev/null +++ b/esp/src/test-ui/tests/framework/config/URLConfig.java @@ -0,0 +1,53 @@ +package framework.config; + +import framework.model.URLMapping; +import framework.utility.Common; + +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +public class URLConfig { + + public static final String LOCAL_IP = "http://192.168.0.221:8010/"; + public static final String GITHUB_ACTION_IP = "http://127.0.0.1:8010/"; + + public static final String NAV_ACTIVITIES = "Activities"; + public static final String NAV_ECL = "ECL"; + public static final String NAV_FILES = "Files"; + public static final String NAV_PUBLISHED_QUERIES = "Published Queries"; + public static final String NAV_OPERATIONS = "Operations"; + public static final String TAB_ACTIVITIES_ACTIVITIES = "Activities"; + public static final String TAB_ACTIVITIES_EVENT_SCHEDULER = "Event Scheduler"; + public static final String TAB_ECL_WORKUNITS = "Workunits"; + public static final String TAB_ECL_PLAYGROUND = "Playground"; + public static final String TAB_FILES_LOGICAL_FILES = "Logical Files"; + public static final String TAB_FILES_LANDING_ZONES = "Landing Zones"; + public static final String TAB_FILES_WORKUNITS = "Workunits"; + public static final String TAB_FILES_XREF = "XRef (L)"; + public static final String TAB_PUBLISHED_QUERIES_QUERIES = "Queries"; + public static final String TAB_PUBLISHED_QUERIES_PACKAGE_MAPS = "Package Maps"; + public static final String TAB_OPERATIONS_TOPOLOGY = "Topology (L)"; + public static final String TAB_OPERATIONS_DISK_USAGE = "Disk Usage (L)"; + public static final String TAB_OPERATIONS_TARGET_CLUSTERS = "Target Clusters (L)"; + public static final String TAB_OPERATIONS_CLUSTER_PROCESSES = "Cluster Processes (L)"; + public static final String TAB_OPERATIONS_SYSTEM_SERVERS = "System Servers (L)"; + public static final String TAB_OPERATIONS_SECURITY = "Security (L)"; + public static final String TAB_OPERATIONS_DYNAMIC_ESDL = "Dynamic ESDL (L)"; + + public static final String[] navNamesArray = {NAV_ACTIVITIES, NAV_ECL, NAV_FILES, NAV_PUBLISHED_QUERIES, NAV_OPERATIONS}; + public static final Map> tabsListMap = Map.of( + NAV_ACTIVITIES, List.of(TAB_ACTIVITIES_ACTIVITIES, TAB_ACTIVITIES_EVENT_SCHEDULER), + NAV_ECL, List.of(TAB_ECL_WORKUNITS, TAB_ECL_PLAYGROUND), + NAV_FILES, List.of(TAB_FILES_LOGICAL_FILES, TAB_FILES_LANDING_ZONES, TAB_FILES_WORKUNITS, TAB_FILES_XREF), + NAV_PUBLISHED_QUERIES, List.of(TAB_PUBLISHED_QUERIES_QUERIES, TAB_PUBLISHED_QUERIES_PACKAGE_MAPS), + NAV_OPERATIONS, List.of(TAB_OPERATIONS_TOPOLOGY, TAB_OPERATIONS_DISK_USAGE, TAB_OPERATIONS_TARGET_CLUSTERS, + TAB_OPERATIONS_CLUSTER_PROCESSES, TAB_OPERATIONS_SYSTEM_SERVERS, TAB_OPERATIONS_SECURITY, TAB_OPERATIONS_DYNAMIC_ESDL) + ); + + public static final HashMap urlMap = new HashMap<>(); + + static { + urlMap.put(NAV_ACTIVITIES, new URLMapping(NAV_ACTIVITIES, Common.getIP())); + } +} diff --git a/esp/src/test-ui/tests/framework/documentation/ActivitiesTest.png b/esp/src/test-ui/tests/framework/documentation/ActivitiesTest.png new file mode 100644 index 0000000000000000000000000000000000000000..21131f104bc510429f53d1f42c42cb87ab6c6652 GIT binary patch literal 14585 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 alIHpOy6)V!Gf_*S?;iY(;MX7h?*9NK(EAPm literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/BaseTableTest1.png b/esp/src/test-ui/tests/framework/documentation/BaseTableTest1.png new file mode 100644 index 0000000000000000000000000000000000000000..e2d9d1f61110345b2f659d69ba1abdbf3e3629f3 GIT binary patch literal 40540 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/BaseTableTest2.png b/esp/src/test-ui/tests/framework/documentation/BaseTableTest2.png new file mode 100644 index 0000000000000000000000000000000000000000..fb247a555736e9236a0d2427cbae88f4a39567f7 GIT binary patch literal 17203 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 oeP7tZ;$QU5&3L9ay3$-vCd;kCd literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/Common1.png b/esp/src/test-ui/tests/framework/documentation/Common1.png new file mode 100644 index 0000000000000000000000000000000000000000..6615ae3ab0002e4f294c8bedd69b948dd73c494f GIT binary patch literal 29898 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

CLP8w&V-e^FLtlIAbo#>yyWp7v$x? zwyn==JDIW3RV^236lg3C@nEee%g|(+-31COh1Sw`9x_qHr$i80O%VnlAxwEw{4spxf+;PECo70ThFjW0Vli1L9c$p!(+f^nzP4yxsIubyjw)gj;Kz74rp+ zW2xI337C#h`{mJwO4a?rQK<=5+YdHh`jTg`EB-r)prWS`o9E3Fv6FXnqA zahl12#}zRkLzp_dQzOstP8Bus^tyVA<4Loaw-`LVbT0kgF24P$pRTo WZ}Qx;ZVn;vclMBXOLzYHum1s64%Q|B literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/Common2.png b/esp/src/test-ui/tests/framework/documentation/Common2.png new file mode 100644 index 0000000000000000000000000000000000000000..665896f727cfdeb19414e0049394894f53faeaa2 GIT binary patch literal 3083 zcmcJRYgiLk8plU%t<}X|cq+9iQn#%rVy&bE1rpW@qAh|*CMhn07m{=Vr7I91fyCbM zK@_SGG+>enPa#Y~BFQ2PO6b-W3*=J7Ax4Z*X$T~dl!PP@a@mP)FT4At`(eM#nKSdA z^MB8K-t&7;<=*7PfaR|&2LK>o_pbPT0I(z)c_%Jgiah1$x7z_=#m(LE+wmotkJMw; z<8QA_820*j+>h5R`*7>Ypn<<07C06^Rad&}2KDA{la9IA=k?g$M8*OopLJvnIKmH| z3s{8ozTVt1>~BAM72}^DsJ_1Q8UN?M-k$V#x2{A5``a6bnwKEW{jGV0{_d+M&#Xk6 z6XTbE89cXa;kDkWf|-`w1H;|o1~qk zaKv@Br|nF&R+v@4K^A31mGQFfO-p;j9W0+IIV5M&wBIJ5+n@2%h4up#@3e*pq=8x!eRLQGgh1NO# zNG^QY&}ILN@uF;^Mse(bUgJslcO%xqBH$p=fx>>a8#5aZ+g*1<=;>V2d!u5Bbx)us zL)r%(ZOK8iEE!|Z!+Yw8;`39=HXMjf$7Ih(O;fI*rQb{&L~6DwmnpcD1e3$W(`yHt zdMd3(dxUly+?um)s3N@YcpLzjAqMlCH&FWz1%1(;GGkCEL*PC26!tm0APJ7up+m-W z5GJjYJqRk&un?xo%D*?v0m~t>B=37`xWFu*md6g1c^qAk0)MpSIB3JyK#HbVsJM^< z@*s9NN)kH+0Dm8v^9(RPMRaJWqRozjl9vV|-ZT-}P^pZ9yMnA`!b04X0>{G5Inwnt z#$au_&QOE(oxXJ{JDo(KQYLm!n7aICx)q(Vztt$`#W~#QeH+h_0O0Od?sDtXwE?|n z`$Cx44i{mb2;8W?YUdNCk{qN zEKPS;gTN|hr2_!d3556HL-6&S6ga6ap(styhBd=F&VY+59aNf?^5t}Lac?z8&5MH?`8a-f3=-n? zRmPS@yfNx6zsn82=7+?7CzV_w@Lks7F1I5n*KgGbl03<^Z63ykgnM-*kXYwUhH-ToEh+C4< zb0uY~p5dlvJ6YPSdhLTTv3lm)+&d3+3u)6U09m_hWk7A-q?m4%mRdV|zcZpU9{DOl zGe<%%ev~zgt;-tS?KppN_SNK`AYkruYu89n5@uG7o!7+9-??|#zFOp9UJn#c1>R$1 z!FEb%z30tP?8(~crvb);l%s`avPY6M)0?Whzu>kKq)>};yoeU-dxcXF?}1t*+STw8 z6T zOh8rjqA6&`XvjtmcP)+VEoqTV#=h5TIK|5CK5TcVGW`R!Jjf#dL#p_)|V7|49lxPn|vhj!IS5M@XpNh+;dilo?pw$VpNYF$AHuXRo4HO|5Mi{GyFhh@lNF_sS zkPqQQQn`TEoB@N1Y|UQ%j#fiEL>C+q9>{l&W+SikNVpWg9IZTqgr>%}XoXicvBXr}pJ#_U0ZbM5}KehU90N{sMj!28Oy zA$0G>955Q2Uyr0~0+pOpj4`FLAq7-6IeTq8H3Vn7!5u_uh+n$w7KHE^;MeK4$PP*_ zMOmHWHAJD3>C%3r?1kmw`euO z6cU6HSTBv(mTJo9i2l3)6X{ ztq~wvOVLerKb%&5JfO5ZiBwV{6CzWQ;F1@;=E{tCQm&f?_Ct|l<>mfT{l|1s1c+*K zeHn2zh!p_I&`12V%`0AKtTKK&2pKR!vcKH%MWlK=i9I!&FujaVI;?YNFuUm7? zl4`1*jus%Fp#VKinV-H8v)AH0DsU+bnnD zJ;0ZE9SRA;gUoz}BStMl7qqq>NcnM-`dJ|#rv|HK97;EO2~d2f+4kx6@*g17CN6x)j78(0%^($2A~)bb2xHm8*JLZQe3J*BO=n zb^ZL`aF1C3_U`(BAABAY9lq+xQdU*je|)Jj&@WlC;Xm%&-W9_PU}>qho9h3)J0Uq< J^zOk={t0N>FHisg literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/Config.png b/esp/src/test-ui/tests/framework/documentation/Config.png new file mode 100644 index 0000000000000000000000000000000000000000..019311a89aad8e7e550c4692b69a771e0db1cbf2 GIT binary patch literal 37917 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/CustomFormatter.png b/esp/src/test-ui/tests/framework/documentation/CustomFormatter.png new file mode 100644 index 0000000000000000000000000000000000000000..bd5f7b8c3b6b412daa2be6dda56c20597e365b0c GIT binary patch literal 4449 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/DFULogicalFile.png b/esp/src/test-ui/tests/framework/documentation/DFULogicalFile.png new file mode 100644 index 0000000000000000000000000000000000000000..5d4819a1f079b9414c539ffc72c878ad709a92c5 GIT binary patch literal 13225 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/DFULogicalFiles.png b/esp/src/test-ui/tests/framework/documentation/DFULogicalFiles.png new file mode 100644 index 0000000000000000000000000000000000000000..7c642af861a7157289bbe85c58f14dac9d55005f GIT binary patch literal 3211 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/DesignDocument.md b/esp/src/test-ui/tests/framework/documentation/DesignDocument.md new file mode 100644 index 00000000000..d893354de68 --- /dev/null +++ b/esp/src/test-ui/tests/framework/documentation/DesignDocument.md @@ -0,0 +1,1276 @@ +## The Class Design and Code Flow for ECL Watch Test Suite + +### TestRunner.java + +![img.png](TestRunner.png) + +The TestRunner class is responsible for setting up and executing automated tests using the TestNG framework in a +Selenium-based testing environment. It handles the initialization of logging and WebDriver, dynamically loads the +test classes to be executed, and runs the tests. + +Variables: +args: Command line arguments passed to the main method. While running the test suite, you can pass arguments in this way -> "-l log_level -p path". +- "log_level" is of two types "debug" and "detail" +- "debug" means generate error and exception log file with a debug log file. +- "detail" means generate error and exception log file with a detailed debug file. +- If no -l and log_level is passed in the argument, only error and exception log will be generated +- "path" is the path of the folder where the json files are +- The code will log an error if the '-p' and 'path' arguments are not provided, as the JSON folder path is required for the test suite. +- -h in the CLI arguments prints the details of parameter usage to the console + +path could be something like: + +for GitHub Actions -> /home/runner/HPCCSystems-regression/log/ + +for local machine -> C:/Users/{your_working_directory_of_json_files}/ + +So an example of complete CLI arguments would look like this: + +-l detail -p /home/runner/HPCCSystems-regression/log/ + +**Methods:** + +1. main Method + +The entry point of the application. It performs the following steps: + +- Calls getCommandLineParameters method to get the log level and the path of JSON files from CLI arguments and configure them in the Config file. +- Calls Common.initializeLoggerAndDriver to configure the logging system and web driver. +- If the drives sets up properly and is not null, it creates an instance of TestNG. +- Sets the test classes to be run by calling loadClasses. +- Runs the tests using TestNG. +- Quits the WebDriver session after the tests have completed. +- Calls Common.printNumOfErrorsAndExceptions to print total number of errors and exceptions to the console, so that we get an idea of the count after the tests are finished. + +2. loadClasses Method + +Loads the test classes specified in TestClasses.testClassesList: + +- Iterates over the testClassesList and loads each class by its fully qualified name. +- Adds the loaded class to the list classes. +- Catches and prints any ClassNotFoundException. + +3. getCommandLineParameters Method + +This method is designed to parse command-line arguments to set the log level and the path to a folder containing JSON files. It validates the presence of these arguments and sets configuration values accordingly. + +- It takes a single parameter, args, which is an array of strings representing the command-line arguments passed to the program. +- Three local variables, log_level, path and help, are initialized to null. These will hold the values extracted from the command-line arguments. +- A for loop iterates through the args array to examine each argument. +- The loop uses i as the index variable to traverse the array. +- Inside the loop, check if the current argument equals "-l". +- If "-l" is found and there is another argument following it (i + 1 < args.length), assign the value of the next argument to log_level and increment the index i to skip the processed argument. +- Similarly, check if the current argument equals "-p". +- If "-p" is found and there is another argument following it (i + 1 < args.length), assign the value of the next argument to path and increment the index i to skip the processed argument. +- If "-h" is found, assign true to help. +- After the loop, check if log_level is not null. If it is not null, assign its value to Config.LOG_LEVEL. +- Check if path is not null. If it is not null, assign its value to Config.PATH_FOLDER_JSON. +- If path is still null after the loop, log an error message indicating that the JSON folder path is required. Use Common.logError to log this error. +- Check if help is true. If help is true call printParameterUsage method to print the details of parameter usage to the console. + +4. printParameterUsage Method + +This method just print the details of parameter usage to the console for better user interaction. + +### TestClasses.java + +![img_4.png](TestClasses.png) + +The TestClasses class in the framework.config package is responsible for maintaining +a list of test classes used in the framework. This class provides a centralized and +immutable collection of test class metadata, which includes the name of each test +class and its fully qualified class name. This setup helps organize and reference +the test classes easily throughout the testing framework. By using the TestClasses +class, the framework can dynamically load and execute tests, enhancing modularity +and maintainability. + +### TestClass.java + +![img_3.png](TestClass.png) + +The TestClass class in the framework.model package is a simple model class designed +to encapsulate metadata about a test class within the testing framework. It contains +the name of the test class and its fully qualified class name (path). This class +provides a structured way to store and retrieve information about each test class, +which can be utilized by other components in the framework for dynamically loading and +executing tests. + +### URLMapping.java + +![img.png](URLMapping.png) + +The URLMapping class in the framework.model package is a model that represents a URL +and its associated metadata. It provides a structured way to store URLs and their hierarchical +relationships, facilitating easy navigation and retrieval of URLs in a web application. + +#### Variables: +- name: A String representing the name of the navigation element or tab. +- url: A String representing the URL associated with the navigation element or tab. +- urlMappings: A HashMap that maps the names of nested navigation elements or tabs to their corresponding URLMapping objects. + +### Config.java + +![img.png](Config.png) + +The Config class in the framework.config package serves as a centralized configuration +repository for the application. It contains various constants that are used throughout +the framework, providing a single point of reference for configuration settings such as +file paths, file names, flags, and other constants. This approach enhances the maintainability and +readability of the code by avoiding hard-coded values scattered across different classes. + +### URLConfig.java + +![img.png](URLConfig1.png) +![img.png](URLConfig2.png) + +The URLConfig class is designed to manage the URL configurations for different navigation +and tab pages within the test cases. It provides a structured way to store and retrieve +URLs associated with various sections of the ECL Watch UI. This class uses constants to define +URLs and page names, and it leverages a HashMap to store URL mappings, facilitating easy access +to the URLs of different pages and tabs. + +- NAV_ACTIVITIES, NAV_ECL, NAV_FILES, NAV_PUBLISHED_QUERIES, NAV_OPERATIONS: Constants representing main navigation sections. +- TAB_ACTIVITIES_ACTIVITIES, TAB_ACTIVITIES_EVENT_SCHEDULER: Tab names under the Activities navigation. +- TAB_ECL_WORKUNITS, TAB_ECL_PLAYGROUND: Tab names under the ECL navigation. +- TAB_FILES_LOGICAL_FILES, TAB_FILES_LANDING_ZONES, TAB_FILES_WORKUNITS, TAB_FILES_XREF: Tab names under the Files navigation. +- TAB_PUBLISHED_QUERIES_QUERIES, TAB_PUBLISHED_QUERIES_PACKAGE_MAPS: Tab names under the Published Queries navigation. +- TAB_OPERATIONS_TOPOLOGY, TAB_OPERATIONS_DISK_USAGE, TAB_OPERATIONS_TARGET_CLUSTERS, TAB_OPERATIONS_CLUSTER_PROCESSES, TAB_OPERATIONS_SYSTEM_SERVERS, TAB_OPERATIONS_SECURITY, TAB_OPERATIONS_DYNAMIC_ESDL: Tab names under the Operations navigation. +- navNamesArray: Array containing all the main navigation names. +- tabsListMap: Map containing lists of tab names associated with each main navigation section. +- urlMap: A HashMap (urlMap) is created to store URL mappings. This map will use the page name as the key and a URLMapping object as the value. The URLMapping object contains the page name, its URL, and another HashMap for nested pages and tabs. +- A static block is used to initialize the urlMap with the initial URL mapping for the Activities navigation. The URL is retrieved using a method from the Common utility class, which handles the dynamic retrieval of the IP address based on the environment whether it is local or GitHub Actions. + +#### Implementation Steps for URL Management + +- For each main navigation section, a URLMapping object is created. This object includes the page name and its corresponding URL. Additionally, it contains another HashMap to store URLs for nested tabs and pages. +- Each URLMapping object is stored in the urlMap with the main navigation name as the key. This initial setup ensures that each navigation section has its base URL stored and accessible. +- For instance, for any navigation page, each page has multiple tabs, and within those tabs, there are multiple pages and tabs. This structure facilitates easy access to the URL of a particular page. +- Starting from the Activities page, for each main navigation section, the code iterates over its associated tabs (as defined in tabsListMap). For each tab, a new URLMapping object is created and added to the HashMap within the corresponding URLMapping object of the main navigation section. This creates a tree-like structure, allowing easy access to URLs for both navigation sections and their nested tabs. +- By following these implementation steps, the URLConfig class ensures that all URLs within the application are well-organized and easily accessible through a hierarchical structure. This setup simplifies navigation and URL management within the application, making it easier to handle complex page structures and dynamic URL retrievals. + +### Common.java + +![img.png](Common1.png) +![img.png](Common2.png) + +The Common class in the framework.utility package provides a set of utility +methods that are frequently used throughout the testing framework. These +methods handle common tasks and leverages constants from the Config class to maintain consistency +and facilitate configuration management. + +Variables: + +- driver: This public static variable stores the WebDriver instance used for interacting with the web browser. +- errorLogger: This public static variable stores a logger instance used for logging error messages. +- specificLogger: This public static variable stores a logger instance used for logging specific messages +based on the provided level (debug or detail). +- num_errors: A counter to keep a count of total numbers of errors generated in the error log file. +- num_exceptions: A counter to keep a count of total numbers of exceptions generated in the exceptions log file. + +**Methods:** + +1. checkTextPresent Method + +Checks if the specified text is present on the current webpage and logs the result. + +- Retrieves the page source using driver.getPageSource(). +- Checks if the page source contains the specified text. +- Logs a success message if the text is found; otherwise, logs an error message and records it using the provided + logger. + +2. openWebPage Method + +Opens the specified URL in the browser and maximizes the window. + +- Navigates to the specified URL using driver.get(url). +- Maximizes the browser window using driver.manage().window().maximize(). +- Calls the sleep method to pause the execution for a short period to allow the page to load completely. + +3. sleep Method + +The sleep method pauses the execution of the program for a specified duration +(4 seconds in this case). This can be useful in scenarios where a delay is required, +such as waiting for a webpage to load completely before proceeding with further actions. + +4. isRunningOnLocal Method + +Determines if the code is running on a local environment. + +- Checks if the operating system name starts with the value of Config.LOCAL_OS. +- Checks if the user profile path starts with the value of Config.LOCAL_USER_PROFILE. +- Returns true if both conditions are met, indicating a local environment; otherwise, returns false. + +5. getIP Method + +The getIP method determines the appropriate IP address to use based on whether the application +is running in a local environment or in a GitHub Actions environment. +This helps in dynamically adjusting the base URL of the application depending on the +execution context. + +- Calls isRunningOnLocal to check the environment. +- If running locally, returns the URLConfig.LOCAL_IP. +- If running in GitHub Actions, returns the URLConfig.GITHUB_ACTION_IP. + +6. waitForElement Method + +- This method waits for a web element to be present in the DOM. +- Uses WebDriverWait to wait up to 10 seconds for the presence of the specified web element. + +7. logError Method + +- This method logs error messages. +- Prints the error message to the standard error stream. +- Logs the message using errorLogger. +- Increments the variable num_errors by 1, to maintain the count of number of errors generated. + +8. logException Method + +- This method logs exception messages with complete stack trace. +- Prints the exception message with complete stack trace to the standard error stream. +- Logs the message using errorLogger. +- Increments the variable num_exceptions by 1, to maintain the count of number of num_exceptions generated. + +9. logDebug Method + +- This method logs debug or detailed messages. +- Prints the message to the standard output stream. +- Logs the message using specificLogger if the logging level is INFO or FINE. + +10. logDetail method + +- This method logs detailed messages. +- Prints the message to the standard output stream. +- Logs the message using specificLogger if the logging level is FINE. + +11. initializeLoggerAndDriver Method + +- This method initializes the logger and WebDriver instances. +- Sets up the specificLogger based on the provided argument. +- Sets up the WebDriver instance using setupWebDriver() method. + +12. setupWebDriver Method + +- This method sets up the WebDriver instance based on the environment. +- Configures ChromeOptions for headless mode, no sandbox, and suppressed log output. +- Sets up the WebDriver based on the environment (local or GitHub Actions). +- Logs an error message if an exception occurs during setup. + +13. setupLogger Method + +- This method sets up a logger instance based on the provided log level (error, exception, debug and detail). +- It creates a CustomFormatter instance. This formatter will be used to format log messages according to a specified pattern. +- Configures the logger to disable console logging and set up file handlers for different log levels (error, exception, debug, detail). +- For "error" and "exception" levels, it sets up a FileHandler with a corresponding file path from Config.LOG_FILE_ERROR or Config.LOG_FILE_EXCEPTION, respectively. The log level is set to Level.SEVERE. +- For "debug", it sets up a FileHandler with Config.LOG_FILE_DEBUG and sets the log level to Level.INFO. +- For "detail", it sets up a FileHandler with Config.LOG_FILE_DETAIL and sets the log level to Level.FINE. +- Turns off all logging from Selenium WebDriver. +- Logs an error message if an exception occurs during logger setup. +- Finally, the method returns the configured Logger instance. + +14. sleepWithTime Method + +- The sleepWithTime method pauses the execution of the current thread for a specified number of seconds. +- It is a simple utility method used to introduce a delay in the execution flow. +- Used to wait for a certain condition or state before proceeding with further execution. + +15. waitForElementToBeClickable Method + +- This method waits until the specified web element is clickable. +- A WebDriverWait instance is created with a timeout duration specified by Config.WAIT_TIME_THRESHOLD_IN_SECONDS. +- The method waits until the element is clickable, using Selenium's ExpectedConditions.elementToBeClickable condition. +- It is useful in scenarios where an element needs to be interacted with, but it might not be immediately clickable due to loading times or other conditions. + +16. waitForElementToBeDisabled Method + +The waitForElementToBeDisabled method waits until the specified web element is disabled by checking its aria-disabled attribute. +The attribute "aria-disabled" is part of the Accessible Rich Internet Applications (ARIA) specification, which is used to improve the accessibility of web pages. +aria-disabled is a standard attribute and is widely used in HTML to indicate whether an element is disabled. It is system-defined, meaning it is part of the standard +HTML specifications and not a custom class or attribute that might change frequently. By using aria-disabled, you ensure that the check for the disabled state is +consistent and less likely to break due to UI changes. Custom class names or attributes defined by developers can change frequently during updates or redesigns, +but standard attributes like aria-disabled are much more stable. + +- A WebDriverWait instance is created with a timeout duration specified by Config.WAIT_TIME_THRESHOLD_IN_SECONDS. +- The method waits until the aria-disabled attribute of the element contains the value "true", indicating that the element is disabled. +- Ensures that an element has transitioned to a disabled state before proceeding with further actions. + +### CustomFormatter.java + +![img.png](CustomFormatter.png) + +The CustomFormatter class in the framework.utility package extends Formatter and uses a SimpleDateFormat instance to define the format of the timestamps in the log messages. The format specified is "yyyy-MM-dd HH:mm:ss", which outputs timestamps in a year-month-day hour:minute:seconds +format. + +1. format Method + +- The format method overrides the abstract method from the Formatter class to define how log records should be formatted. +- It takes a LogRecord object as a parameter, which contains information about the log message, including the timestamp and the message itself. +- The method converts the timestamp from milliseconds since the epoch (obtained from record.getMillis()) into a Date object. +- It then formats this Date object using the SimpleDateFormat instance to get a readable date-time string. +- Finally, it returns a formatted string consisting of the formatted date-time followed by the log message and a newline character. This ensures that each log entry starts with a timestamp and is followed by the log message. + +### TimeUtils.java + +![img.png](TimeUtils.png) + +The TimeUtils class in the framework.utility package provides utility functions to handle +time strings and convert them into milliseconds. This is useful for standardizing time +representations and performing time-based calculations in a consistent manner. + +TIME_PATTERN: A regular expression pattern used to match various time formats. The supported formats are: + +- d days h:m:s.s +- h:m:s.s +- m:s.s +- s.s + +**Methods:** + +1. convertToMilliseconds Method + +This method converts a time string into milliseconds based on the matched pattern. If the +time string does not match any recognized format, it returns a predefined constant for +malformed time strings. + +- The method first attempts to match the input time string against the TIME_PATTERN. +- If the string matches the pattern, it initializes the time components (days, hours, minutes, seconds, milliseconds) to + zero. +- The method then extracts values based on the matching groups. +- If any parsing errors occur (e.g., NumberFormatException) or if the string does not match the pattern, the method + returns Config.MALFORMED_TIME_STRING. +- If the parsing is successful, the method calculates the total duration in milliseconds + +### NavigationWebElement + +![img.png](NavigationWebElement.png) + +This NavigationWebElement class in the framework.model package, is a record class used to represent a navigation element +within a web application framework. It offers a concise way to store and manage information about such +elements. + +- The class is defined as a record which is a recent addition to Java that simplifies creating immutable data classes. +- It has three properties: + - name: A String representing the name or identifier of the navigation element in the menu bar(e.g., "Activities", " + ECL", "Files). + - hrefValue: A String representing the href attribute value of the element, which typically specifies the URL linked + to by the element. + - webElement: A WebElement object from the Selenium library. This holds the actual WebElement instance representing + the element in the web page. + +- Due to the record nature, a constructor is not explicitly defined. The compiler generates a constructor that takes + arguments for each property and initializes them. +- The NavigationWebElement class offers a structured way to manage data related to navigation elements in a web + application framework. + +### Java Classes for Representing Workunit JSON Data + +This section details the class structure used to map JSON data file of list of "Workunit" +entities into Java objects. These classes provide a clear representation of the data and +allow for easy access to its values throughout the codebase. This structure is particularly +beneficial for writing test cases, as it simplifies working with the data in a well-defined +format. Below are the UML diagram of the classes used for JSON mapping to java objects for +workunits JSON file. + +So the first java object created to map the json is WUQueryRoot (equivalent to the root object of JSON) +which contains the WUQueryResponse object (equivalent to the WUQueryResponse key in json file) inside it +and WUQueryResponse contains the WUWorkunits object (equivalent to the Workunits key in json file) +inside it, and then finally a list of WUECLWorkunit object (equivalent to the ECLWorkunit key in json file) is inside the WUWorkunits object. +Inside each WUECLWorkunit object, there is WUApplicationValues object (equivalent to the ApplicationValues key in json), which further have a list of +WUApplicationValue objects (equivalent to the ApplicationValue key in json file) inside it. +This is how the structure of Java objects created with respect to the JSON structure of workunits. + +![img.png](WUQueryRoot.png) + +![img.png](WUQueryResponse.png) + +![img.png](WUWorkunits.png) + +![img.png](WUECLWorkunit.png) + +![img.png](WUApplicationValues.png) + +![img.png](WUApplicationValue.png) + +### ActivitiesTest + +![img.png](ActivitiesTest.png) + +This ActivitiesTest class in the framework.pages package, implements a TestNG test (@Test) +for the Activities page of ECL Watch UI. It focuses on verifying the following aspects of the +Activities page: + +- Presence of specific text elements +- Functionality of navigation links and their corresponding sub-tabs + +**Class Variables:** + +- textArray: A static final String array containing expected text elements to be present on the Activities page (e.g., " + Target/Wuid", "Graph"). + +**Methods:** + +1. testActivitiesPage (Test Method) + +- This is the main test method annotated with @Test to be run as a test case. +- Initializes the WebDriver instance. +- Use the Common.openWebPage method to navigate to the URL of the Activities page, retrieved from the urlMap. +- Logs the start of the tests for the "Activities" page. +- Calls testForAllText(driver) to check for the presence of predefined texts. +- Retrieves the navigation web elements by calling getNavWebElements(driver). +- Calls testForNavigationLinks(driver, navWebElements) to test the navigation links. +- Logs the completion of the tests for the "Activities" page. + +2. testForAllText Method + +- This method checks if specific texts are present on the "Activities" page. +- Logs the start of text presence tests. +- Iterates over each text in textArray. +- Calls Common.checkTextPresent(driver, text, "Activities Page") to verify the presence of each text on the page. + +3. testForNavigationLinks Method + +- This method tests each navigation link to ensure they direct to the correct pages with the expected tabs. +- Logs the start of navigation link tests. +- Iterates over each NavigationWebElement in navWebElements. +- Clicks on each navigation element and verifies the presence of corresponding tabs by calling testTabsForNavigationLinks(driver, element). +- Logs success if all tabs are present; otherwise, logs an error with the current page details. +- Catches and logs any exceptions that occur during the process. + +4. getCurrentPage Method + +- This method determines the current page by checking the presence of specific tabs. +- Iterates over each entry in tabsListMap. +- Checks if all tabs for each page are present in the page source. +- Returns the page name if all tabs are present; otherwise, returns "Invalid Page". + +5. testTabsForNavigationLinks Method: + +- This method verifies the presence of tabs for a given navigation element and updates the URL map with the tab URLs. +- Get the list of tabs associated with the navigation element from URLConfig.tabsListMap. +- Loop through each tab in the list. +- Use Common.waitForElement to wait for the tab element to be present in the DOM. +- Add the tab's URL to the urlMap under the corresponding navigation element. +- If a timeout exception occurs, return false. Otherwise, return true. + +6. getNavWebElements Method: + +- This method retrieves the web elements for the main navigation links on the Activities page. +- Creates an empty list to store NavigationWebElement objects. +- Iterates through the URLConfig.navNamesArray. +- For each navigation link name: + - Finds the WebElement using driver.findElement with By.name strategy. + - Extracts the href attribute value. + - Creates a new NavigationWebElement object with the name, href value, and WebElement reference. + - Adds the NavigationWebElement to the list. + - Add the navigation element's URL to the urlMap. + - Log any errors that occur during the process. +- Returns the list of NavigationWebElement objects. + +### BaseTableTest + +![img.png](BaseTableTest1.png) +![img.png](BaseTableTest2.png) + +The BaseTableTest class is designed as a superclass for testing web pages containing tabular data. It is intended for use in automated tests, +particularly for pages like workunits, files, and queries, and includes functionality for testing their respective details pages. +It defines methods for common functionalities like: + +- Verifying the presence of expected text elements on the page. +- Comparing the content displayed in the table with corresponding data from a JSON file. +- Testing the sorting functionality of the table columns. +- Verifying links within the table cells and their navigation behavior. +- Testing common details functionality, such as tab clicks. + +**Abstract Methods:** + +These methods must be implemented by subclasses to provide specific information and functionality for the page being tested: + +1. getPageName(): Returns the name of the page. +2. getPageUrl(): Returns the URL of the page. +3. getJsonFilePath(): Returns the file path for the JSON data file. +4. getSaveButtonDetailsPage(): Returns the text for the save button on the details page. +5. getColumnNames(): Returns an array of column names displayed in the table. +6. getDetailNames(): Returns an array of detail names for the details page. +7. getColumnKeys(): Returns an array of column keys for accessing data. +8. getDetailKeys(): Returns an array of detail keys for accessing data on the details page. +9. getCheckboxTypeForDetailsPage(): Returns the attribute type for checkboxes on the details page. +10. getAttributeTypeForDetailsPage(): Returns the attribute type for elements on the details page. +11. getAttributeValueForDetailsPage(): Returns the attribute value for elements on the details page. +12. getDetailKeysForPageLoad(): Returns an array of detail keys to check when the details page loads. +13. getUniqueKeyName(): Returns the unique key name for identifying data. +14. getUniqueKey(): Returns the unique key used for identifying data in the table. +15. getColumnKeysWithLinks(): Returns an array of column keys that contain links. +16. parseDataUIValue(dataUIValue, dataJSONValue, columnName, dataIDUIValue): Parses the UI data value. +17. parseDataJSONValue(dataJSONValue, columnName, dataIDUIValue): Parses the JSON data value. +18. parseJson(filePath): Parses the JSON file and returns a list of objects. +19. getColumnDataFromJson(object, columnKey): Gets column data from a JSON object. +20. sortJsonUsingSortOrder(currentSortOrder, columnKey): Sorts the JSON data using the specified order and column key. +21. getCurrentPage(): Returns the current page. +22. getJsonMap(): Returns a map of JSON data. +23. getColumnNamesForTabsDetailsPage(): Returns a map of column names for tabs on the details page. +24. getTabValuesForDetailsPage(): Returns an array of tab values for the details page. +25. testDetailSpecificFunctionality(name, i): Tests specific functionality on the details page. + +**Non-Abstract Methods:** + +#### Method: `testPage()` + +This method initiates and executes a series of tests on a web page. + +1. **Open the Web Page**: The method begins by opening the specified web page using `Common.openWebPage(getPageUrl())`. + +2. **Logging**: It logs the start of tests for the page with `Common.logDebug`. + +3. **Test for All Text**: The method `testForAllText()` is called to verify the presence of expected text elements on the page. + +4. **Retrieve JSON Objects**: It retrieves all JSON objects using `getAllObjectsFromJson()` and stores them in `jsonObjects`. + +5. **Dropdown Selection**: If `jsonObjects` is not null, it determines the number of items in JSON (`numOfItemsJSON`) and uses `clickDropdown(numOfItemsJSON)` to select the appropriate dropdown value for displaying items on the page. + +6. **Content and Sorting Tests**: The method `testContentAndSortingOrder()` is called to verify the content and sorting order of items on the page. + +7. **Link Tests**: It calls `testLinksInTable()` to test all the links in the table present on the page. + +8. **Logging**: Logs the completion of tests with `Common.logDebug`. + +9. **Exception Handling**: Catches any exceptions that occur during the test and logs the error with `Common.logError`. + +#### Method: `testDetailsPage(String name, int i)` + +This method tests various aspects of a details page for a specific item. + +1. **Test Detail Page Field Names**: Depending on the configuration `Config.TEST_DETAIL_PAGE_FIELD_NAMES_ALL`, it calls `testForAllTextInDetailsPage(name)` either for all items or just the first item. + +2. **Details Content Page Test**: It calls `testDetailsContentPage(name)` to verify the content of the details page. + +3. **Detail Specific Functionality Test**: The method `testDetailSpecificFunctionality(name, i)` is called to test specific functionality on the details page. + +4. **Test Tab Clicks**: Depending on the configuration `Config.TEST_DETAIL_PAGE_TAB_CLICK_ALL`, it calls `testTabClickOnDetailsPage()` either for all items or just the first item. + +#### Method: `testTabClickOnDetailsPage()` + +This method tests the functionality of clicking through different tabs on the details page. + +1. **Logging**: Logs the start of the tab click test with `Common.logDebug`. + +2. **Wait for Page Load**: Calls `waitToLoadDetailsPage()` to ensure the details page is fully loaded. + +3. **Iterate Through Tabs**: For each tab value retrieved from `getTabValuesForDetailsPage()`: + - It tries to find the tab button element and clicks it using `javaScriptElementClick`. + - It then calls `testPresenceOfColumnNames(getColumnNamesForTabsDetailsPage().get(tabValue), tabValue)` to verify the presence of expected column names for the selected tab. + - If an error occurs, it logs the error with `Common.logError`. + +#### Method: `javaScriptElementClick(WebElement element)` + +This method clicks a web element using JavaScript. + +1. **Execute Script**: Uses a `JavascriptExecutor` to click the specified web element by executing the script `arguments[0].click();`. + +#### Method: `testPresenceOfColumnNames(List columnNames, String tabValue)` + +This method verifies the presence of specific column names within a tab on the details page. + +1. **Check Column Names**: For each column name in the provided list: + - It waits for the element containing the column name using `Common.waitForElement(By.xpath("//*[text()='" + columnName + "']"))`. + - If a `TimeoutException` occurs, it logs an error and sets a flag `allPresent` to false. + +2. **Logging**: If all column names are present, it logs success with `Common.logDetail`. If any column name is missing, it logs an error. + +#### Method: `clickOnSaveButton()` + +This method handles the action of clicking the save button on a details page. + +1. **Wait for Save Button**: It retrieves the save button element using `getSaveButtonWebElementDetailsPage()` and waits for it to be clickable with `Common.waitForElementToBeClickable`. + +2. **Click Save Button**: Clicks the save button. + +3. **Wait for Save Completion**: Waits for the save button to become disabled using `Common.waitForElementToBeDisabled`. + +#### Method: `testLinksInTable()` + +This method tests the functionality and navigation of links present in a table on the page. + +1. **Logging**: Logs the start of the link tests with `Common.logDebug`. + +2. **Page refresh**: refreshing page as page has scrolled to right for testing the sorting functionality of column headers, so bringing it back to normal view by refreshing it + +3. **Iterate Through Columns with Links**: For each column key with links retrieved from `getColumnKeysWithLinks()`: + - It retrieves the data values from the UI using `getDataFromUIUsingColumnKey(columnKey)`. + - For each value, it tries to find and click the corresponding link element. + - After clicking, it checks if the page source contains the name to confirm successful navigation. + - Logs success or failure based on the result. + - It then navigates back to the original page and refreshes it. + - It verifies if the dropdown value remains unchanged after navigation. + +3. **Exception Handling**: Catches any exceptions during the link tests and logs the errors. + +#### Method: `waitToLoadDetailsPage()` + +This method ensures the details page is fully loaded by waiting for specific elements to be visible. + +1. **Initialize Wait Time**: Starts with an initial wait time specified by `Config.WAIT_TIME_IN_SECONDS`. + +2. **Check Element Visibility**: Repeatedly checks the visibility of detail keys retrieved from `getDetailKeysForPageLoad()`: + - Waits for each element by its ID using `Common.waitForElement`. + - If any element's attribute is empty, it continues waiting and increments the wait time. + +3. **Logging**: Logs the total wait time used to load the details page with `Common.logDebug`. + +#### Method: `testDetailsContentPage(String name)` + +This method tests the content of a details page for a specific item. + +1. **Logging**: Logs the start of the content tests with `Common.logDebug`. + +2. **Wait for Page Load**: Calls `waitToLoadDetailsPage()` to ensure the page is fully loaded. + +3. **Compare Content**: For each detail key: + - It retrieves the corresponding web element and its value. + - Compares the value with the expected JSON value. + - Logs errors if there are discrepancies. + +4. **Logging**: Logs success if all values match the expected values. + +#### Method: `testForAllTextInDetailsPage(String name)` + +This method tests the presence of specific text elements on a details page. + +1. **Logging**: Logs the start of the text tests with `Common.logDebug`. + +2. **Check Text Presence**: For each expected text element retrieved from `getDetailNames()`: + - Calls `Common.checkTextPresent` to verify the text presence on the page. + +#### Method: `testContentAndSortingOrder()` + +This method tests the content and sorting order of items on a page. + +1. **Logging**: Logs the start of content tests with `Common.logDebug`. + +2. **Test Table Content**: Calls `testTableContent()` to verify the content of the table. + +3. **Test Sorting Order**: If the content test passes, it logs the start of sorting order tests and: + - Iterates through each column key and name. + - Calls `testTheSortingOrderForOneColumn(columnKey, columnName)` to verify the sorting order. + +#### Method: `testTheSortingOrderForOneColumn(String columnKey, String columnName)` + +This method tests the sorting order of a specific column. + +1. **Retrieve Sorting Order**: Calls `getCurrentSortingOrder(columnKey)` to retrieve the current sorting order for the column. + +2. **Compare Data**: If a sorting order is retrieved, it: + - Retrieves data from the UI and JSON. + - Sorts JSON data using the retrieved sorting order. + - Compares the UI and JSON data. + - Logs success if the data is correctly sorted; otherwise, logs an error. + +#### Method: `getCurrentSortingOrder(String columnKey)` + +This method retrieves the current sorting order of a column. + +1. **Find Column Header**: Locates the column header element for the specified column key. + +2. **Scrolls To Find Header**: Uses JavascriptExecutor to scroll and bring the column header into view. + +3. **Retrieve Sort Order**: Retrieves the current sorting order from the column header's attribute `aria-sort`. + +4. **Click to Change Sort Order**: Clicks the column header to change the sort order and waits for the sorting order to change using `waitToLoadChangedSortOrder`. + +5. **Return New Sort Order**: Returns the new sorting order. + +#### Method: `waitToLoadChangedSortOrder(String oldSortOrder, String columnKey)` + +This method waits for the sorting order to change after clicking a column header. + +1. **Initialize Wait Time**: Starts with an initial wait time specified by `Config.WAIT_TIME_IN_SECONDS`. + +2. **Check Sort Order Change**: Repeatedly checks the sorting order of the column header until it changes from the old sort order. + +3. **Return New Sort Order**: Returns the new sorting order once it has changed. + +#### Method: `getDataFromJSONUsingColumnKey(String columnKey)` + +This method retrieves data from JSON objects for a specified column key. + +1. **Iterate Through JSON Objects**: For each JSON object: + - Retrieves the column data using `getColumnDataFromJson`. + +2. **Return Data**: Returns the list of column data. + +#### Method: `getDataFromUIUsingColumnKey(String columnKey)` + +This method retrieves data from the UI based on a specified column key. + +1. **Initialize List**: Starts by creating an empty list to store the data. +2. **Load UI Elements**: Calls the `waitToLoadListOfAllUIObjects` method to get all web elements corresponding to the column key. +3. **Extract Text**: Iterates through the list of web elements (excluding the header) and adds the text content of each element to the list. +4. **Error Handling**: If any exception occurs, it logs an error message. +5. **Return Data**: Finally, returns the list of extracted data. + +#### Method: `waitToLoadListOfAllUIObjects(String columnKey)` + +This method waits until all UI elements corresponding to a column key are loaded. + +1. **Initialize Wait Time**: Starts with an initial wait time. +2. **Find Elements**: Uses a loop to repeatedly find web elements in a grid cell matching the column key's XPath. +3. **Check Element Count**: Checks if the number of elements (excluding the header) matches the number of JSON objects. +4. **Wait and Retry**: If the elements are not fully loaded, it sleeps for the current wait time, increments the wait time, and retries. +5. **Return Elements**: Once the elements are fully loaded, returns the list of web elements. + +#### Method: `ascendingSortJson(String columnKey)` + +This method sorts the JSON objects in ascending order based on a column key. + +1. **Sort JSON Objects**: Uses a comparator to sort the list of JSON objects by comparing the values of the specified column key. +2. **Error Handling**: If any exception occurs during sorting, logs an error message. + +#### Method: `descendingSortJson(String columnKey)` + +This method sorts the JSON objects in descending order based on a column key. + +1. **Sort JSON Objects**: Similar to `ascendingSortJson`, but uses a reversed comparator to sort the list in descending order. +2. **Error Handling**: If any exception occurs during sorting, logs an error message. + +#### Method: `testTableContent()` + +This method tests the content of a table by comparing the UI data with JSON data. + +1. **Log Number of Objects**: Logs the number of objects retrieved from JSON. +2. **Retrieve UI Data**: Calls `getDataFromUIUsingColumnKey` to get the data from the UI for the unique key. +3. **Compare Counts**: Compares the number of objects from JSON and UI. Logs an error if they are not equal and returns false. +4. **Initialize Pass Flag**: Initializes a flag to track if all comparisons pass. +5. **Iterate Through Columns**: For each column key: + - Retrieves data from UI and JSON. + - Calls `compareData` to compare the data and updates the pass flag based on the result. +6. **Return Result**: Returns the pass flag indicating whether all comparisons passed. + +#### Method: `getAllObjectsFromJson()` + +This method retrieves all objects from a JSON file. + +1. **Get File Path**: Retrieves the path of the JSON file. +2. **Parse JSON**: Calls `parseJson` to parse the JSON file and return a list of objects. +3. **Error Handling**: If any exception occurs during parsing, logs an error message. +4. **Return Data**: Returns the list of parsed objects. + +#### Method: `compareData(List dataUI, List dataJSON, List dataIDUI, String columnName)` + +This method compares the data from UI and JSON for a specific column. + +1. **Initialize Pass Flag**: Initializes a flag to track if all comparisons pass. +2. **Iterate Through Data**: For each item in the data lists: + - Parses the UI and JSON values. + - Calls `checkValues` to compare the values and updates the pass flag based on the result. +3. **Log Success**: If all comparisons pass, logs a success message. +4. **Return Result**: Returns the pass flag indicating whether all comparisons passed. + +#### Method: `checkValues(Object dataUIValue, Object dataJSONValue, Object dataIDUIValue, String columnName)` + +This method checks if the UI value matches the JSON value for a specific column. + +1. **Compare Values**: Compares the UI value with the JSON value. +2. **Log Error**: If the values do not match, logs an error message. +3. **Return Result**: Returns a boolean indicating whether the values match. + +#### Method: `clickDropdown(int numOfItemsJSON)` + +This method selects a dropdown value based on the number of JSON items. + +1. **Find Dropdown**: Locates and clicks the dropdown element. +2. **Wait for Dropdown List**: Waits for the dropdown list to become visible. +3. **Determine Selected Value**: Determines the smallest dropdown value greater than the number of JSON items. +4. **Select Option**: Iterates through the options and clicks the one matching the selected value. +5. **Wait for Invisibility**: Waits for the dropdown list to become invisible. +6. **Log and Refresh**: Logs the selected value and refreshes the page. +7. **Error Handling**: If any exception occurs, logs an error message. + +#### Method: `getSelectedDropdownValue()` + +This method retrieves the currently selected value of a dropdown. + +1. **Find Dropdown**: Locates the dropdown element. +2. **Retrieve Text**: Gets and trims the text of the dropdown element. +3. **Error Handling**: If any exception occurs, logs an error message. +4. **Return Value**: Returns the retrieved text. + +#### Method: `getSaveButtonWebElementDetailsPage()` + +This method retrieves the web element for the save button on the details page. + +1. **Locate Save Button**: Uses an XPath to find the save button element. +2. **Return Element**: Returns the located web element. + +#### Method: `testForAllText()` + +This method tests the presence of specific text elements on a page. + +1. **Log Start**: Logs the start of the text tests. +2. **Check Text**: For each expected text element: + - Calls `Common.checkTextPresent` to verify the text presence on the page. + + + +### ECLWorkUnitsTest + +![img.png](ECLWorkUnitsTest.png) + +This class, `ECLWorkUnitsTest`, in the framework.pages package, extends the `BaseTableTest` +class and specifically implements test cases for the ECL Workunits page within the ECL Watch UI. +It inherits functionalities for common table testing procedures and specializes them for the +ECL Workunits data and behavior. It includes test cases for both the main page and the details page of the workunits. + +#### Class Variables: +- badStates: A list containing states considered as 'bad', such as "compiled" and "failed". +- sortByColumnKeyWhenSortedByNone: A string representing the column key used for sorting columns when no other sorting order is specified, or the sorting order is "none". +- costColumns: A list containing column names related to cost, used for specific parsing operations. +- jsonMap: A map storing ECLWorkunit objects keyed by their unique identifiers (WUID). + +#### Methods: + +#### Method: `testingECLWorkUnitsPage` +This method is a test method annotated with `@Test`, which indicates that it is a test case to be run using a testing framework like JUnit or TestNG. The method calls `testPage()` to execute the test logic for the ECL Work Units page. + +1. Annotate the method with `@Test` to indicate that it is a test case. +2. Within the method, call the `testPage()` method to perform the necessary testing actions on the ECL Work Units page. + +#### Method: `getPageName` +This method returns the name of the page being tested, which in this case is "ECL Workunits". + +1. Override the method to return the string "ECL Workunits". + +#### Method: `getPageUrl` +The getPageUrl method is designed to retrieve the URL of a specific page within the application. It overrides a method from a superclass, BaseTableTest, and provides the URL for the "Workunits" tab under the "ECL" navigation section. This URL is fetched from a pre-configured map of URLs. + +1. The method accesses the urlMap from the URLConfig class, which stores URL mappings for different navigation sections and their tabs. +2. The method fetches the URLMapping object for the "ECL" navigation section from the urlMap using URLConfig.NAV_ECL as the key. +3. From the URLMapping object of the "ECL" navigation section, the method retrieves the nested URL mapping for the "Workunits" tab using URLConfig.TAB_ECL_WORKUNITS as the key. +4. Finally, the method returns the URL associated with the "Workunits" tab. + +#### Method: `getJsonFilePath` +This method returns the file path of the JSON file containing Workunits test data. +Combine the directory path (Config.PATH_FOLDER_JSON) with the filename (Config.WORKUNITS_JSON_FILE_NAME) to form the complete file path. + +1. Override the method to return the local path `Config.PATH_LOCAL_WORKUNITS_JSON` if the test is running locally. +2. Return the CI/CD path `Config.PATH_GH_ACTION_WORKUNITS_JSON` otherwise. + +#### Method: `getColumnNames` +This method returns an array of column names to be displayed on the ECL Work Units page. + +1. Override the method to return an array of column names: "WUID", "Owner", "Job Name", "Cluster", "State", "Total Cluster Time", "Compile Cost", "Execution Cost", "File Access Cost". + +#### Method: `getColumnKeys` +This method returns an array of column keys corresponding to the column names used in the JSON data. + +1. Override the method to return an array of column keys: "Wuid", "Owner", "Jobname", "Cluster", "State", "TotalClusterTime", "Compile Cost", "Execution Cost", "File Access Cost". + +#### Method: `getSaveButtonDetailsPage` +This method returns the label of the save button on the details page. + +1. Override the method to return the string "Save". + +#### Method: `getDetailNames` +This method returns an array of detail names to be displayed on the work unit details page. + +1. Override the method to return an array of detail names: "WUID", "Action", "State", "Owner", "Job Name", "Description", "Potential Savings", "Compile Cost", "Execution Cost", "File Access Cost", "Protected", "Cluster", "Total Cluster Time", "Aborted by", "Aborted time", "Services". + +#### Method: `getDetailKeys` +This method returns an array of detail keys corresponding to the detail names used in the JSON data. + +1. Override the method to wait for the element with ID "state" to load. +2. Check if the state value is in the list of bad states (`compiled`, `failed`). +3. If the state is in the bad states list, return a subset of detail keys: "wuid", "action", "state", "owner", "jobname", "cluster". +4. Otherwise, return the full array of detail keys: "wuid", "action", "state", "owner", "jobname", "compileCost", "executeCost", "fileAccessCost", "protected", "cluster", "totalClusterTime". + +#### Method: `getDetailKeysForPageLoad` +This method returns an array of keys to be used for loading the details page. + +1. Override the method to return an array of keys: "wuid", "state", "jobname", "cluster". + +#### Method: `getCheckboxTypeForDetailsPage` +This method returns the type of the checkbox used on the details page. + +1. Override the method to return the string "checkbox". + +#### Method: `getAttributeTypeForDetailsPage` +This method returns the attribute type used to identify elements on the details page. + +1. Override the method to return the string "type". + +#### Method: `getAttributeValueForDetailsPage` +This method returns the attribute value used to identify elements on the details page. + +1. Override the method to return the string "value". + +#### Method: `getColumnKeysWithLinks` +This method returns an array of column keys that contain links. + +1. Override the method to return an array with a single key: "Wuid". + +#### Method: `getUniqueKeyName` +This method returns the name of the unique key used to identify work units. + +1. Override the method to return the string "WUID". + +#### Method: `getUniqueKey` +This method returns the key used to uniquely identify work units. + +1. Override the method to return the string "Wuid". + +#### Method: `getTabValuesForDetailsPage` +This method returns an array of tab values for the details page. + +1. Define a method to return an array of tab values: "variables", "outputs", "inputs", "metrics", "workflows", "queries", "resources", "helpers", "xml". + +#### Method: `getColumnNamesForTabsDetailsPage` +This method returns a map where each key is the value attribute of an HTML element of a tab, and the value is a list of column names for that tab. + +1. Override the method to return a map of tab values and their corresponding column names: + - "variables": ["Type", "Name", "Value"] + - "outputs": ["Name", "File Name", "Value", "Views"] + - "inputs": ["Name", "File Cluster", "Usage"] + - "metrics": ["Refresh", "Hot spots", "Timeline", "Options"] - there is no column in this tab, so checking the presence of these buttons + - "workflows": ["Name", "Subtype", "Count", "Remaining"] + - "queries": ["ID", "Priority", "Name", "Target", "WUID", "Dll", "Published By", "Status"] + - "resources": ["Name", "Refresh", "Open", "Preview"] - Preview is not a column, it is a button, keeping it for additional check for Resources tab + - "helpers": ["Type", "Description", "File Size"] + - "xml": ["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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/NavigationWebElement.png b/esp/src/test-ui/tests/framework/documentation/NavigationWebElement.png new file mode 100644 index 0000000000000000000000000000000000000000..b32e3dc1ae1e182fb6fad59ed7efba80101093b2 GIT binary patch literal 10506 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/Relationship.png b/esp/src/test-ui/tests/framework/documentation/Relationship.png new file mode 100644 index 0000000000000000000000000000000000000000..9fc90e8db183f4c854e086fc768ff93d5477cbfd GIT binary patch literal 41362 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 literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/TestClass.png b/esp/src/test-ui/tests/framework/documentation/TestClass.png new file mode 100644 index 0000000000000000000000000000000000000000..6d9370dada3720217e4b76ce76377d045fb9ae70 GIT binary patch literal 3296 zcmcIne^gV~9e=0?N{>Rfo}QK|v1j#2R~%L;#u^h^>*`RE9v#F033D#wKq#OgL=qmb z%@sWrSw%piNpwz_gd~+nAqfdO9R3I-0_Md?l1GXVAdtr-KOiAwFW}l*+a24v`^P=+ zym#;S-uwOh_}+K?qdhxTy!f*h0RUK$@aye+0U)RneQtgs813buAG{6#%Nr85zyF&f zl*#)Qr)-F9SfjW5-eeNO=td zl&PPgo0yuJwuZkNX!_FH#)P>@3;?V$?*4ni%!j~xNx5t9ttoy3I3I<5Is2!op8$8O z&W6T?!Y=@$w|kCqkLE1{91kOQKPssPIPPlg#vViniJ zGcsT>%RbSRFw<3c=uL%9IAM&AXF&1lAb|1swGqehHR_ZOnRK2M2ApBgXy^i((d&2) zZ~Y~wR4G=`A)Lb9B*v%qZvp0hZ{ICMqebSzZ>8x>TX4e5CUZ2BmmE?i(D1U*-L`d3 z;x@8)&NpjU(Bk53Ix8I}ecTWR#Mj~U=&25s4_KRDF z=ak!L4AWPa-!yspE{beM(b_rO<^Vye{lOoqu8v)sLb*oGKer-0CeXx(E_0Xp)k}e? z_~*%ir}sItFH$g!1u?)8Eb9|!P4Q4vni5X-WgRT5a9==n=vkl+1YT#*cA@z!Y1Vnk zI>Fzxnh(~{fT_#Ih-8##NxlA`v$PnQkp3}L#X;H>td<>cD3B5Y*GV`VNHPl#it0!xTT(ciAnA_OUTe?C$G)~xf#BaAjim8 z119ouPz;Bv?8LLm2i)WAOsJuVqTpr%Yz8f$9YWTs{n9_}qKlI_nKCiUWMJF}?M?h? z6Wg99BG=G?!NpL03PCwz4g+jl++vm4xAokR8~BTCL3o0{`fEeY>;1uVVZdqJtN*iH zpN8s~_awqdKPYlarjp%3W!=1AqT)KnaVWaK3l4~xr;!@wqwRL`WUsqkdz(&-S2Geb zX!24h=oxzFgFAaaiZXHPb#F9jCBc*8U3 z4tWejZ%9Wxnw)@79)qL(aLSzP{{+a7vf*#zdc30|eW4BQjSG#Z0^jT3-=6p*%fLa_ zP1d7OSRK_KGKUd7LYzsE#mf*}L2C!gXu!@7GxbD+j94<#R9e%~AJtyY>MxCQ;O;uf z8B9Tv)RQjs^p|)L?>#X7xc5ZB-iI`*cACqREaZ$aXg=v2S>>ID-;bAi!Ga2ZtEVKz z88s#?{SxtxAxLzac0cL^IQxW}qoG)LW%+U>LWJwo*zrPoC3Kb0w$-ggT|vZQ@OxYt zhge8Lsxz#8coK zXwwqWWnyPP)!E)^rB#c@b9p@7yyxYF zYbZ4Q9cm{jv4`!PeZWc?xEwC!j^v7nVxAHL<$m1gG*gt~JODMwy!?%|%58`hVrThI zKulk*rFncKE@EHBX ze(P=$UyG?S)>aSwTUWRa96> zBE#-Ic((|OQMX*DDvVyPZramCtFf{5Jbbyysu_P57Fkug2UOVz)bZH@zm93IMrd^te)Gwx#G5_8i8O15z9n~l>F>@&mV(g%t$PH#)9uAC$o5yCJP;1A7@EF_2r%B}k) zVd0nXLS}s&IDd;m3YX61rwmxk?yy!w=j;xwQ5e?)Y*qVu0urd|`or6Np;qfOST+bn zuE?u)b_>dtEeykfnv4yckvkeMq5Esb1P>N19tR#%2>ff@hV)O~RjnUX8sT&_ud04l zyg$J%l`Ur!X9VDo1?~@h|7tG|itv{Ju-a!r*Ze^=`2M^0a}HCwQq~q|U6-<#0N0ju hE(P)( literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/TestClasses.png b/esp/src/test-ui/tests/framework/documentation/TestClasses.png new file mode 100644 index 0000000000000000000000000000000000000000..913e8d519fa36031a34507ec607ddb3447a9089e GIT binary patch literal 2324 zcmd^BeM}Q)7(b&!XQs=xWF}kGE!%X+tQz7XA3`P?M8zslI)-gcTw$UXw{kscr3@3} z+=n<3>ld&LYV8NDQUU34c8ggSQ8@+bS<00Wg-Va!F|a+_>-Bb;Y|;6z%a-h)_j%sr z$?x|*@B99Kc@A&coH=Lq8?ylbn3J`BZ4LlTyTa#(X2tRA&g#@v05Hpwwf3XDebT_) z+lqV1ugQ7tHufb7q};b5VaN38?z&mt*>YZ{NIJ8JiBs?hA3?OJULYv~kf( zX~31Qf1df1c*PVjwIpsKAk)v^Um1&D{NQ?vrCpk^=h$pux#aYY=NKvfK7(+-4H}|f zj%JTC=dyE#WS5y#Sc#69*%!O3dYT))@JTUrTb(R0J$mV3Meicx-e1XYAf34SY<4|@ z+L5cI1PXVz!jZ5D*0@l~6O~7smLVBB3~_-Hv6(HYHE7uZwUu-cYK2^9>nko`JJnW! z5EnRifhv#^NzuODI`K-X8=&4}Qxr9MAG^UEsT~h?kpuQ{370piuBdIfDiAtIMX3(L z#PMdbzT$pZn=;dz8-WHE8Mn+Y?hQ+{=fk904w3Htt(B&7#Hw=ENUrpN-5OVIcOLGI zm%IGC*{E6|hdYZPo@^Og_$-+})t44WSH>2!4(tf@X@B|k;p($yAtN%R6Jn*Ga_mH`G4?1amzC!oB8xNEBZg z$H=$~`93j|TSt{it_fu2ZL}$(fPHs|6VTAyV3+x%~E>ap)ig^xa{A#pO>rb{EQnx{OgDOSIxTY*pA#&q<`@nVQJVuNoX z$}>%CKz1DlijzHZ43T&xGQGv234QNl0-dmQv>SJuF?1FA`&b1TawwsYAN08s;U^0{ z!=f%#KE5MeE^3Ery*6$BO<`~oLJr_I%!~+i-VRg9I16--OL1q)~pM))i za3W^uA3juePTD|{7|~n84kxTvQXWjlAC0vaj;&rrMYtqHu zHwkPSo3z2fI-qS-J?G+Ifbf!$zQCZ{BN^OUSz3Lb@5&#d#sXF$3_WO3P>(9im_gwr zU67?5_Y7l*f_udrE8&i|WZWc_cRD?qXV|8e`WEKVZp>0|WF*}9ZYhP=nKd{XykRU_ zIap_8;IZ3@Ak~lb52v7Igc3xz(8MVR6p#ic_)>XL%YKXYevqmPk zDR8H=K8qQ3SWTW3{aEhf9)Xh&JkItW#_-qPU+9E;G={MnJHtnWktjrla121 zdG`m|gN3}5sn)atwex>rEB};T{l{>R9N-OFj!b@1uoxJxdUhBYc1B-MjzgPOoA_x5 O$Xd5~t#!@z1AhQwxV?n{ literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/TestInjector.png b/esp/src/test-ui/tests/framework/documentation/TestInjector.png new file mode 100644 index 0000000000000000000000000000000000000000..e45e93682a7382aa26d7fc47e06ebbb6e4e26ea3 GIT binary patch literal 10050 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 VQ{f#6-UHgZ8~Kj#5AXlwe*rvW%)F`F=m| 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 RHUx6e|L_O2|NiuE{|CeIHQxXL literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/WUApplicationValue.png b/esp/src/test-ui/tests/framework/documentation/WUApplicationValue.png new file mode 100644 index 0000000000000000000000000000000000000000..47bb55d8d921f2c9f7655e61ff28cd0c7a022b7c GIT binary patch literal 2919 zcmd^BaZnTK6<@?@sTCP&JD$v(-q_A%#_JrmG*-Zba&Rc8l2crA1_NHTQi%nQB?J<) zXzSE-(CT@msYntnLe5P{=7gYRNy43XSUE2d!?Gn10wM+yl2}5rCLy7_;B}_sIn#Fh z<7RIE*!kw$?|bjSaEA<8-5gI zDlaj3%NezUqb0IU zjKJ!5H(L*>1N`zEMMF}EVEZ!(J_EJxKqGxk*Jk&jY3;-73;b8Ccw+#WGN{=IM5F5! zrADFtdK$#B`<11v2P4H9iTa#SRskBh8b=@)GC_`_% z*>pP7du%#-+BuTj);_4;ZBbToa2WF+GDZB)gfg_HJzXR)UbqN~?+t zWibq`O*l}%2eJD{#6_-AT7{>Jtmtk|WYa0k5Irz=w~V23{sGnSxn^x?GAQSD4i{hh zIR{18QNLrys*Is^)vhaue4yI3yAb2IX~(asgW#h(O(G9j=3iKn3&AktE?q#>_-g^u zd4XF>CQ7rH=um=sGGi?BkzIr`tbT?r~0cE`SK@(qw^n@SeV${#N7a82#h&i{kI7Xsqv8F^xV zzrMcy3%!mc#j6`$C~ux|Ii;9n)IAW#8qfMf|De+IJ!a70$W+R+a5Y<>Q1N-KiJ~Us z6WZvFQOGa3qCP=`WH^=TRgoPJu2%EQG1=iN29D_S%c>>)A`8=}vl3bay>L*_Q1$mYVy0q2=VY|U*Bujg(plk#VK&tLzu zQX!brAB!io^UT}yLEXX%jd1}zD%@o}Cha@?eXn_-a$Tg_W)8pFi)4t$Qm4?UuE{pUm0!fy4cx znZFt(P6lE<+K{hIX)M;S@~8zA6xM-b4YG>4}!lNjnyrA zO@tHR=*?S8_Q#1a$GP;UADH;O5uTrUPvTqYf5E*chBYxSql|`@qh{@}Kx*(#cPZ8` zB)&vwp0a23i(u-~i41=ijCck2%$pNbU4Jy*3f2XNn%x-f%Le46;jv zK!m0=e5f*bD%;UBqiEoor`QgGa$?TE)2D=L;vJ})_BZ^uaNOL9iX%n#b~Hof%+hM; zZm?g>)$DcTl?qw9x;O022M@8myRpN{CXTd!-syjb7q6K-W~6fKx%XXe*#1?QXd%>V zmoB1yfhDRVJIlNLyhV!AyZ_{iQ0=l3)OS=1NRiLd7GsAo!(f$uk2L+7d(gh?-#9Sy zY=zk4G!rpQg5~cm`bsY5Q7twS0o^9>)P3ZnX>FlwzMY1Q*!+FYra=c^tDtxPWyyu^ z;MT3p8s<(P%r9q$bqYsEYZzPGBEnJ``Iw(0y?BI+E1Gtuxlya!rTB`HH4I1CO%h=) z4-F5wV^d@jqS9broCP{ixsADLqP8~q>5r43*nuSe*k#D>7KVNH}KX?5P yg7u9x^8Uivo8&X1RLX(l<-K1&$38paE&%XK2RakGhMb%MAotC@Z1o!le*Z6W{vrzi literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/WUApplicationValues.png b/esp/src/test-ui/tests/framework/documentation/WUApplicationValues.png new file mode 100644 index 0000000000000000000000000000000000000000..7aa4a5756328c3af6c39de72be3cb2ce924bae11 GIT binary patch literal 2566 zcmeHJYfu|y6pc@`V~4RmI`sjfIM|}`fmIPgwA7|4wc1)p!ERDV)677zT0&y7B$@ik z*lJa-FxmmXYPB?Z{RFmblV~ZgRz*qcJ(F(;|c?fKfdl-`m0cDmM|FC4{=vNz4;A( z^OIw~tmb;*57vwi{2yL8x?zU>khtyTLhf+B1Xx55N`@MPVGag7VP;=qcM z*M#Kh-cY-!ljC$2g7_%{Ca96}x_!xbaz8F7O3AQk%^~a@lh_ ze-##@fHk(ODDNxBe!fa%2lr8 z0DEYZa2Q!&Rz>1)NZR$NFWMl40=wF#1`EN~13a}SCiAW=3UZ~Z5Gc*t4k@~k4ccS< zOldcwN7roED7tXj`2xZ68dRq)R|F`a)|0Rt@b71Gs_n6kJ4<-FcNwOO*;|8DU8F;`84c~3JE2EySyq3CsttVB zUPT&Pl~a#sy28@tPhh5|*`&i(6b#691y~ zd%%$)m!$(x+(`FtmfN`cjQ1OwwRQg*P&S?hslCwR3 z-~Xg$yyH^FV`!v$q#0BWC;5G#FTr#A9(Q4+-kZGq2T`|hAAgWq)=4)38t?I8J6Ysm zP{O;%=_ALKR6V;0nLXWU03z*vbJA0nZ4*gX{v1|te3wAvCW>FFpZ?15zF-TvS6d^Z zEaf{B;hfoK?X2C$9L^!fQH!-ZXdXvxd|#8_>Pd9)qO4A^c82nA^;pBDwb8@EVChU5p1{?Hq&C&6>*kL%Kl2mKpLWrFUR z{<(brg6?@`Gv{Kug(txAL-57{LuqKHmyowN@OA1|>P$_v`czOidOq4&2~HKKlZsX^ zXG3)A`M|yrU7={_$YjtZFEQKqeKgqqd2}x=fZkQ4(Fn^g8$u;rhK^^u<5_(u$kG$rF*ocY zPt$|P&w<>EV|-SofpR6JU13O;`lG%ryNH5GM{y?5G9vLG1~X+uwfT2@u~$v?frBxZ zn_DZ*>MCcUmA9{DMFrlbNe8m$_eQL^uymzPi+pLt;qJdyzZ?Fxd3Lhd8?)$|jYFhU zlB#rSqoZ!y3?~&9GHXVok1PiPW&yx5mv28nTac-V6tkt6(Vho-VcA?L019cR#)YI@ zqL^rkIB>f|yfkj_r)zZ#5Q#8#Hg8OUNEH*zfOJ1iaD)01)Mvm>hRUEFWXhLFd-Mxw zQ)$fIL)`kG1N!K0{zC8sSC`#_D;sk+ihI(w_&@j;Nl;|QC(~i0?%qOgQ-X_swnzE@ c_F*c+&?A#WXLnsm?{^IDGaFW$R&9IpHwH3<*Z=?k literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/WUECLWorkunit.png b/esp/src/test-ui/tests/framework/documentation/WUECLWorkunit.png new file mode 100644 index 0000000000000000000000000000000000000000..5bbf9018ae624487a62be8bdd4f47a20885873b7 GIT binary patch literal 21832 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 SH>5%zeelV@v)@1d<^KoiLcxCk literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/WUQueryResponse.png b/esp/src/test-ui/tests/framework/documentation/WUQueryResponse.png new file mode 100644 index 0000000000000000000000000000000000000000..a84157c8b310b1e4c7e299cec64dd4083bb11e1c GIT binary patch literal 12328 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 W`en>^cknq6?A;R?#(y{A)V~4M-*OTF literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/documentation/WUWorkunits.png b/esp/src/test-ui/tests/framework/documentation/WUWorkunits.png new file mode 100644 index 0000000000000000000000000000000000000000..0c50d24564b5366198622c12137dd4f283d639f5 GIT binary patch literal 4176 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 Yq`9yCl)1MN`!N90-zKC$Zyh@IPlJbg&j0`b literal 0 HcmV?d00001 diff --git a/esp/src/test-ui/tests/framework/model/DFULogicalFile.java b/esp/src/test-ui/tests/framework/model/DFULogicalFile.java new file mode 100644 index 00000000000..5d1aa8bed4d --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/DFULogicalFile.java @@ -0,0 +1,382 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +public class DFULogicalFile { + + @JsonProperty("Prefix") + private String prefix; + + @JsonProperty("NodeGroup") + private String nodeGroup; + + @JsonProperty("Directory") + private String directory; + + @JsonProperty("Description") + private String description; + + @JsonProperty("Parts") + private String parts; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Owner") + private String owner; + + @JsonProperty("Totalsize") + private String totalSize; + + @JsonProperty("RecordCount") + private String recordCount; + + @JsonProperty("Modified") + private String modified; + + @JsonProperty("LongSize") + private String longSize; + + @JsonProperty("LongRecordCount") + private String longRecordCount; + + @JsonProperty("isSuperfile") + private boolean isSuperfile; + + @JsonProperty("isDirectory") + private boolean isDirectoryBoolean; + + @JsonProperty("Replicate") + private boolean replicate; + + @JsonProperty("IntSize") + private int intSize; + + @JsonProperty("IntRecordCount") + private int intRecordCount; + + @JsonProperty("FromRoxieCluster") + private String fromRoxieCluster; + + @JsonProperty("BrowseData") + private boolean browseData; + + @JsonProperty("IsCompressed") + private boolean isCompressed; + + @JsonProperty("ContentType") + private String contentType; + + @JsonProperty("CompressedFileSize") + private int compressedFileSize; + + @JsonProperty("SuperOwners") + private String superOwners; + + @JsonProperty("Persistent") + private boolean persistent; + + @JsonProperty("IsProtected") + private boolean isProtected; + + @JsonProperty("KeyType") + private String keyType; + + @JsonProperty("NumOfSubfiles") + private String numOfSubfiles; + + @JsonProperty("Accessed") + private String accessed; + + @JsonProperty("AtRestCost") + private double atRestCost; + + @JsonProperty("AccessCost") + private double accessCost; + + @JsonProperty("MinSkew") + private int minSkew; + + @JsonProperty("MaxSkew") + private int maxSkew; + + @JsonProperty("MinSkewPart") + private int minSkewPart; + + @JsonProperty("MaxSkewPart") + private int maxSkewPart; + + // Getters and Setters for all fields + + public String getPrefix() { + return prefix; + } + + public void setPrefix(String prefix) { + this.prefix = prefix; + } + + public String getNodeGroup() { + return nodeGroup; + } + + public void setNodeGroup(String nodeGroup) { + this.nodeGroup = nodeGroup; + } + + public String getDirectory() { + return directory; + } + + public void setDirectory(String directory) { + this.directory = directory; + } + + public String getDescription() { + return description; + } + + public void setDescription(String description) { + this.description = description; + } + + public String getParts() { + return parts; + } + + public void setParts(String parts) { + this.parts = parts; + } + + public String getName() { + return name; + } + + public void setName(String name) { + this.name = name; + } + + public String getOwner() { + return owner; + } + + public void setOwner(String owner) { + this.owner = owner; + } + + public String getTotalSize() { + return totalSize; + } + + public void setTotalSize(String totalSize) { + this.totalSize = totalSize; + } + + public String getRecordCount() { + return recordCount; + } + + public void setRecordCount(String recordCount) { + this.recordCount = recordCount; + } + + public String getModified() { + return modified; + } + + public void setModified(String modified) { + this.modified = modified; + } + + public String getLongSize() { + return longSize; + } + + public void setLongSize(String longSize) { + this.longSize = longSize; + } + + public String getLongRecordCount() { + return longRecordCount; + } + + public void setLongRecordCount(String longRecordCount) { + this.longRecordCount = longRecordCount; + } + + public boolean isSuperfile() { + return isSuperfile; + } + + public void setSuperfile(boolean superfile) { + isSuperfile = superfile; + } + + public boolean isDirectoryBoolean() { + return isDirectoryBoolean; + } + + public void setDirectoryBoolean(boolean isDirectory) { + isDirectoryBoolean = isDirectory; + } + + public boolean isReplicate() { + return replicate; + } + + public void setReplicate(boolean replicate) { + this.replicate = replicate; + } + + public int getIntSize() { + return intSize; + } + + public void setIntSize(int intSize) { + this.intSize = intSize; + } + + public int getIntRecordCount() { + return intRecordCount; + } + + public void setIntRecordCount(int intRecordCount) { + this.intRecordCount = intRecordCount; + } + + public String getFromRoxieCluster() { + return fromRoxieCluster; + } + + public void setFromRoxieCluster(String fromRoxieCluster) { + this.fromRoxieCluster = fromRoxieCluster; + } + + public boolean isBrowseData() { + return browseData; + } + + public void setBrowseData(boolean browseData) { + this.browseData = browseData; + } + + public boolean isCompressed() { + return isCompressed; + } + + public void setCompressed(boolean isCompressed) { + this.isCompressed = isCompressed; + } + + public String getContentType() { + return contentType; + } + + public void setContentType(String contentType) { + this.contentType = contentType; + } + + public int getCompressedFileSize() { + return compressedFileSize; + } + + public void setCompressedFileSize(int compressedFileSize) { + this.compressedFileSize = compressedFileSize; + } + + public String getSuperOwners() { + return superOwners; + } + + public void setSuperOwners(String superOwners) { + this.superOwners = superOwners; + } + + public boolean isPersistent() { + return persistent; + } + + public void setPersistent(boolean persistent) { + this.persistent = persistent; + } + + public boolean isProtected() { + return isProtected; + } + + public void setProtected(boolean isProtected) { + this.isProtected = isProtected; + } + + public String getKeyType() { + return keyType; + } + + public void setKeyType(String keyType) { + this.keyType = keyType; + } + + public String getNumOfSubfiles() { + return numOfSubfiles; + } + + public void setNumOfSubfiles(String numOfSubfiles) { + this.numOfSubfiles = numOfSubfiles; + } + + public String getAccessed() { + return accessed; + } + + public void setAccessed(String accessed) { + this.accessed = accessed; + } + + public double getAtRestCost() { + return atRestCost; + } + + public void setAtRestCost(double atRestCost) { + this.atRestCost = atRestCost; + } + + public double getAccessCost() { + return accessCost; + } + + public void setAccessCost(double accessCost) { + this.accessCost = accessCost; + } + + public int getMinSkew() { + return minSkew; + } + + public void setMinSkew(int minSkew) { + this.minSkew = minSkew; + } + + public int getMaxSkew() { + return maxSkew; + } + + public void setMaxSkew(int maxSkew) { + this.maxSkew = maxSkew; + } + + public int getMinSkewPart() { + return minSkewPart; + } + + public void setMinSkewPart(int minSkewPart) { + this.minSkewPart = minSkewPart; + } + + public int getMaxSkewPart() { + return maxSkewPart; + } + + public void setMaxSkewPart(int maxSkewPart) { + this.maxSkewPart = maxSkewPart; + } +} diff --git a/esp/src/test-ui/tests/framework/model/DFULogicalFiles.java b/esp/src/test-ui/tests/framework/model/DFULogicalFiles.java new file mode 100644 index 00000000000..0e4bf769dfc --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/DFULogicalFiles.java @@ -0,0 +1,19 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +import java.util.List; + +public class DFULogicalFiles { + + @JsonProperty("DFULogicalFile") + private List dfuLogicalFile; + + public List getDFULogicalFile() { + return dfuLogicalFile; + } + + public void setDFULogicalFile(List dfuLogicalFile) { + this.dfuLogicalFile = dfuLogicalFile; + } +} diff --git a/esp/src/test-ui/tests/framework/model/DFUQueryResponse.java b/esp/src/test-ui/tests/framework/model/DFUQueryResponse.java new file mode 100644 index 00000000000..8f7165cd450 --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/DFUQueryResponse.java @@ -0,0 +1,95 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +public class DFUQueryResponse { + + @JsonProperty("DFULogicalFiles") + private DFULogicalFiles dfuLogicalFiles; + + @JsonProperty("Prefix") + private String prefix; + + @JsonProperty("NodeGroup") + private String nodeGroup; + + @JsonProperty("LogicalName") + private String logicalName; + + @JsonProperty("Description") + private String description; + + @JsonProperty("Owner") + private String owner; + + @JsonProperty("StartDate") + private String startDate; + + @JsonProperty("EndDate") + private String endDate; + + @JsonProperty("FileType") + private String fileType; + + @JsonProperty("FileSizeFrom") + private int fileSizeFrom; + + @JsonProperty("FileSizeTo") + private int fileSizeTo; + + @JsonProperty("FirstN") + private int firstN; + + @JsonProperty("PageSize") + private int pageSize; + + @JsonProperty("PageStartFrom") + private int pageStartFrom; + + @JsonProperty("LastPageFrom") + private int lastPageFrom; + + @JsonProperty("PageEndAt") + private int pageEndAt; + + @JsonProperty("PrevPageFrom") + private int prevPageFrom; + + @JsonProperty("NextPageFrom") + private int nextPageFrom; + + @JsonProperty("NumFiles") + private int numFiles; + + @JsonProperty("Sortby") + private String sortby; + + @JsonProperty("Descending") + private boolean descending; + + @JsonProperty("BasicQuery") + private String basicQuery; + + @JsonProperty("ParametersForPaging") + private String parametersForPaging; + + @JsonProperty("Filters") + private String filters; + + @JsonProperty("CacheHint") + private long cacheHint; + + @JsonProperty("IsSubsetOfFiles") + private String isSubsetOfFiles; + + @JsonProperty("Warning") + private String warning; + + public DFULogicalFiles getDFULogicalFiles() { + return dfuLogicalFiles; + } + + public void setDFULogicalFiles(DFULogicalFiles dfuLogicalFiles) { + this.dfuLogicalFiles = dfuLogicalFiles; + } +} diff --git a/esp/src/test-ui/tests/framework/model/DFUQueryRoot.java b/esp/src/test-ui/tests/framework/model/DFUQueryRoot.java new file mode 100644 index 00000000000..aae671c680f --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/DFUQueryRoot.java @@ -0,0 +1,17 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +public class DFUQueryRoot { + + @JsonProperty("DFUQueryResponse") + private DFUQueryResponse dfuQueryResponse; + + public DFUQueryResponse getDFUQueryResponse() { + return dfuQueryResponse; + } + + public void setDFUQueryResponse(DFUQueryResponse dfuQueryResponse) { + this.dfuQueryResponse = dfuQueryResponse; + } +} diff --git a/esp/src/test-ui/tests/framework/model/NavigationWebElement.java b/esp/src/test-ui/tests/framework/model/NavigationWebElement.java new file mode 100644 index 00000000000..cf095ad554f --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/NavigationWebElement.java @@ -0,0 +1,14 @@ +package framework.model; + +import org.openqa.selenium.WebElement; + +public record NavigationWebElement(String name, String hrefValue, WebElement webElement) { + + @Override + public String toString() { + return "NavigationWebElement{" + + "name='" + name + '\'' + + ", hrefValue='" + hrefValue + '\'' + + '}'; + } +} diff --git a/esp/src/test-ui/tests/framework/model/TestClass.java b/esp/src/test-ui/tests/framework/model/TestClass.java new file mode 100644 index 00000000000..240cdd98a90 --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/TestClass.java @@ -0,0 +1,16 @@ +package framework.model; + +public class TestClass { + + String name; + String path; + + public TestClass(String name, String path) { + this.name = name; + this.path = path; + } + + public String getPath() { + return path; + } +} diff --git a/esp/src/test-ui/tests/framework/model/URLMapping.java b/esp/src/test-ui/tests/framework/model/URLMapping.java new file mode 100644 index 00000000000..bd68f686444 --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/URLMapping.java @@ -0,0 +1,33 @@ +package framework.model; + +import java.util.HashMap; + +public class URLMapping { + + String name; + String url; + HashMap urlMappings; + + public URLMapping(String name, String url) { + this.name = name; + this.url = url; + urlMappings = new HashMap<>(); + } + + public String getUrl() { + return url; + } + + public HashMap getUrlMappings() { + return urlMappings; + } + + @Override + public String toString() { + return "URLMapping{" + + "name='" + name + '\'' + + ", url='" + url + '\'' + + ", urlMappings=" + urlMappings + + '}'; + } +} diff --git a/esp/src/test-ui/tests/framework/model/WUApplicationValue.java b/esp/src/test-ui/tests/framework/model/WUApplicationValue.java new file mode 100644 index 00000000000..9c6949887a1 --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/WUApplicationValue.java @@ -0,0 +1,15 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +public class WUApplicationValue { + + @JsonProperty("Application") + private String application; + + @JsonProperty("Name") + private String name; + + @JsonProperty("Value") + private String value; +} diff --git a/esp/src/test-ui/tests/framework/model/WUApplicationValues.java b/esp/src/test-ui/tests/framework/model/WUApplicationValues.java new file mode 100644 index 00000000000..57afee6d752 --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/WUApplicationValues.java @@ -0,0 +1,11 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +import java.util.List; + +public class WUApplicationValues { + + @JsonProperty("ApplicationValue") + private List WUApplicationValue; +} diff --git a/esp/src/test-ui/tests/framework/model/WUECLWorkunit.java b/esp/src/test-ui/tests/framework/model/WUECLWorkunit.java new file mode 100644 index 00000000000..3db4ffc9f3a --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/WUECLWorkunit.java @@ -0,0 +1,111 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; +import com.fasterxml.jackson.annotation.JsonSetter; +import framework.utility.TimeUtils; + +public class WUECLWorkunit { + + @JsonProperty("Wuid") + private String wuid; + @JsonProperty("Owner") + private String owner; + + @JsonProperty("Cluster") + private String cluster; + + @JsonProperty("Jobname") + private String jobname; + + @JsonProperty("StateID") + private int stateID; + + @JsonProperty("State") + private String state; + + @JsonProperty("Protected") + private boolean isProtected; + + @JsonProperty("Action") + private int action; + + @JsonProperty("ActionEx") + private String actionEx; + + @JsonProperty("DateTimeScheduled") + private String dateTimeScheduled; + + @JsonProperty("IsPausing") + private boolean isPausing; + + @JsonProperty("ThorLCR") + private boolean thorLCR; + + @JsonProperty("TotalClusterTime") + private long totalClusterTime; + + @JsonProperty("ApplicationValues") + private WUApplicationValues WUApplicationValues; + + @JsonProperty("ExecuteCost") + private double executeCost; + + @JsonProperty("FileAccessCost") + private double fileAccessCost; + + @JsonProperty("CompileCost") + private double compileCost; + + @JsonProperty("NoAccess") + private boolean noAccess; + + public String getWuid() { + return wuid; + } + + public String getOwner() { + return owner; + } + + public String getCluster() { + return cluster; + } + + public String getJobname() { + return jobname; + } + + public String getState() { + return state; + } + + public long getTotalClusterTime() { + return totalClusterTime; + } + + public String getActionEx() { + return actionEx; + } + + public boolean isProtected() { + return isProtected; + } + + @JsonSetter("TotalClusterTime") + public void setTotalClusterTime(String totalClusterTime) { + this.totalClusterTime = TimeUtils.convertToMilliseconds(totalClusterTime); + } + + public double getExecuteCost() { + return executeCost; + } + + public double getFileAccessCost() { + return fileAccessCost; + } + + public double getCompileCost() { + return compileCost; + } + +} diff --git a/esp/src/test-ui/tests/framework/model/WUQueryResponse.java b/esp/src/test-ui/tests/framework/model/WUQueryResponse.java new file mode 100644 index 00000000000..a387e516f0f --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/WUQueryResponse.java @@ -0,0 +1,55 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +public class WUQueryResponse { + + @JsonProperty("Type") + private String type; + + @JsonProperty("LogicalFileSearchType") + private String logicalFileSearchType; + + @JsonProperty("Count") + private int count; + + @JsonProperty("PageSize") + private int pageSize; + + @JsonProperty("NextPage") + private int nextPage; + + @JsonProperty("LastPage") + private int lastPage; + + @JsonProperty("NumWUs") + private int numWUs; + + @JsonProperty("First") + private boolean first; + + @JsonProperty("PageStartFrom") + private int pageStartFrom; + + @JsonProperty("PageEndAt") + private int pageEndAt; + + @JsonProperty("Descending") + private boolean descending; + + @JsonProperty("BasicQuery") + private String basicQuery; + + @JsonProperty("Filters") + private String filters; + + @JsonProperty("CacheHint") + private long cacheHint; + + @JsonProperty("Workunits") + private WUWorkunits WUWorkunits; + + public WUWorkunits getWorkunits() { + return WUWorkunits; + } +} diff --git a/esp/src/test-ui/tests/framework/model/WUQueryRoot.java b/esp/src/test-ui/tests/framework/model/WUQueryRoot.java new file mode 100644 index 00000000000..c5a1c54993d --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/WUQueryRoot.java @@ -0,0 +1,12 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +public class WUQueryRoot { + @JsonProperty("WUQueryResponse") + private WUQueryResponse wuQueryResponse; + + public WUQueryResponse getWUQueryResponse() { + return wuQueryResponse; + } +} diff --git a/esp/src/test-ui/tests/framework/model/WUWorkunits.java b/esp/src/test-ui/tests/framework/model/WUWorkunits.java new file mode 100644 index 00000000000..136770b2234 --- /dev/null +++ b/esp/src/test-ui/tests/framework/model/WUWorkunits.java @@ -0,0 +1,15 @@ +package framework.model; + +import com.fasterxml.jackson.annotation.JsonProperty; + +import java.util.List; + +public class WUWorkunits { + + @JsonProperty("ECLWorkunit") + private List WUECLWorkunit; + + public List getECLWorkunit() { + return WUECLWorkunit; + } +} diff --git a/esp/src/test-ui/tests/framework/pages/ActivitiesTest.java b/esp/src/test-ui/tests/framework/pages/ActivitiesTest.java new file mode 100644 index 00000000000..9229481f79a --- /dev/null +++ b/esp/src/test-ui/tests/framework/pages/ActivitiesTest.java @@ -0,0 +1,123 @@ +package framework.pages; + +import framework.config.URLConfig; +import framework.model.NavigationWebElement; +import framework.model.URLMapping; +import framework.utility.Common; +import org.openqa.selenium.By; +import org.openqa.selenium.TimeoutException; +import org.openqa.selenium.WebElement; +import org.testng.annotations.Test; + +import java.util.ArrayList; +import java.util.List; + +import static framework.config.URLConfig.urlMap; + +public class ActivitiesTest { + + static final String[] textArray = {"Target/Wuid", "Graph", "State", "Owner", "Job Name"}; + + @Test + public void testActivitiesPage() { + if(!Common.openWebPage(urlMap.get(URLConfig.NAV_ACTIVITIES).getUrl())){ + return; + } + + Common.logDebug("Tests started for: Activities page."); + + testForAllText(); + + List navWebElements = getNavWebElements(); + + testForNavigationLinks(navWebElements); + + Common.logDebug("Tests finished for: Activities page."); + Common.logDebug("URL Map Generated: " + urlMap); + } + + private void testForNavigationLinks(List navWebElements) { + + Common.logDebug("Tests started for: Activities page: Testing Navigation Links"); + + for (NavigationWebElement element : navWebElements) { + + try { + element.webElement().click(); + + if (testTabsForNavigationLinks(element)) { + String msg = "Success: Navigation Menu Link for " + element.name() + ". URL : " + element.hrefValue(); + Common.logDetail(msg); + } else { + String currentPage = getCurrentPage(); + String errorMsg = "Failure: Navigation Menu Link for " + element.name() + " page failed. The current navigation page that we landed on is " + currentPage + ". Current URL : " + element.hrefValue(); + Common.logError(errorMsg); + } + } catch (Exception ex) { + Common.logException("Failure: Exception in Navigation Link for " + element.name() + ". URL : " + element.hrefValue() + " Error: " + ex.getMessage(), ex); + } + } + } + + private String getCurrentPage() { + + for (var entry : URLConfig.tabsListMap.entrySet()) { + + List tabs = entry.getValue(); + boolean allTabsPresent = true; + for (String tab : tabs) { + if (!Common.driver.getPageSource().contains(tab)) { + allTabsPresent = false; + break; + } + } + + if (allTabsPresent) { + return entry.getKey(); + } + } + + return "Invalid Page"; + } + + private boolean testTabsForNavigationLinks(NavigationWebElement element) { + List tabsList = URLConfig.tabsListMap.get(element.name()); + + for (String tab : tabsList) { + try { + WebElement webElement = Common.waitForElement(By.xpath("//a[text()='" + tab + "']")); + urlMap.get(element.name()).getUrlMappings().put(tab, new URLMapping(tab, webElement.getAttribute("href"))); + } catch (TimeoutException ex) { + return false; + } + } + + return true; + } + + private List getNavWebElements() { + + List navWebElements = new ArrayList<>(); + + for (String navName : URLConfig.navNamesArray) { + try { + WebElement webElement = Common.driver.findElement(By.name(navName)).findElement(By.tagName("a")); + String hrefValue = webElement.getAttribute("href"); + navWebElements.add(new NavigationWebElement(navName, hrefValue, webElement)); + + urlMap.put(navName, new URLMapping(navName, hrefValue)); + } catch (Exception ex) { + Common.logException("Failure: Activities Page for Navigation Element: " + navName + ": Error: " + ex.getMessage(), ex); + } + } + + return navWebElements; + } + + private void testForAllText() { + Common.logDebug("Tests started for: Activities page: Testing Text"); + for (String text : textArray) { + Common.checkTextPresent(text, "Activities Page"); + } + } +} diff --git a/esp/src/test-ui/tests/framework/pages/BaseTableTest.java b/esp/src/test-ui/tests/framework/pages/BaseTableTest.java new file mode 100644 index 00000000000..e53088c9153 --- /dev/null +++ b/esp/src/test-ui/tests/framework/pages/BaseTableTest.java @@ -0,0 +1,587 @@ +package framework.pages; + +import framework.config.Config; +import framework.utility.Common; +import org.openqa.selenium.By; +import org.openqa.selenium.JavascriptExecutor; +import org.openqa.selenium.TimeoutException; +import org.openqa.selenium.WebElement; +import org.openqa.selenium.support.ui.ExpectedConditions; +import org.openqa.selenium.support.ui.WebDriverWait; + +import java.time.Duration; +import java.util.*; +import java.util.function.Function; + + +// This class is the super class for all the those web pages that contains a tabular data for e.g. workunits, files, queries etc. It also contains the tests for their respective details page including testing text, content and tabs. + +public abstract class BaseTableTest { + + protected abstract String getPageName(); + + protected abstract String getPageUrl(); + + protected abstract String getJsonFilePath(); + + protected abstract String getSaveButtonDetailsPage(); + + protected abstract String[] getColumnNames(); + + protected abstract String[] getDetailNames(); + + protected abstract String[] getColumnKeys(); + + protected abstract String[] getDetailKeys(); + + protected abstract String getCheckboxTypeForDetailsPage(); + + protected abstract String getAttributeTypeForDetailsPage(); + + protected abstract String getAttributeValueForDetailsPage(); + + protected abstract String[] getDetailKeysForPageLoad(); + + protected abstract String getUniqueKeyName(); + + protected abstract String getUniqueKey(); + + protected abstract String[] getColumnKeysWithLinks(); + + protected abstract Object parseDataUIValue(Object dataUIValue, Object dataJSONValue, String columnName, Object dataIDUIValue); + + protected abstract Object parseDataJSONValue(Object dataJSONValue, String columnName, Object dataIDUIValue); + + protected abstract List parseJson(String filePath) throws Exception; + + protected abstract Object getColumnDataFromJson(T object, String columnKey); + + protected abstract void sortJsonUsingSortOrder(String currentSortOrder, String columnKey); + + protected abstract String getCurrentPage(); + + protected abstract Map getJsonMap(); + + protected abstract Map> getColumnNamesForTabsDetailsPage(); + + protected abstract String[] getTabValuesForDetailsPage(); + + protected abstract void testDetailSpecificFunctionality(String name, int i); + + protected List jsonObjects; + + protected void testPage() { + if(!Common.openWebPage(getPageUrl())){ + return; + } + + try { + Common.logDebug("Tests started for: " + getPageName() + " page."); + + testForAllText(); + + jsonObjects = getAllObjectsFromJson(); + if (jsonObjects != null) { + + int numOfItemsJSON = jsonObjects.size(); + clickDropdown(numOfItemsJSON); + + testContentAndSortingOrder(); + } + + testLinksInTable(); + + Common.logDebug("Tests finished for: " + getPageName() + " page."); + + } catch (Exception ex) { + Common.logException("Error: " + getPageName() + ": Exception: " + ex.getMessage(), ex); + } + } + + private void testDetailsPage(String name, int i) { + + if (Config.TEST_DETAIL_PAGE_FIELD_NAMES_ALL) { // TEST_DETAIL_PAGE_FIELD_NAMES_ALL = true means the test will run for all items and false means it will only run for the first item + testForAllTextInDetailsPage(name); + } else if (i == 0) { + testForAllTextInDetailsPage(name); + } + + testDetailsContentPage(name); + testDetailSpecificFunctionality(name, i); + + if (Config.TEST_DETAIL_PAGE_TAB_CLICK_ALL) { // TEST_DETAIL_PAGE_TAB_CLICK_ALL = true means the test will run for all items and false means it will only run for the first item + testTabClickOnDetailsPage(name); + } else if (i == 0) { + testTabClickOnDetailsPage(name); + } + } + + private void testTabClickOnDetailsPage(String name) { + + Common.logDebug("Test started for: Tab Click on " + getPageName() + " Details Page. For: " + name); + + waitToLoadDetailsPage(); + + for (var tabValue : getTabValuesForDetailsPage()) { + + try { + WebElement element = Common.driver.findElement(By.xpath("//button[@value='" + tabValue + "']")); + javaScriptElementClick(element); + testPresenceOfColumnNames(getColumnNamesForTabsDetailsPage().get(tabValue), tabValue); + } catch (Exception ex) { + Common.logException("Error: " + getPageName() + " Details Page. Testing tab click on: " + tabValue + " Error: " + ex.getMessage(), ex); + } + } + + } + + protected void javaScriptElementClick(WebElement element) { + + JavascriptExecutor js = (JavascriptExecutor) Common.driver; + js.executeScript("arguments[0].click();", element); + } + + private void testPresenceOfColumnNames(List columnNames, String tabValue) { + + boolean allPresent = true; + String currentURL = Common.driver.getCurrentUrl(); + for (String columnName : columnNames) { + + try { + Common.waitForElement(By.xpath("//*[text()='" + columnName + "']")); + } catch (TimeoutException ex) { + Common.logError("Failure: " + getPageName() + " Details Page. Text : " + columnName + " not present for tab: " + tabValue + ". Current URL: " + currentURL); + allPresent = false; + } + } + + if (allPresent) { + Common.logDetail("Success: " + getPageName() + " Details Page. Tab click on: " + tabValue); + } + } + + protected void clickOnSaveButton() { + + WebElement saveButton = getSaveButtonWebElementDetailsPage(); + Common.waitForElementToBeClickable(saveButton); + saveButton.click(); + + saveButton = getSaveButtonWebElementDetailsPage(); + Common.waitForElementToBeDisabled(saveButton); + } + + private void testLinksInTable() { + + Common.logDebug("Tests started for: " + getPageName() + " page: Testing Links"); + + Common.driver.navigate().refresh(); // refreshing page as page has scrolled to right for testing the sorting functionality of column headers, so bringing it back to normal view by refreshing it + + for (String columnKey : getColumnKeysWithLinks()) { + + List values = getDataFromUIUsingColumnKey(columnKey); + + for (int i = 0; i < values.size(); i++) { + + try { + String name = values.get(i).toString().trim(); + + WebElement element = Common.waitForElement(By.xpath("//div[text()='" + name + "']/..")); + String href = Common.driver.getCurrentUrl(); + + String dropdownValueBefore = getSelectedDropdownValue(); + element.click(); + + if (Common.driver.getPageSource().contains(name)) { + String msg = "Success: " + getPageName() + ": Link Test Pass for " + (i + 1) + ". " + name + ". URL : " + href; + Common.logDetail(msg); + // after the link test has passed, the code tests the details page(including, the text, content, checkbox, description and tabs) + testDetailsPage(name, i); + + } else { + String currentPage = getCurrentPage(); + String errorMsg = "Failure: " + getPageName() + ": Link Test Fail for " + (i + 1) + ". " + name + " page failed. The current navigation page that we landed on is " + currentPage + ". Current URL : " + href; + Common.logError(errorMsg); + } + + Common.driver.navigate().to(getPageUrl()); + Common.driver.navigate().refresh(); + + String dropdownValueAfter = getSelectedDropdownValue(); + + // Log error if the dropdown value has changed + if (!dropdownValueBefore.equals(dropdownValueAfter)) { + String dropdownErrorMsg = "Failure: " + getPageName() + ": Dropdown value changed after navigating back. Before: " + dropdownValueBefore + ", After: " + dropdownValueAfter; + Common.logError(dropdownErrorMsg); + } + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + ": Exception in testing links for column: " + columnKey + " value: " + values.get(i) + " Error: " + ex.getMessage(), ex); + } + } + } + } + + protected void waitToLoadDetailsPage() { + + int waitTimeInSecs = Config.WAIT_TIME_IN_SECONDS; + + boolean allVisible; + + do { + Common.sleepWithTime(waitTimeInSecs); + allVisible = true; + + for (String detailKey : getDetailKeysForPageLoad()) { + WebElement element = Common.waitForElement(By.id(detailKey)); + if ("".equals(element.getAttribute(getAttributeValueForDetailsPage()))) { + allVisible = false; + break; + } + } + + if (allVisible) { + break; + } + + waitTimeInSecs++; + + } while (waitTimeInSecs < Config.WAIT_TIME_THRESHOLD_IN_SECONDS); + + Common.logDebug("Sleep Time In Seconds To Load Details Page: " + waitTimeInSecs); + } + + private void testDetailsContentPage(String name) { + Common.logDebug("Tests started for : " + getPageName() + " Details page: Testing Content: " + getUniqueKeyName() + " : " + name); + try { + waitToLoadDetailsPage(); // sleep until the specific detail fields have loaded + } catch (Exception ex) { + Common.logException("Error: " + getPageName() + ": Exception in waitToLoadDetailsPage: " + getUniqueKeyName() + " : " + name + " Exception: " + ex.getMessage(), ex); + } + + boolean pass = true; + + T object = getJsonMap().get(name); + + for (String detailKey : getDetailKeys()) { + try { + WebElement element = Common.waitForElement(By.id(detailKey)); + + Object dataUIValue, dataJSONValue; + + if (element.getAttribute(getAttributeTypeForDetailsPage()).equals(getCheckboxTypeForDetailsPage())) { + dataUIValue = element.isSelected(); + } else { + dataUIValue = element.getAttribute(getAttributeValueForDetailsPage()); + } + + dataJSONValue = getColumnDataFromJson(object, detailKey); + + dataUIValue = parseDataUIValue(dataUIValue, dataJSONValue, detailKey, name); + dataJSONValue = parseDataJSONValue(dataJSONValue, detailKey, name); + + if (!dataUIValue.equals(dataJSONValue)) { + Common.logError("Failure: " + getPageName() + " Details page, Incorrect " + detailKey + " : " + dataUIValue + " in UI for " + getUniqueKeyName() + " : " + name + ". Correct " + detailKey + " is: " + dataJSONValue); + pass = false; + } + + } catch (Exception ex) { + Common.logException("Error: Details " + getPageName() + "Page, for: " + getUniqueKeyName() + " : " + name + " Error: " + ex.getMessage(), ex); + } + } + + if (pass) { + Common.logDetail("Success: " + getPageName() + " Details page: All content test passed for: " + getUniqueKeyName() + " : " + name); + } + } + + private void testForAllTextInDetailsPage(String name) { + Common.logDebug("Tests started for: " + getPageName() + " Details page: " + getUniqueKeyName() + ": " + name + ": Testing Text"); + for (String text : getDetailNames()) { + Common.checkTextPresent(text, getPageName() + " Details page: " + getUniqueKeyName() + ": " + name); + } + } + + private void testContentAndSortingOrder() { + + Common.logDebug("Tests started for: " + getPageName() + " page: Testing Content"); + + if (testTableContent()) { + + Common.logDebug("Tests started for: " + getPageName() + " page: Testing Sorting Order"); + + for (int i = 0; i < getColumnKeys().length; i++) { + try { + testTheSortingOrderForOneColumn(getColumnKeys()[i], getColumnNames()[i]); + } catch (Exception ex) { + Common.logException("Error: " + getPageName() + ": Exception while testing sort order for column: " + getColumnNames()[i] + ": Exception:" + ex.getMessage(), ex); + } + } + } + } + + private void testTheSortingOrderForOneColumn(String columnKey, String columnName) { + for (int i = 0; i < 3; i++) { + + String currentSortOrder = getCurrentSortingOrder(columnKey); + if (currentSortOrder != null) { + List columnDataFromUI = getDataFromUIUsingColumnKey(columnKey); + + sortJsonUsingSortOrder(currentSortOrder, columnKey); + + List columnDataFromJSON = getDataFromJSONUsingColumnKey(columnKey); + List columnDataIDFromUI = getDataFromUIUsingColumnKey(getUniqueKey()); + + if (compareData(columnDataFromUI, columnDataFromJSON, columnDataIDFromUI, columnName)) { + Common.logDebug("Success: " + getPageName() + ": Values are correctly sorted in " + currentSortOrder + " order by: " + columnName); + } else { + String errMsg = "Failure: " + getPageName() + ": Values are not correctly sorted in " + currentSortOrder + " order by: " + columnName; + Common.logError(errMsg); + } + } else { + Common.logError("Failure: " + getPageName() + " Unable to get sort order for column: " + columnKey); + } + } + } + + private String getCurrentSortingOrder(String columnKey) { + + try { + + WebElement columnHeader = Common.driver.findElement(By.xpath("//*[@role='columnheader' and @*[.='" + columnKey + "']]")); + + // Scroll to bring the column header into view + ((JavascriptExecutor) Common.driver).executeScript("arguments[0].scrollIntoView(true);", columnHeader); + + String oldSortOrder = columnHeader.getAttribute("aria-sort"); + + columnHeader.click(); + + return waitToLoadChangedSortOrder(oldSortOrder, columnKey); + + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + " Exception in getting sort order for column: " + columnKey + " Error: " + ex.getMessage(), ex); + } + + return null; + } + + private String waitToLoadChangedSortOrder(String oldSortOrder, String columnKey) { + + int waitTimeInSecs = Config.WAIT_TIME_IN_SECONDS; + String newSortOrder; + + do { + Common.sleepWithTime(waitTimeInSecs); + + WebElement columnHeaderNew = Common.driver.findElement(By.xpath("//*[@role='columnheader' and @*[.='" + columnKey + "']]")); + + newSortOrder = columnHeaderNew.getAttribute("aria-sort"); + + if (!Objects.equals(newSortOrder, oldSortOrder)) { + break; + } + + waitTimeInSecs++; + + } while (waitTimeInSecs < Config.WAIT_TIME_THRESHOLD_IN_SECONDS); + + return newSortOrder; + } + + private List getDataFromJSONUsingColumnKey(String columnKey) { + List columnDataFromJSON = new ArrayList<>(); + for (T jsonObject : jsonObjects) { + columnDataFromJSON.add(getColumnDataFromJson(jsonObject, columnKey)); + } + return columnDataFromJSON; + } + + protected List getDataFromUIUsingColumnKey(String columnKey) { + + List columnData = new ArrayList<>(); + + try { + + List elements = waitToLoadListOfAllUIObjects(columnKey); + for (WebElement element : elements) { + columnData.add(element.getText().trim()); + } + + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + ": Error in getting data from UI using column key: " + columnKey + "Error: " + ex.getMessage(), ex); + } + + return columnData; + } + + private List waitToLoadListOfAllUIObjects(String columnKey) { + + int waitTimeInSecs = Config.WAIT_TIME_IN_SECONDS; + List elements; + + do { + + elements = Common.driver.findElements(By.xpath("//*[@role='gridcell' and @*[.='" + columnKey + "']]")); + + if (elements.size() >= jsonObjects.size()) { + break; + } + + Common.sleepWithTime(waitTimeInSecs++); + + } while (waitTimeInSecs < Config.WAIT_TIME_THRESHOLD_IN_SECONDS); + + return elements; + } + + @SuppressWarnings("unchecked") + void ascendingSortJson(String columnKey) { + try { + jsonObjects.sort(Comparator.comparing(jsonObject -> (Comparable) getColumnDataFromJson(jsonObject, columnKey))); + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + ": Exception in sorting JSON in ascending order using column key: " + columnKey + " Error: " + ex.getMessage(), ex); + } + } + + @SuppressWarnings("unchecked") + void descendingSortJson(String columnKey) { + try { + jsonObjects.sort(Comparator.comparing((Function>) jsonObject -> (Comparable) getColumnDataFromJson(jsonObject, columnKey)).reversed()); + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + ": Exception in sorting JSON in descending order using column key: " + columnKey + " Error: " + ex.getMessage(), ex); + } + } + + private boolean testTableContent() { + Common.logDebug("Page: " + getPageName() + ": Number of Objects from Json: " + jsonObjects.size()); + + List columnDataIDFromUI = getDataFromUIUsingColumnKey(getUniqueKey()); + + Common.logDebug("Page: " + getPageName() + ": Number of Objects from UI: " + columnDataIDFromUI.size()); + + if (jsonObjects.size() != columnDataIDFromUI.size()) { + String errMsg = "Failure: " + getPageName() + ": Number of items on UI are not equal to the number of items in JSON" + + "\nNumber of Objects from Json: " + jsonObjects.size() + + "\nNumber of Objects from UI: " + columnDataIDFromUI.size(); + Common.logError(errMsg); + return false; + } + + boolean pass = true; + + for (int i = 0; i < getColumnKeys().length; i++) { + List columnDataFromUI = getDataFromUIUsingColumnKey(getColumnKeys()[i]); + List columnDataFromJSON = getDataFromJSONUsingColumnKey(getColumnKeys()[i]); + if (!compareData(columnDataFromUI, columnDataFromJSON, columnDataIDFromUI, getColumnNames()[i])) { + pass = false; + } + } + + return pass; + } + + private List getAllObjectsFromJson() { + String filePath = getJsonFilePath(); + try { + return parseJson(filePath); + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + " Unable to parse JSON File: Exception: " + ex.getMessage(), ex); + } + + return null; + } + + private boolean compareData(List dataUI, List dataJSON, List dataIDUI, String columnName) { + + boolean pass = true; + + for (int i = 0; i < dataUI.size(); i++) { + + Object dataUIValue = dataUI.get(i); + Object dataJSONValue = dataJSON.get(i); + Object dataIDUIValue = dataIDUI.get(i); + + dataUIValue = parseDataUIValue(dataUIValue, dataJSONValue, columnName, dataIDUIValue); + dataJSONValue = parseDataJSONValue(dataJSONValue, columnName, dataIDUIValue); + + if (!checkValues(dataUIValue, dataJSONValue, dataIDUIValue, columnName)) { + pass = false; + } + } + + if (pass) { + Common.logDetail("Success: " + getPageName() + ": Content test passed for column: " + columnName); + } + + return pass; + } + + private boolean checkValues(Object dataUIValue, Object dataJSONValue, Object dataIDUIValue, String columnName) { + if (!dataUIValue.equals(dataJSONValue)) { + String errMsg = "Failure: " + getPageName() + ": Incorrect " + columnName + " : " + dataUIValue + " in UI for " + getUniqueKeyName() + " : " + dataIDUIValue + ". Correct " + columnName + " is: " + dataJSONValue; + Common.logError(errMsg); + return false; + } + + return true; + } + + private void clickDropdown(int numOfItemsJSON) { + + try { + + WebElement dropdown = Common.waitForElement(By.id("pageSize")); + dropdown.click(); + + WebDriverWait wait = new WebDriverWait(Common.driver, Duration.ofSeconds(Config.WAIT_TIME_THRESHOLD_IN_SECONDS)); + + WebElement dropdownList = wait.until(ExpectedConditions.visibilityOfElementLocated(By.id("pageSize-list"))); + + List options = dropdownList.findElements(By.tagName("button")); + + int selectedValue = 0; + + // the smallest dropdown value greater than numOfItemsJSON + + for (WebElement option : options) { + int value = Integer.parseInt(option.getText()); + if (numOfItemsJSON < value) { + option.click(); + selectedValue = value; + break; + } + } + + wait.until(ExpectedConditions.invisibilityOfElementLocated(By.id("pageSize-list"))); + + Common.logDebug("Page: " + getPageName() + ": Dropdown selected: " + selectedValue); + + Common.driver.navigate().refresh(); + Common.sleep(); + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + ": Error in clicking dropdown: " + ex.getMessage(), ex); + } + } + + private String getSelectedDropdownValue() { + try { + WebElement dropdown = Common.waitForElement(By.id("pageSize")); + return dropdown.getText().trim(); + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + ": Error in getting dropdown value: " + ex.getMessage(), ex); + } + + return ""; + } + + protected WebElement getSaveButtonWebElementDetailsPage() { + return Common.waitForElement(By.xpath("//button//*[text()='" + getSaveButtonDetailsPage() + "']/../../..")); + } + + private void testForAllText() { + Common.logDebug("Tests started for: " + getPageName() + " page: Testing Text"); + for (String text : getColumnNames()) { + Common.checkTextPresent(text, getPageName()); + } + } +} + diff --git a/esp/src/test-ui/tests/framework/pages/ECLWorkUnitsTest.java b/esp/src/test-ui/tests/framework/pages/ECLWorkUnitsTest.java new file mode 100644 index 00000000000..ac9185a0685 --- /dev/null +++ b/esp/src/test-ui/tests/framework/pages/ECLWorkUnitsTest.java @@ -0,0 +1,424 @@ +package framework.pages; + +import com.fasterxml.jackson.databind.ObjectMapper; +import framework.config.Config; +import framework.config.URLConfig; +import framework.model.WUECLWorkunit; +import framework.model.WUQueryResponse; +import framework.model.WUQueryRoot; +import framework.utility.Common; +import framework.utility.TimeUtils; +import org.openqa.selenium.By; +import org.openqa.selenium.Keys; +import org.openqa.selenium.WebElement; +import org.testng.annotations.Test; + +import java.io.File; +import java.util.*; + +// This class is a subclass of BaseTableTest, and it includes test cases for ECL Workunits page and along with the tests of workunits details page. + +public class ECLWorkUnitsTest extends BaseTableTest { + + @Test + public void testingECLWorkUnitsPage() { + testPage(); + } + + @Override + protected String getPageName() { + return "ECL Workunits"; + } + + @Override + protected String getPageUrl() { + try { + return URLConfig.urlMap.get(URLConfig.NAV_ECL).getUrlMappings().get(URLConfig.TAB_ECL_WORKUNITS).getUrl(); + } catch (Exception ex){ + Common.logException("Error in getting page URL "+ getPageName() +": Exception: "+ ex.getMessage(), ex); + } + return ""; + } + + @Override + protected String getJsonFilePath() { + return Config.PATH_FOLDER_JSON + Config.WORKUNITS_JSON_FILE_NAME; + } + + @Override + protected String[] getColumnNames() { + return new String[]{"WUID", "Owner", "Job Name", "Cluster", "State", "Total Cluster Time", "Compile Cost", "Execution Cost", "File Access Cost"}; + } + + @Override + protected String[] getColumnKeys() { + return new String[]{"Wuid", "Owner", "Jobname", "Cluster", "State", "TotalClusterTime", "Compile Cost", "Execution Cost", "File Access Cost"}; + } + + private final List badStates = Arrays.asList("compiled", "failed"); + + @Override + protected String getSaveButtonDetailsPage() { + return "Save"; + } + + @Override + protected String[] getDetailNames() { + return new String[]{"WUID", "Action", "State", "Owner", "Job Name", "Description", "Potential Savings", "Compile Cost", "Execution Cost", "File Access Cost", "Protected", "Cluster", "Total Cluster Time", "Aborted by", "Aborted time", "Services"}; + } + + @Override + protected String[] getDetailKeys() { + WebElement element = Common.waitForElement(By.id("state")); + + if (badStates.contains(element.getAttribute(getAttributeValueForDetailsPage()))) { // compiled means it's published but not executed - no wu exists for this, check only these columns + return new String[]{"wuid", "action", "state", "owner", "jobname", "cluster"}; + } else { + return new String[]{"wuid", "action", "state", "owner", "jobname", "compileCost", "executeCost", "fileAccessCost", "protected", "cluster", "totalClusterTime"}; + } + } + + @Override + protected String[] getDetailKeysForPageLoad() { + return new String[]{"wuid", "state", "jobname", "cluster"}; + } + + @Override + protected String getCheckboxTypeForDetailsPage() { + return "checkbox"; + } + + @Override + protected String getAttributeTypeForDetailsPage() { + return "type"; + } + + @Override + protected String getAttributeValueForDetailsPage() { + return "value"; + } + + @Override + protected String[] getColumnKeysWithLinks() { + return new String[]{"Wuid"}; + } + + @Override + protected String getUniqueKeyName() { + return "WUID"; + } + + @Override + protected String getUniqueKey() { + return "Wuid"; + } + + String sortByColumnKeyWhenSortedByNone = "Wuid"; + + protected String[] getTabValuesForDetailsPage() { + return new String[]{"variables", "outputs", "inputs", "metrics", "workflows", + "queries", "resources", "helpers", "xml"}; + } + + @Override + protected Map> getColumnNamesForTabsDetailsPage() { // key in this map is the value attribute of the html element of a tab. + return Map.ofEntries( + Map.entry("variables", List.of("Type", "Name", "Value")), + Map.entry("outputs", List.of("Name", "File Name", "Value", "Views")), + Map.entry("inputs", List.of("Name", "File Cluster", "Usage")), + Map.entry("metrics", List.of("Refresh", "Hot spots", "Timeline", "Options")), // there is no column in this tab, so checking the presence of these buttons + Map.entry("workflows", List.of("Name", "Subtype", "Count", "Remaining")), + Map.entry("queries", List.of("ID", "Priority", "Name", "Target", "WUID", "Dll", "Published By", "Status")), + Map.entry("resources", List.of("Name", "Refresh", "Open", "Preview")), // Preview is not a column, it is a button, keeping it for additional check for Resources tab + Map.entry("helpers", List.of("Type", "Description", "File Size")), + Map.entry("xml", List.of(" costColumns = Arrays.asList("Compile Cost", "Execution Cost", "File Access Cost", "compileCost", "executeCost", "fileAccessCost"); + + @Override + protected Map getJsonMap() { + return jsonMap; + } + + @Override + protected void testDetailSpecificFunctionality(String wuName, int i) { + + if (Config.TEST_WU_DETAIL_PAGE_PROTECTED_ALL) { // TEST_WU_DETAIL_PAGE_PROTECTED_ALL = true means the test will run for all workunits and false means it will only run for the first workunit + testProtectedButtonFunctionality(wuName); + } else if (i == 0) { + testProtectedButtonFunctionality(wuName); + } + + if (Config.TEST_WU_DETAIL_PAGE_DESCRIPTION_ALL) { // TEST_WU_DETAIL_PAGE_DESCRIPTION_ALL = true means the test will run for all workunits and false means it will only run for the first workunit + testDescriptionUpdateFunctionality(wuName); + } else if (i == 0) { + testDescriptionUpdateFunctionality(wuName); + } + } + + private void testDescriptionUpdateFunctionality(String wuName) { + Common.logDebug("Test started for: Description " + getPageName() + " Details Page. For: " + wuName); + + try { + String newDescription = Config.TEST_DESCRIPTION_TEXT; + + String oldDescription = updateDescriptionAndSave(newDescription); + + testIfTheDescriptionUpdated(wuName, newDescription); + + Common.logDetail("Reverting Description Update To Old Value: " + getPageName() + " Details page, for WUID: " + wuName); + + updateDescriptionAndSave(oldDescription); + + } catch (Exception ex) { + Common.logException("Error: " + getPageName() + " Details page for WUID: " + wuName + ", Exception occurred while testing for description. Exception: " + ex.getMessage(), ex); + } + } + + private void testIfTheDescriptionUpdated(String wuName, String newDescription) { + + Common.driver.navigate().refresh(); + + waitToLoadDetailsPage(); + + WebElement element = Common.waitForElement(By.id("description")); + + String updatedDescription = element.getAttribute(getAttributeValueForDetailsPage()).trim(); + + if (newDescription.equals(updatedDescription)) { + Common.logDetail("Success: " + getPageName() + " Details page. Description Updated Successfully on click of save button: Description After refresh showing on UI: " + updatedDescription + ", Updated description was: " + newDescription + " for WUID: " + wuName); + } else { + Common.logError("Failure: " + getPageName() + " Details page. Description Did Not Update on click of save button : Description After refresh showing on UI: " + updatedDescription + ", Updated description should be: " + newDescription + " for WUID: " + wuName); + } + } + + private String updateDescriptionAndSave(String newDescription) { + + waitToLoadDetailsPage(); + + WebElement element = Common.waitForElement(By.id("description")); + + String oldDescription = element.getAttribute(getAttributeValueForDetailsPage()); + + element.sendKeys(Keys.CONTROL + "a"); // Select all text + element.sendKeys(Keys.DELETE); // Delete all selected text + + element.sendKeys(newDescription); + + clickOnSaveButton(); + + element = Common.waitForElement(By.id("description")); + Common.logDetail("Old Description: " + oldDescription + ", Updated Description After Save: " + element.getAttribute(getAttributeValueForDetailsPage())); + + return oldDescription; + } + + private void testProtectedButtonFunctionality(String wuName) { + Common.logDebug("Test started for: Protected checkbox " + getPageName() + " Details Page. For: " + wuName); + + try { + + boolean newCheckboxValue = clickProtectedCheckboxAndSave(wuName); + + checkProtectedStatusOnECLWorkunitsPage(wuName, newCheckboxValue); + + WebElement element = Common.waitForElement(By.xpath("//div[text()='" + wuName + "']/..")); + element.click(); + + if (checkIfNewCheckBoxValuePresentInDetailsPage(newCheckboxValue)) { + Common.logDetail(getPageName() + " Details Page: Reverting the checkbox value for WUID: " + wuName); + clickProtectedCheckboxAndSave(wuName); + } + + } catch (Exception ex) { + Common.logException("Error: ECL WorkUnits: Exception in testing the protected functionality for WUID: " + wuName + ": Error: " + ex.getMessage(), ex); + } + } + + private boolean checkIfNewCheckBoxValuePresentInDetailsPage(boolean newCheckboxValue) { + + waitToLoadDetailsPage(); + + WebElement protectedCheckbox = Common.waitForElement(By.id("protected")); + boolean currentCheckboxValue = protectedCheckbox.isSelected(); + + if (currentCheckboxValue != newCheckboxValue) { + Common.logError("Failure: " + getPageName() + " Details Page: Testing Protected checkbox value after coming back from ECL Workunits page: Updated value: " + newCheckboxValue + ": Showing: " + currentCheckboxValue + " on Workunits Details page"); + return false; + } else { + Common.logDetail("Success: " + getPageName() + " Details Page: Testing Protected checkbox value after coming back from ECL Workunits page: Updated value: " + newCheckboxValue + ": Showing: " + currentCheckboxValue + " on Workunits Details page"); + return true; + } + } + + private boolean clickProtectedCheckboxAndSave(String wuName) { + + waitToLoadDetailsPage(); + + WebElement protectedCheckbox = Common.waitForElement(By.id("protected")); + boolean oldCheckboxValue = protectedCheckbox.isSelected(); + + javaScriptElementClick(protectedCheckbox); + + waitToLoadUpdatedProtectedCheckbox(oldCheckboxValue); + + clickOnSaveButton(); + + boolean newCheckboxValue = protectedCheckbox.isSelected(); + + Common.logDetail("Protected checkbox old Value: " + oldCheckboxValue + ", current value after save: " + newCheckboxValue + " for WUID: " + wuName); + + return newCheckboxValue; + } + + private void waitToLoadUpdatedProtectedCheckbox(boolean oldCheckboxValue) { + + int waitTimeInSecs = Config.WAIT_TIME_IN_SECONDS; + boolean newCheckboxValue; + + do { + Common.sleepWithTime(waitTimeInSecs); + + WebElement protectedCheckbox = Common.waitForElement(By.id("protected")); + newCheckboxValue = protectedCheckbox.isSelected(); + + if (!Objects.equals(newCheckboxValue, oldCheckboxValue)) { + break; + } + + waitTimeInSecs++; + + } while (waitTimeInSecs < Config.WAIT_TIME_THRESHOLD_IN_SECONDS); + } + + private void checkProtectedStatusOnECLWorkunitsPage(String wuName, boolean newCheckboxValue) { + + Common.driver.navigate().to(getPageUrl()); + Common.driver.navigate().refresh(); + + List columnDataWUID = getDataFromUIUsingColumnKey(getUniqueKey()); + List columnDataProtected = getDataFromUIUsingColumnKey("Protected"); + + for (int i = 0; i < columnDataProtected.size(); i++) { + + if (wuName.equals(columnDataWUID.get(i))) { + + String lockStatus = !columnDataProtected.get(i).toString().isEmpty() ? "Locked" : "Unlocked"; + + if (newCheckboxValue && lockStatus.equals("Locked")) { + Common.logDetail("Success: " + getPageName() + " Details Page: Testing Protected checkbox for value: true : Showing Locked on ECL Workunits page"); + } else if (!newCheckboxValue && lockStatus.equals("Unlocked")) { + Common.logDetail("Success: " + getPageName() + " Details Page: Testing Protected checkbox for value: false : Showing Unlocked on ECL Workunits page"); + } else { + Common.logError("Failure: " + getPageName() + " Details Page: Testing Protected checkbox for value: " + newCheckboxValue + ": Showing: " + lockStatus + " on ECL Workunits page"); + } + + break; + } + } + } + + + Map jsonMap = new HashMap<>(); + + @Override + protected List parseJson(String filePath) throws Exception { + ObjectMapper objectMapper = new ObjectMapper(); + WUQueryRoot wuQueryRoot = objectMapper.readValue(new File(filePath), WUQueryRoot.class); + WUQueryResponse wuQueryResponse = wuQueryRoot.getWUQueryResponse(); + List WUECLWorkunits = wuQueryResponse.getWorkunits().getECLWorkunit(); + + for (WUECLWorkunit workunit : WUECLWorkunits) { + jsonMap.put(workunit.getWuid(), workunit); + } + + return WUECLWorkunits; + } + + @Override + protected Object getColumnDataFromJson(WUECLWorkunit workunit, String columnKey) { + return switch (columnKey) { + case "Wuid", "wuid" -> workunit.getWuid(); + case "Owner", "owner" -> workunit.getOwner(); + case "Jobname", "jobname" -> workunit.getJobname(); + case "Cluster", "cluster" -> workunit.getCluster(); + case "State", "state" -> workunit.getState(); + case "TotalClusterTime", "totalClusterTime" -> workunit.getTotalClusterTime(); + case "Compile Cost", "compileCost" -> workunit.getCompileCost(); + case "Execution Cost", "executeCost" -> workunit.getExecuteCost(); + case "File Access Cost", "fileAccessCost" -> workunit.getFileAccessCost(); + case "action" -> workunit.getActionEx(); + case "protected" -> workunit.isProtected(); + default -> null; + }; + } + + @Override + protected Object parseDataUIValue(Object dataUIValue, Object dataJSONValue, String columnName, Object dataIDUIValue) { + try { + if (costColumns.contains(columnName)) { + dataUIValue = Double.parseDouble(((String) dataUIValue).split(" ")[0]); + } else if (columnName.equals("Total Cluster Time") || columnName.equals("totalClusterTime")) { + + if ((long) dataJSONValue == 0 && "".equals(dataUIValue)) { + return 0L; // test ok, if Total Cluster Time is 0 in json and blank in UI - Jira raised - HPCC-32147 + } + + long timeInMilliSecs = TimeUtils.convertToMilliseconds((String) dataUIValue); + if (timeInMilliSecs == Config.MALFORMED_TIME_STRING) { + String errMsg = "Failure: " + getPageName() + ": Incorrect time format for " + columnName + " : " + dataUIValue + " in UI for " + getUniqueKeyName() + " : " + dataIDUIValue; + Common.logError(errMsg); + return dataUIValue; + } + + return timeInMilliSecs; + } else if (dataUIValue instanceof String) { + dataUIValue = ((String) dataUIValue).trim(); + } + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + " Error in parsing UI value: " + dataUIValue + " for column: " + columnName + " ID: " + dataIDUIValue + " Error: " + ex.getMessage(), ex); + } + + return dataUIValue; + } + + @Override + protected Object parseDataJSONValue(Object dataJSONValue, String columnName, Object dataIDUIValue) { + if (columnName.equals("Total Cluster Time") || columnName.equals("totalClusterTime")) { + long timeInMilliSecs = (long) dataJSONValue; + if (timeInMilliSecs == Config.MALFORMED_TIME_STRING) { + String errMsg = "Failure: " + getPageName() + ": Incorrect time format for " + columnName + " in JSON for " + getUniqueKeyName() + " : " + dataIDUIValue; + Common.logError(errMsg); + return dataJSONValue; + } + + return timeInMilliSecs; + } + + return dataJSONValue; + } + + @Override + protected void sortJsonUsingSortOrder(String currentSortOrder, String columnKey) { + switch (currentSortOrder) { + case "ascending" -> ascendingSortJson(columnKey); + case "descending" -> descendingSortJson(columnKey); + case "none" -> descendingSortJson(sortByColumnKeyWhenSortedByNone); + } + } + + @Override + protected String getCurrentPage() { + try { + WebElement element = Common.waitForElement(By.id("wuid")); + if (element != null) { + return element.getAttribute("title"); + } + } catch (Exception ex) { + Common.logException("Error: " + getPageName() + ex.getMessage(), ex); + } + return "Invalid Page"; + } +} diff --git a/esp/src/test-ui/tests/framework/pages/FilesLogicalFilesTest.java b/esp/src/test-ui/tests/framework/pages/FilesLogicalFilesTest.java new file mode 100644 index 00000000000..c16dbb3611b --- /dev/null +++ b/esp/src/test-ui/tests/framework/pages/FilesLogicalFilesTest.java @@ -0,0 +1,200 @@ +package framework.pages; + +import com.fasterxml.jackson.databind.ObjectMapper; +import framework.config.Config; +import framework.config.URLConfig; +import framework.model.DFULogicalFile; +import framework.model.DFUQueryResponse; +import framework.model.DFUQueryRoot; +import framework.utility.Common; +import org.testng.annotations.Test; + +import java.io.File; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +// This class is a subclass of BaseTableTest, and it includes test cases for Files LogicalFiles tab and along with the tests of logical files details page. + +public class FilesLogicalFilesTest extends BaseTableTest { + + @Test + public void testingECLWorkUnitsPage() { + testPage(); + } + + @Override + protected String getPageName() { + return "Files LogicalFiles"; + } + + @Override + protected String getPageUrl() { + try { + return URLConfig.urlMap.get(URLConfig.NAV_FILES).getUrlMappings().get(URLConfig.TAB_FILES_LOGICAL_FILES).getUrl(); + } catch (Exception ex) { + Common.logException("Error in getting page URL " + getPageName() + ": Exception: " + ex.getMessage(), ex); + } + return ""; + } + + @Override + protected String getJsonFilePath() { + return Config.PATH_FOLDER_JSON + Config.FILES_JSON_FILE_NAME; + } + + @Override + protected String[] getColumnNames() { // these are the names of the columns headers displayed on the UI + return new String[]{"Logical Name", "Owner", "Super Owner", "Description", "Cluster", "Records", "Size", "Compressed Size", "Parts", "Min Skew", "Max Skew", "Modified (UTC/GMT)", "Last Accessed", "File Cost At Rest", "File Access Cost"}; + } + + @Override + protected String[] getColumnKeys() { // these are the identifiers used in the HTML code for the respective columns + return new String[]{"Name", "Owner", "SuperOwners", "Description", "NodeGroup", "Records", "FileSize", "CompressedFileSizeString", "Parts", "MinSkew", "MaxSkew", "Modified", "Accessed", "AtRestCost", "AccessCost"}; + } + + @Override + protected String getSaveButtonDetailsPage() { + return "Save"; + } + + @Override + protected String[] getDetailNames() { + return new String[]{}; + } + + @Override + protected String[] getDetailKeys() { + return new String[]{}; + } + + @Override + protected String[] getDetailKeysForPageLoad() { + return new String[]{}; + } + + @Override + protected String getCheckboxTypeForDetailsPage() { + return "checkbox"; + } + + @Override + protected String getAttributeTypeForDetailsPage() { + return "type"; + } + + @Override + protected String getAttributeValueForDetailsPage() { + return "value"; + } + + @Override + protected String[] getColumnKeysWithLinks() { + return new String[]{"Name"}; + } + + @Override + protected String getUniqueKeyName() { + return "Logical Name"; + } + + @Override + protected String getUniqueKey() { + return "Name"; + } + + String sortByColumnKeyWhenSortedByNone = "Modified"; + + protected String[] getTabValuesForDetailsPage() { + return new String[]{}; + } + + @Override + protected Map> getColumnNamesForTabsDetailsPage() { // key in this map is the value attribute of the html element of a tab. + return Map.ofEntries(); + } + + private final List costColumns = Arrays.asList("File Cost At Rest", "File Access Cost", "AtRestCost", "AccessCost"); + + @Override + protected Map getJsonMap() { + return jsonMap; + } + + @Override + protected void testDetailSpecificFunctionality(String wuName, int i) { + } + + Map jsonMap = new HashMap<>(); + + @Override + protected List parseJson(String filePath) throws Exception { + ObjectMapper objectMapper = new ObjectMapper(); + DFUQueryRoot dfuQueryRoot = objectMapper.readValue(new File(filePath), DFUQueryRoot.class); + DFUQueryResponse dfuQueryResponse = dfuQueryRoot.getDFUQueryResponse(); + List dfuLogicalFiles = dfuQueryResponse.getDFULogicalFiles().getDFULogicalFile(); + + for (DFULogicalFile logicalFile : dfuLogicalFiles) { + jsonMap.put(logicalFile.getName(), logicalFile); + } + + return dfuLogicalFiles; + } + + @Override + protected Object getColumnDataFromJson(DFULogicalFile logicalFile, String columnKey) { + return switch (columnKey) { + case "Name" -> logicalFile.getName(); + case "Owner" -> logicalFile.getOwner(); + case "SuperOwners" -> logicalFile.getSuperOwners(); + case "Description" -> logicalFile.getDescription(); + case "NodeGroup" -> logicalFile.getNodeGroup(); + case "Records" -> logicalFile.getRecordCount(); + case "FileSize" -> logicalFile.getIntSize(); + case "CompressedFileSizeString" -> logicalFile.getCompressedFileSize(); + case "Parts" -> logicalFile.getParts(); + case "MinSkew" -> logicalFile.getMinSkew(); + case "MaxSkew" -> logicalFile.getMaxSkew(); + case "Modified" -> logicalFile.getModified(); + case "Accessed" -> logicalFile.getAccessed(); + case "AtRestCost" -> logicalFile.getAtRestCost(); + case "AccessCost" -> logicalFile.getAccessCost(); + default -> null; + }; + } + + @Override + protected Object parseDataUIValue(Object dataUIValue, Object dataJSONValue, String columnName, Object dataIDUIValue) { + try { + if (costColumns.contains(columnName)) { + dataUIValue = Double.parseDouble(((String) dataUIValue).split(" ")[0]); + } else if (dataUIValue instanceof String) { + dataUIValue = ((String) dataUIValue).trim(); + } + } catch (Exception ex) { + Common.logException("Failure: " + getPageName() + " Error in parsing UI value: " + dataUIValue + " for column: " + columnName + " ID: " + dataIDUIValue + " Error: " + ex.getMessage(), ex); + } + + return dataUIValue; + } + + @Override + protected Object parseDataJSONValue(Object dataJSONValue, String columnName, Object dataIDUIValue) { + return dataJSONValue; + } + + @Override + protected void sortJsonUsingSortOrder(String currentSortOrder, String columnKey) { + switch (currentSortOrder) { + case "ascending" -> ascendingSortJson(columnKey); + case "descending" -> descendingSortJson(columnKey); + case "none" -> descendingSortJson(sortByColumnKeyWhenSortedByNone); + } + } + + @Override + protected String getCurrentPage() { + return "Invalid Page"; + } +} diff --git a/esp/src/test-ui/tests/framework/utility/Common.java b/esp/src/test-ui/tests/framework/utility/Common.java new file mode 100644 index 00000000000..17b440a30ea --- /dev/null +++ b/esp/src/test-ui/tests/framework/utility/Common.java @@ -0,0 +1,206 @@ +package framework.utility; + +import framework.config.Config; +import framework.config.URLConfig; +import org.openqa.selenium.By; +import org.openqa.selenium.TimeoutException; +import org.openqa.selenium.WebDriver; +import org.openqa.selenium.WebElement; +import org.openqa.selenium.chrome.ChromeDriver; +import org.openqa.selenium.chrome.ChromeOptions; +import org.openqa.selenium.support.ui.ExpectedConditions; +import org.openqa.selenium.support.ui.WebDriverWait; + +import java.io.IOException; +import java.time.Duration; +import java.util.Arrays; +import java.util.logging.FileHandler; +import java.util.logging.Level; +import java.util.logging.Logger; + +public class Common { + + public static WebDriver driver; + public static Logger errorLogger = setupLogger("error"); + public static Logger exceptionLogger = setupLogger("exception"); + public static Logger specificLogger; + public static int num_exceptions = 0; + public static int num_errors = 0; + + public static void checkTextPresent(String text, String page) { + try { + WebElement element = Common.waitForElement(By.xpath("//*[text()='" + text + "']")); + if (element != null) { + String msg = "Success: " + page + ": Text present: " + text; + logDetail(msg); + } + } catch (TimeoutException ex) { + String errorMsg = "Failure: " + page + ": Text not present: " + text; + logError(errorMsg); + } + } + + public static boolean openWebPage(String url) { + try { + driver.get(url); + driver.manage().window().maximize(); + sleep(); + return true; + } catch (Exception ex) { + Common.logException("Error in opening web page: " + url, ex); + return false; + } + } + + public static void sleep() { + try { + Thread.sleep(Duration.ofSeconds(Config.WAIT_TIME_IN_SECONDS)); + } catch (InterruptedException e) { + Common.logException("Error in sleep: " + e.getMessage(), e); + } + } + + public static void sleepWithTime(int seconds) { + try { + Thread.sleep(Duration.ofSeconds(seconds)); + } catch (InterruptedException e) { + Common.logException("Error in sleep: " + e.getMessage(), e); + } + } + + public static boolean isRunningOnLocal() { + return System.getProperty("os.name").startsWith(Config.LOCAL_OS) && System.getenv("USERPROFILE").startsWith(Config.LOCAL_USER_PROFILE); + } + + public static String getIP() { + + return isRunningOnLocal() ? URLConfig.LOCAL_IP : URLConfig.GITHUB_ACTION_IP; + } + + public static WebElement waitForElement(By locator) { + return new WebDriverWait(driver, Duration.ofSeconds(Config.WAIT_TIME_THRESHOLD_IN_SECONDS)).until(ExpectedConditions.presenceOfElementLocated(locator)); + } + + public static void waitForElementToBeClickable(WebElement element) { + new WebDriverWait(driver, Duration.ofSeconds(Config.WAIT_TIME_THRESHOLD_IN_SECONDS)).until(ExpectedConditions.elementToBeClickable(element)); + } + + // aria-disabled is a standard attribute and is widely used in HTML to indicate whether an element is disabled. It is system-defined, meaning it is part of the standard HTML specifications and not a custom class or attribute that might change frequently. + // By using aria-disabled, you ensure that the check for the disabled state is consistent and less likely to break due to UI changes. Custom class names or attributes defined by developers can change frequently during updates or redesigns, but standard attributes like aria-disabled are much more stable. + + public static void waitForElementToBeDisabled(WebElement element) { + new WebDriverWait(driver, Duration.ofSeconds(Config.WAIT_TIME_THRESHOLD_IN_SECONDS)) + .until(ExpectedConditions.attributeContains(element, "aria-disabled", "true")); + } + + public static void logError(String message) { + System.err.println(message); + errorLogger.severe(message); + num_errors++; + } + + public static void logException(String message, Exception ex) { + + message += Arrays.toString(ex.getStackTrace()); + + System.err.println(message); + exceptionLogger.severe(message); + num_exceptions++; + } + + public static void logDebug(String message) { + System.out.println(message); + + if (specificLogger != null && specificLogger.getLevel() == Level.INFO) { + specificLogger.info(message); + } + + if (specificLogger != null && specificLogger.getLevel() == Level.FINE) { + specificLogger.fine(message); + } + } + + public static void logDetail(String message) { + System.out.println(message); + + if (specificLogger != null && specificLogger.getLevel() == Level.FINE) { + specificLogger.fine(message); + } + } + + public static void initializeLoggerAndDriver() { + specificLogger = setupLogger(Config.LOG_LEVEL); + driver = setupWebDriver(); + } + + public static void printNumOfErrorsAndExceptions() { + System.out.println("Total number of exceptions recorded: " + num_exceptions); + System.out.println("Total number of errors recorded: " + num_errors); + } + + private static WebDriver setupWebDriver() { + + try { + ChromeOptions chromeOptions = new ChromeOptions(); + chromeOptions.addArguments("--headless"); // sets the ChromeDriver to run in headless mode, meaning it runs without opening a visible browser window. + chromeOptions.addArguments("--no-sandbox"); // disables the sandbox security feature in Chrome. + chromeOptions.addArguments("--log-level=3"); // sets the log level for the ChromeDriver. Level 3 corresponds to errors only. + + System.setProperty("webdriver.chrome.silentOutput", "true"); // suppresses the logs generated by the ChromeDriver + + if (Common.isRunningOnLocal()) { + System.setProperty("webdriver.chrome.driver", Config.PATH_LOCAL_CHROME_DRIVER); // sets the system property to the path of the ChromeDriver executable. + } else { + System.setProperty("webdriver.chrome.driver", Config.PATH_GH_ACTION_CHROME_DRIVER); + } + + return new ChromeDriver(chromeOptions); + } catch (Exception ex) { + logException("Failure: Error in setting up web driver: " + ex.getMessage(), ex); + } + + return null; + } + + private static Logger setupLogger(String logLevel) { + + if (logLevel.isEmpty()) { + return null; + } + + Logger logger = Logger.getLogger(logLevel); + logger.setUseParentHandlers(false); // Disable console logging + + try { + FileHandler fileHandler; + CustomFormatter formatter = new CustomFormatter(); + + if (logLevel.equalsIgnoreCase("error")) { + fileHandler = new FileHandler(Config.LOG_FILE_ERROR); + fileHandler.setFormatter(formatter); + logger.addHandler(fileHandler); + logger.setLevel(Level.SEVERE); + } else if (logLevel.equalsIgnoreCase("exception")) { + fileHandler = new FileHandler(Config.LOG_FILE_EXCEPTION); + fileHandler.setFormatter(formatter); + logger.addHandler(fileHandler); + logger.setLevel(Level.SEVERE); + } else if (logLevel.equalsIgnoreCase("debug")) { + fileHandler = new FileHandler(Config.LOG_FILE_DEBUG); + fileHandler.setFormatter(formatter); + logger.addHandler(fileHandler); + logger.setLevel(Level.INFO); + } else if (logLevel.equalsIgnoreCase("detail")) { + fileHandler = new FileHandler(Config.LOG_FILE_DETAIL); + fileHandler.setFormatter(formatter); + logger.addHandler(fileHandler); + logger.setLevel(Level.FINE); + } + } catch (IOException e) { + System.err.println("Failure: Failed to setup logger: " + e.getMessage()); + } + + Logger.getLogger("org.openqa.selenium").setLevel(Level.OFF); // Turn off all logging from the Selenium WebDriver. + return logger; + } +} diff --git a/esp/src/test-ui/tests/framework/utility/CustomFormatter.java b/esp/src/test-ui/tests/framework/utility/CustomFormatter.java new file mode 100644 index 00000000000..2f8a62d7db2 --- /dev/null +++ b/esp/src/test-ui/tests/framework/utility/CustomFormatter.java @@ -0,0 +1,16 @@ +package framework.utility; + +import java.util.logging.Formatter; +import java.util.logging.LogRecord; +import java.text.SimpleDateFormat; +import java.util.Date; + +public class CustomFormatter extends Formatter { + private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); + + @Override + public String format(LogRecord record) { + Date date = new Date(record.getMillis()); + return String.format("%s %s%n", dateFormat.format(date), record.getMessage()); + } +} diff --git a/esp/src/test-ui/tests/framework/utility/TimeUtils.java b/esp/src/test-ui/tests/framework/utility/TimeUtils.java new file mode 100644 index 00000000000..d15f72b2a13 --- /dev/null +++ b/esp/src/test-ui/tests/framework/utility/TimeUtils.java @@ -0,0 +1,63 @@ +package framework.utility; + +import framework.config.Config; + +import java.util.regex.Matcher; +import java.util.regex.Pattern; + +public class TimeUtils { + private static final Pattern TIME_PATTERN = Pattern.compile( + "\\s*(\\d+)\\s+days\\s+(\\d+):(\\d+):(\\d+).(\\d+)|" + + "(\\d+):(\\d+):(\\d+).(\\d+)|" + + "(\\d+):(\\d+).(\\d+)|" + + "(\\d+).(\\d+)\\s*"); + + private static final long MILLISECONDS_IN_A_SECOND = 1000; + private static final long MILLISECONDS_IN_A_MINUTE = 60 * MILLISECONDS_IN_A_SECOND; + private static final long MILLISECONDS_IN_AN_HOUR = 60 * MILLISECONDS_IN_A_MINUTE; + private static final long MILLISECONDS_IN_A_DAY = 24 * MILLISECONDS_IN_AN_HOUR; + + public static long convertToMilliseconds(String time) { + Matcher matcher = TIME_PATTERN.matcher(time); + if (matcher.matches()) { + int days = 0; + int hours = 0; + int minutes = 0; + int seconds = 0; + int milliSeconds = 0; + + try { + if (matcher.group(1) != null) { // Matches d days h:m:s.s + days = Integer.parseInt(matcher.group(1)); + hours = Integer.parseInt(matcher.group(2)); + minutes = Integer.parseInt(matcher.group(3)); + seconds = Integer.parseInt(matcher.group(4)); + milliSeconds = Integer.parseInt(matcher.group(5)); + } else if (matcher.group(6) != null) { // Matches h:m:s.s + hours = Integer.parseInt(matcher.group(6)); + minutes = Integer.parseInt(matcher.group(7)); + seconds = Integer.parseInt(matcher.group(8)); + milliSeconds = Integer.parseInt(matcher.group(9)); + } else if (matcher.group(10) != null) { // Matches m:s.s + minutes = Integer.parseInt(matcher.group(10)); + seconds = Integer.parseInt(matcher.group(11)); + milliSeconds = Integer.parseInt(matcher.group(12)); + } else if (matcher.group(13) != null) { // Matches s.s + seconds = Integer.parseInt(matcher.group(13)); + milliSeconds = Integer.parseInt(matcher.group(14)); + } + } catch (NumberFormatException e) { + return Config.MALFORMED_TIME_STRING; + } + + return days * MILLISECONDS_IN_A_DAY + + hours * MILLISECONDS_IN_AN_HOUR + + minutes * MILLISECONDS_IN_A_MINUTE + + seconds * MILLISECONDS_IN_A_SECOND + + milliSeconds; + + } else { + return Config.MALFORMED_TIME_STRING; + } + } +}