From f878759404fe652a3d86f453108c33fbf72d5a8a Mon Sep 17 00:00:00 2001 From: Edwin Lee Date: Wed, 4 Sep 2024 09:17:31 -0500 Subject: [PATCH 1/3] Disable some GHA workflows, re-enable some Decent builds --- .decent_ci-Linux.yaml | 35 +++++++++ .decent_ci-Windows.yaml | 6 ++ .decent_ci.yaml | 12 ++++ .github/workflows/README.md | 63 ++++++++++++++++ ...l => disabled_test_code_integrity.yml.txt} | 0 ...yml => disabled_test_debug_builds.yml.txt} | 0 .github/workflows/test_develop_commits.yml | 72 +++++++++---------- .github/workflows/test_epjson.yml | 37 ++-------- .github/workflows/test_pull_requests.yml | 30 ++++---- 9 files changed, 174 insertions(+), 81 deletions(-) create mode 100644 .decent_ci-Linux.yaml create mode 100644 .decent_ci-Windows.yaml create mode 100644 .decent_ci.yaml create mode 100644 .github/workflows/README.md rename .github/workflows/{test_code_integrity.yml => disabled_test_code_integrity.yml.txt} (100%) rename .github/workflows/{test_debug_builds.yml => disabled_test_debug_builds.yml.txt} (100%) diff --git a/.decent_ci-Linux.yaml b/.decent_ci-Linux.yaml new file mode 100644 index 00000000000..e0872c4a609 --- /dev/null +++ b/.decent_ci-Linux.yaml @@ -0,0 +1,35 @@ +compilers: + - name: "gcc" + version: "11.4" + cmake_extra_flags: -DLINK_WITH_PYTHON:BOOL=ON -DPython_REQUIRED_VERSION:STRING=3.12.2 -DPython_ROOT_DIR:PATH=~/.pyenv/versions/3.12.2/ -DBUILD_FORTRAN:BOOL=ON -DBUILD_TESTING:BOOL=ON -DENABLE_GTEST_DEBUG_MODE:BOOL=OFF -DBUILD_PERFORMANCE_TESTS:BOOL=ON -DVALGRIND_ANALYZE_PERFORMANCE_TESTS:BOOL=ON -DENABLE_PCH:BOOL=OFF + collect_performance_results: true + skip_regression: true + s3_upload_bucket: energyplus + + - name: "gcc" + version: "11.4" + build_type: RelWithDebInfo + cmake_extra_flags: -DLINK_WITH_PYTHON:BOOL=ON -DPython_REQUIRED_VERSION:STRING=3.12.2 -DPython_ROOT_DIR:PATH=~/.pyenv/versions/3.12.2/ -DBUILD_FORTRAN:BOOL=ON -DBUILD_TESTING:BOOL=ON -DENABLE_REGRESSION_TESTING:BOOL=OFF -DCOMMIT_SHA:STRING=$COMMIT_SHA -DENABLE_COVERAGE:BOOL=ON -DENABLE_GTEST_DEBUG_MODE:BOOL=OFF -DENABLE_PCH:BOOL=OFF + coverage_enabled: true + coverage_base_dir: src/EnergyPlus + coverage_pass_limit: 41.0 + coverage_warn_limit: 40.0 + coverage_s3_bucket: energyplus + build_tag: UnitTestsCoverage + ctest_filter: -E "integration.*" + skip_regression: true + skip_packaging: true + + - name: "gcc" + version: "11.4" + build_type: RelWithDebInfo + cmake_extra_flags: -DLINK_WITH_PYTHON:BOOL=ON -DPython_REQUIRED_VERSION:STRING=3.12.2 -DPython_ROOT_DIR:PATH=~/.pyenv/versions/3.12.2/ -DBUILD_FORTRAN:BOOL=ON -DBUILD_TESTING:BOOL=ON -DENABLE_REGRESSION_TESTING:BOOL=OFF -DCOMMIT_SHA:STRING=$COMMIT_SHA -DENABLE_COVERAGE:BOOL=ON -DENABLE_GTEST_DEBUG_MODE:BOOL=OFF -DENABLE_PCH:BOOL=OFF + coverage_enabled: true + coverage_base_dir: src/EnergyPlus + coverage_pass_limit: 66.0 + coverage_warn_limit: 67.0 + coverage_s3_bucket: energyplus + build_tag: IntegrationCoverage + ctest_filter: -R "integration.*" + skip_regression: true + skip_packaging: true diff --git a/.decent_ci-Windows.yaml b/.decent_ci-Windows.yaml new file mode 100644 index 00000000000..496d53df1f3 --- /dev/null +++ b/.decent_ci-Windows.yaml @@ -0,0 +1,6 @@ +compilers: + - name: Visual Studio + version: 16 + architecture: Win64 + cmake_extra_flags: -DBUILD_FORTRAN:BOOL=ON -DBUILD_TESTING:BOOL=ON -DCOMMIT_SHA=%COMMIT_SHA% -DENABLE_GTEST_DEBUG_MODE:BOOL=OFF -DLINK_WITH_PYTHON=ON -DPython_EXECUTABLE:PATH=C:/Users/elee/AppData/Local/Programs/Python/Python312/python.exe + skip_regression: true diff --git a/.decent_ci.yaml b/.decent_ci.yaml new file mode 100644 index 00000000000..011ef0a2485 --- /dev/null +++ b/.decent_ci.yaml @@ -0,0 +1,12 @@ +results_repository : Myoldmopar/EnergyPlusBuildResults +results_path : _posts +results_base_url : https://myoldmopar.github.io/EnergyPlusBuildResults +regression_repository : NREL/EnergyPlusRegressionTool +regression_branch : BumpToBoto3 # this is the branch of NREL/EnergyPlusRegressionTool to use (usually main) +regression_baseline_default : develop # this is the NREL/EnergyPlus branch to use as the baseline for regressions +regression_baseline_develop : "" +regression_baseline_master : "" +notification_recipients: + - myoldmopar +aging_pull_requests_notification: true +aging_pull_requests_numdays: 28 diff --git a/.github/workflows/README.md b/.github/workflows/README.md new file mode 100644 index 00000000000..7f310775eb4 --- /dev/null +++ b/.github/workflows/README.md @@ -0,0 +1,63 @@ +# Overview + +This folder is, of course, where our GitHub Action workflows live. +I made a significant effort to pull us completely over from Decent CI to GitHub Actions, in PR #10683. +While everything looked good in testing, it seems the number of CI runs was too much for the NREL org. +I reverted it right away, and am going to retry later, probably with the help of some self-hosted runners. +This README is really just a place to drop current status while I figure out what to add back in next. + +# Current Configuration + +- build_checksums.yml + - Runs on demand or on branches named "checksum" +- build_documentation.yml + - Runs on Windows and Linux to test MikTeX and TeXLive + - Linux takes 5 minutes and Windows takes 9 minutes +- build_wheels.yml + - Runs on demand, on Linux only for now +- release_*.yml + - Only works on tags + - Runs on all three platforms +- test_code_integrity.yml + - Runs on Linux, about 22 minutes +- test_debug_builds.yml + - Currently not running at all because of runtime + - Maybe push these onto self-hosted runners soon +- test_develop_commits.yml + - Only running Mac builds, as they are fast + - Then maybe push these onto self-hosted runners for Windows and Ubuntu +- test_epjson.yml + - Currently running just on Windows to save CI time + - In the future, probably expand it back out to all platforms + - Approximately 3 minute runtime +- test_pull_requests.yml + - In this PR I am adding Mac builds back in with new regressions + - In the future expand these onto self-hosted runners for Windows and Ubuntu +- verify_pr_labels.yml + - Runs on Linux, and less than a minute + +Thus for a typical PR push, it will start up: + +- Documentation - Windows: 9 minutes +- Documentation - Linux: 5 minutes +- Code integrity - Linux: 22 minutes +- EpJSON - Windows: 3 minutes +- PR Labeling - Linux: 1 minute +- Test PR Including Regressions - Mac: 49 minutes + +Which is a total of Windows: 12 minutes, Linux: 38 minutes, Mac: 49 minutes. + +Decent CI will be supplementing with: + +- Linux Debug Unit Test Coverage +- Linux Debug Integration Test Coverage +- Linux Release Build and Test Without Regressions +- Windows Release Build and Test Without Regressions + +# TODO + +- Could add some logic in the documentation workflow to only run if docs changed + - on: push: paths: '**.tex' + - But...we would also want to consider other changes that affect docs, like CMake config, versioning, etc. +- Get all docs to ReadTheDocs and eliminate our TeX entirely................................. +- Set up Self-Hosted runners and see how that goes diff --git a/.github/workflows/test_code_integrity.yml b/.github/workflows/disabled_test_code_integrity.yml.txt similarity index 100% rename from .github/workflows/test_code_integrity.yml rename to .github/workflows/disabled_test_code_integrity.yml.txt diff --git a/.github/workflows/test_debug_builds.yml b/.github/workflows/disabled_test_debug_builds.yml.txt similarity index 100% rename from .github/workflows/test_debug_builds.yml rename to .github/workflows/disabled_test_debug_builds.yml.txt diff --git a/.github/workflows/test_develop_commits.yml b/.github/workflows/test_develop_commits.yml index e7669111172..849f171c4e5 100644 --- a/.github/workflows/test_develop_commits.yml +++ b/.github/workflows/test_develop_commits.yml @@ -21,14 +21,14 @@ jobs: fail-fast: false matrix: include: - - os: macos-12 - macos_dev_target: 12.1 - arch: x86_64 - python-arch: x64 - generator: "Unix Makefiles" - nproc: 3 - pretty: "Standard Build on Mac x64" - alternate: false +# - os: macos-12 +# macos_dev_target: 12.1 +# arch: x86_64 +# python-arch: x64 +# generator: "Unix Makefiles" +# nproc: 3 +# pretty: "Standard Build on Mac x64" +# alternate: false - os: macos-14 macos_dev_target: 13.0 arch: arm64 @@ -37,34 +37,34 @@ jobs: nproc: 3 pretty: "Standard Build on Mac arm64" alternate: false - - os: ubuntu-24.04 - arch: x86_64 - python-arch: x64 - generator: "Unix Makefiles" - nproc: 4 - pretty: "Standard Build on Ubuntu 24.04" - alternate: false - - os: windows-2019 - arch: x86_64 - python-arch: x64 - generator: "Visual Studio 16 2019" - nproc: 4 - pretty: "Windows VS 2019" - alternate: false - - os: windows-2022 - arch: x86_64 - python-arch: x64 - generator: "Visual Studio 17 2022" - nproc: 4 - pretty: "Standard Build on Windows VS 2022" - alternate: false - - os: ubuntu-24.04 - arch: x86_64 - python-arch: x64 - generator: "Unix Makefiles" - nproc: 4 - pretty: "Alternate Build on Ubuntu 24.04" - alternate: true +# - os: ubuntu-24.04 +# arch: x86_64 +# python-arch: x64 +# generator: "Unix Makefiles" +# nproc: 4 +# pretty: "Standard Build on Ubuntu 24.04" +# alternate: false +# - os: windows-2019 +# arch: x86_64 +# python-arch: x64 +# generator: "Visual Studio 16 2019" +# nproc: 4 +# pretty: "Windows VS 2019" +# alternate: false +# - os: windows-2022 +# arch: x86_64 +# python-arch: x64 +# generator: "Visual Studio 17 2022" +# nproc: 4 +# pretty: "Standard Build on Windows VS 2022" +# alternate: false +# - os: ubuntu-24.04 +# arch: x86_64 +# python-arch: x64 +# generator: "Unix Makefiles" +# nproc: 4 +# pretty: "Alternate Build on Ubuntu 24.04" +# alternate: true steps: diff --git a/.github/workflows/test_epjson.yml b/.github/workflows/test_epjson.yml index 97ce25a0959..12aebf2c2bc 100644 --- a/.github/workflows/test_epjson.yml +++ b/.github/workflows/test_epjson.yml @@ -12,19 +12,8 @@ env: jobs: release: - name: Testing on ${{ matrix.pretty }} - runs-on: ${{ matrix.os }} - strategy: - # fail-fast: Default is true, switch to false to allow one platform to fail and still run others - fail-fast: false - matrix: - include: - - os: ubuntu-latest - pretty: "Ubuntu" - - os: windows-latest - pretty: "Windows" - - os: macos-latest - pretty: "Mac" + name: Testing on Windows + runs-on: windows-latest steps: - uses: actions/checkout@v4 @@ -34,24 +23,13 @@ jobs: with: python-version: '3.10' - - name: Setup + - name: Install Pip Stuff shell: bash - run: | - cmake -E make_directory ./build/ - if [ "$RUNNER_OS" == "Windows" ]; then - echo "Setting CMAKE_GENERATOR options equivalent to ='-G \"Visual Studio 17 2022\" -A x64'" - echo CMAKE_GENERATOR='Visual Studio 17 2022' >> $GITHUB_ENV - echo CMAKE_GENERATOR_PLATFORM=x64 >> $GITHUB_ENV - choco install ninja - elif [ "$RUNNER_OS" == "macOS" ]; then - echo MACOSX_DEPLOYMENT_TARGET=12.1 >> $GITHUB_ENV - fi; - pip install pytest pytest-timeout + run: pip install pytest pytest-timeout - name: Configure CMake - working-directory: ./build - run: | - cmake ../ + working-directory: ./build # TODO: Wait...it will automatically create it? + run: cmake -G "Visual Studio 17 2022" -A x64 ../ - name: Test epjson shell: python @@ -123,5 +101,4 @@ jobs: - name: Run idd_schema pytests working-directory: ./build - run: | - pytest --verbose ../idd + run: pytest --verbose ../idd diff --git a/.github/workflows/test_pull_requests.yml b/.github/workflows/test_pull_requests.yml index 8f58f736b18..1a3fc03c513 100644 --- a/.github/workflows/test_pull_requests.yml +++ b/.github/workflows/test_pull_requests.yml @@ -2,7 +2,7 @@ name: Build and Test on: pull_request: - branches: [ develop ] # TODO: Run this on any PR, and compare to baseline branch, not necessarily develop. Also perhaps non-draft PRs only. + branches: [ develop ] defaults: run: @@ -31,20 +31,20 @@ jobs: nproc: 3 run_regressions: true pretty: "Mac arm64" - - os: ubuntu-24.04 - arch: x86_64 - python-arch: x64 - generator: "Unix Makefiles" - nproc: 4 - run_regressions: false - pretty: "Ubuntu 24.04" - - os: windows-2022 - arch: x86_64 - python-arch: x64 - generator: "Visual Studio 17 2022" - nproc: 4 - run_regressions: false - pretty: "Windows x64" +# - os: ubuntu-24.04 +# arch: x86_64 +# python-arch: x64 +# generator: "Unix Makefiles" +# nproc: 4 +# run_regressions: false +# pretty: "Ubuntu 24.04" +# - os: windows-2022 +# arch: x86_64 +# python-arch: x64 +# generator: "Visual Studio 17 2022" +# nproc: 4 +# run_regressions: false +# pretty: "Windows x64" steps: From 5b243ce4b8f3c546e933c56a9667404b9fb3dbd0 Mon Sep 17 00:00:00 2001 From: Edwin Lee Date: Wed, 4 Sep 2024 09:38:13 -0500 Subject: [PATCH 2/3] Bringing in remaining changes from MacBuildsOnGHA branch --- scripts/dev/add_to_project.sh | 36 -- scripts/dev/build_regression_summary.py | 89 +++ scripts/dev/gha_coverage_summary.py | 81 +++ scripts/dev/gha_regressions.py | 508 ++++++++++++++++++ src/EnergyPlus/DataSystemVariables.cc | 15 +- src/EnergyPlus/DataSystemVariables.hh | 1 + src/EnergyPlus/SimulationManager.cc | 3 + third_party/.gitignore | 1 + .../unit/CommandLineInterface.unit.cc | 2 +- tst/EnergyPlus/unit/Timer.unit.cc | 4 +- 10 files changed, 690 insertions(+), 50 deletions(-) delete mode 100755 scripts/dev/add_to_project.sh create mode 100644 scripts/dev/build_regression_summary.py create mode 100644 scripts/dev/gha_coverage_summary.py create mode 100644 scripts/dev/gha_regressions.py diff --git a/scripts/dev/add_to_project.sh b/scripts/dev/add_to_project.sh deleted file mode 100755 index e5fc0210161..00000000000 --- a/scripts/dev/add_to_project.sh +++ /dev/null @@ -1,36 +0,0 @@ -#!/bin/bash - -# call with the PR number as the only command line argument - -# get the PR num from the command line argument -PR_NUM=$1 - -# the GraphQL Project ID can be retrieved from a given organization's project, where the URL is: -# https://github.com/orgs/ORGANIZATION/projects/SOME_PROJECT_NUMBER/views/2 -# and the associated call to graphql is: -#gh api graphql -f query=' -# query{ -# organization(login: "ORGANIZATION"){ -# projectV2(number: SOME_PROJECT_NUMBER) { -# id -# } -# } -# }' -# TODO: Just specify the project organization and number and get the graphql ID in here -PROJ_ID=PVT_kwDOAB0YcM4AEWD7 -echo "Using PR Num as ${PR_NUM} and project ID as: ${PROJ_ID}" - -# get the current PR ID based on the this checkout -CONTENT=$(gh pr view "$PR_NUM" --json 'id' --jq '.id') -echo "Found PR node ID as: ${CONTENT}" - -# use the gh api command line to act on the Projects-v2 API and add the PR as a new card -# should also add more arguments for the column to use, etc. -gh api graphql -f query=" - mutation { - addProjectV2ItemById(input: {projectId: \"${PROJ_ID}\" contentId: \"${CONTENT}\"}) { - item { - id - } - } - }" diff --git a/scripts/dev/build_regression_summary.py b/scripts/dev/build_regression_summary.py new file mode 100644 index 00000000000..641a228fa3a --- /dev/null +++ b/scripts/dev/build_regression_summary.py @@ -0,0 +1,89 @@ +#!/usr/bin/env python +# EnergyPlus, Copyright (c) 1996-2024, The Board of Trustees of the University +# of Illinois, The Regents of the University of California, through Lawrence +# Berkeley National Laboratory (subject to receipt of any required approvals +# from the U.S. Dept. of Energy), Oak Ridge National Laboratory, managed by UT- +# Battelle, Alliance for Sustainable Energy, LLC, and other contributors. All +# rights reserved. +# +# NOTICE: This Software was developed under funding from the U.S. Department of +# Energy and the U.S. Government consequently retains certain rights. As such, +# the U.S. Government has been granted for itself and others acting on its +# behalf a paid-up, nonexclusive, irrevocable, worldwide license in the +# Software to reproduce, distribute copies to the public, prepare derivative +# works, and perform publicly and display publicly, and to permit others to do +# so. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# (1) Redistributions of source code must retain the above copyright notice, +# this list of conditions and the following disclaimer. +# +# (2) Redistributions in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# (3) Neither the name of the University of California, Lawrence Berkeley +# National Laboratory, the University of Illinois, U.S. Dept. of Energy nor +# the names of its contributors may be used to endorse or promote products +# derived from this software without specific prior written permission. +# +# (4) Use of EnergyPlus(TM) Name. If Licensee (i) distributes the software in +# stand-alone form without changes from the version obtained under this +# License, or (ii) Licensee makes a reference solely to the software +# portion of its product, Licensee must refer to the software as +# "EnergyPlus version X" software, where "X" is the version number Licensee +# obtained under this License and may not use a different name for the +# software. Except as specifically required in this Section (4), Licensee +# shall not use in a company name, a product name, in advertising, +# publicity, or other promotional activities any name, trade name, +# trademark, logo, or other designation of "EnergyPlus", "E+", "e+" or +# confusingly similar designation, without the U.S. Department of Energy's +# prior written consent. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE +# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE +# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. + +from sys import argv + +summary_input_md_file = argv[1] +summary_output_js_file = argv[2] +matrix_os = argv[3] +github_sha = argv[4] +github_run_id = argv[5] +artifact_url = argv[6] + +with open(summary_input_md_file) as md: + md_contents = md.read() + +fixed_up_contents = f""" +### :warning: Regressions detected on {matrix_os} for commit {github_sha} + +{md_contents} + + - [View Results](https://github.com/NREL/EnergyPlus/actions/runs/{github_run_id}) + - [Download Regressions]({artifact_url}) +""" + +with open(summary_output_js_file, 'w') as js: + js_contents = f""" +module.exports = ({{github, context}}) => {{ + github.rest.issues.createComment({{ + issue_number: context.issue.number, + owner: context.repo.owner, + repo: context.repo.repo, + body: `{fixed_up_contents}` + }}) +}} +""" + js.write(js_contents) diff --git a/scripts/dev/gha_coverage_summary.py b/scripts/dev/gha_coverage_summary.py new file mode 100644 index 00000000000..34bb39dcc3c --- /dev/null +++ b/scripts/dev/gha_coverage_summary.py @@ -0,0 +1,81 @@ +#!/usr/bin/env python +# EnergyPlus, Copyright (c) 1996-2024, The Board of Trustees of the University +# of Illinois, The Regents of the University of California, through Lawrence +# Berkeley National Laboratory (subject to receipt of any required approvals +# from the U.S. Dept. of Energy), Oak Ridge National Laboratory, managed by UT- +# Battelle, Alliance for Sustainable Energy, LLC, and other contributors. All +# rights reserved. +# +# NOTICE: This Software was developed under funding from the U.S. Department of +# Energy and the U.S. Government consequently retains certain rights. As such, +# the U.S. Government has been granted for itself and others acting on its +# behalf a paid-up, nonexclusive, irrevocable, worldwide license in the +# Software to reproduce, distribute copies to the public, prepare derivative +# works, and perform publicly and display publicly, and to permit others to do +# so. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# (1) Redistributions of source code must retain the above copyright notice, +# this list of conditions and the following disclaimer. +# +# (2) Redistributions in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# (3) Neither the name of the University of California, Lawrence Berkeley +# National Laboratory, the University of Illinois, U.S. Dept. of Energy nor +# the names of its contributors may be used to endorse or promote products +# derived from this software without specific prior written permission. +# +# (4) Use of EnergyPlus(TM) Name. If Licensee (i) distributes the software in +# stand-alone form without changes from the version obtained under this +# License, or (ii) Licensee makes a reference solely to the software +# portion of its product, Licensee must refer to the software as +# "EnergyPlus version X" software, where "X" is the version number Licensee +# obtained under this License and may not use a different name for the +# software. Except as specifically required in this Section (4), Licensee +# shall not use in a company name, a product name, in advertising, +# publicity, or other promotional activities any name, trade name, +# trademark, logo, or other designation of "EnergyPlus", "E+", "e+" or +# confusingly similar designation, without the U.S. Department of Energy's +# prior written consent. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE +# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE +# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. + +# expecting to find a file called cover.txt in cwd +# need to generate a cover.md file in cwd +# cover.txt looks like: +# lines=48 hit=2 functions=2 hit=1 +# Processing file EnergyPlus/SurfaceGeometry.hh +# lines=44 hit=9 functions=4 hit=2 +# Overall coverage rate: +# lines......: 7.9% (28765 of 364658 lines) +# functions......: 19.6% (2224 of 11327 functions) + +from pathlib import Path +cover_input = Path.cwd() / 'cover.txt' +lines = cover_input.read_text().strip().split('\n') +line_coverage = lines[-2].strip().split(':')[1].strip() +line_percent = line_coverage.split(' ')[0] +function_coverage = lines[-1].strip().split(':')[1].strip() +cover_output = Path.cwd() / 'cover.md' +content = f""" +
+ Coverage Summary - {line_percent} of lines - Download Coverage Artifact for Full Details + + - {line_coverage} + - {function_coverage} +
""" +cover_output.write_text(content) diff --git a/scripts/dev/gha_regressions.py b/scripts/dev/gha_regressions.py new file mode 100644 index 00000000000..b3de9964190 --- /dev/null +++ b/scripts/dev/gha_regressions.py @@ -0,0 +1,508 @@ +#!/usr/bin/env python +# EnergyPlus, Copyright (c) 1996-2024, The Board of Trustees of the University +# of Illinois, The Regents of the University of California, through Lawrence +# Berkeley National Laboratory (subject to receipt of any required approvals +# from the U.S. Dept. of Energy), Oak Ridge National Laboratory, managed by UT- +# Battelle, Alliance for Sustainable Energy, LLC, and other contributors. All +# rights reserved. +# +# NOTICE: This Software was developed under funding from the U.S. Department of +# Energy and the U.S. Government consequently retains certain rights. As such, +# the U.S. Government has been granted for itself and others acting on its +# behalf a paid-up, nonexclusive, irrevocable, worldwide license in the +# Software to reproduce, distribute copies to the public, prepare derivative +# works, and perform publicly and display publicly, and to permit others to do +# so. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are met: +# +# (1) Redistributions of source code must retain the above copyright notice, +# this list of conditions and the following disclaimer. +# +# (2) Redistributions in binary form must reproduce the above copyright notice, +# this list of conditions and the following disclaimer in the documentation +# and/or other materials provided with the distribution. +# +# (3) Neither the name of the University of California, Lawrence Berkeley +# National Laboratory, the University of Illinois, U.S. Dept. of Energy nor +# the names of its contributors may be used to endorse or promote products +# derived from this software without specific prior written permission. +# +# (4) Use of EnergyPlus(TM) Name. If Licensee (i) distributes the software in +# stand-alone form without changes from the version obtained under this +# License, or (ii) Licensee makes a reference solely to the software +# portion of its product, Licensee must refer to the software as +# "EnergyPlus version X" software, where "X" is the version number Licensee +# obtained under this License and may not use a different name for the +# software. Except as specifically required in this Section (4), Licensee +# shall not use in a company name, a product name, in advertising, +# publicity, or other promotional activities any name, trade name, +# trademark, logo, or other designation of "EnergyPlus", "E+", "e+" or +# confusingly similar designation, without the U.S. Department of Energy's +# prior written consent. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE +# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE +# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE +# POSSIBILITY OF SUCH DAMAGE. +from collections import defaultdict +from datetime import datetime, UTC +import json +from shutil import copy +from pathlib import Path +import sys +from shutil import rmtree +from zoneinfo import ZoneInfo + +from energyplus_regressions.runtests import SuiteRunner +from energyplus_regressions.structures import TextDifferences, TestEntry, EndErrSummary + + +class RegressionManager: + + def __init__(self): + self.root_index_files_no_diff = [] + self.root_index_files_diffs = [] + self.diffs_by_idf = defaultdict(list) + self.diffs_by_type = defaultdict(list) + self.summary_results = {} + self.num_idf_inspected = 0 + # self.all_files_compared = [] TODO: need to get this from regression runner + import energyplus_regressions + self.threshold_file = str(Path(energyplus_regressions.__file__).parent / 'diffs' / 'math_diff.config') + + def single_file_regressions(self, baseline: Path, modified: Path) -> [TestEntry, bool]: + + idf = baseline.name + self.num_idf_inspected += 1 + this_file_diffs = [] + + entry = TestEntry(idf, "") + entry, message = SuiteRunner.process_diffs_for_one_case( + entry, + {'build_dir': str(baseline)}, + {'build_dir': str(modified)}, + "", + self.threshold_file, + ci_mode=True + ) # returns an updated entry + self.summary_results[idf] = entry.summary_result + + has_diffs = False + + text_diff_results = { + "Audit": entry.aud_diffs, + "BND": entry.bnd_diffs, + "DELightIn": entry.dl_in_diffs, + "DELightOut": entry.dl_out_diffs, + "DXF": entry.dxf_diffs, + "EIO": entry.eio_diffs, + "ERR": entry.err_diffs, + "Readvars_Audit": entry.readvars_audit_diffs, + "EDD": entry.edd_diffs, + "WRL": entry.wrl_diffs, + "SLN": entry.sln_diffs, + "SCI": entry.sci_diffs, + "MAP": entry.map_diffs, + "DFS": entry.dfs_diffs, + "SCREEN": entry.screen_diffs, + "GLHE": entry.glhe_diffs, + "MDD": entry.mdd_diffs, + "MTD": entry.mtd_diffs, + "RDD": entry.rdd_diffs, + "SHD": entry.shd_diffs, + "PERF_LOG": entry.perf_log_diffs, + "IDF": entry.idf_diffs, + "StdOut": entry.stdout_diffs, + "StdErr": entry.stderr_diffs, + } + for diff_type, diffs in text_diff_results.items(): + if diffs is None: + continue + if diffs.diff_type != TextDifferences.EQUAL: + has_diffs = True + this_file_diffs.append(diff_type) + self.diffs_by_type[diff_type].append(idf) + self.diffs_by_idf[idf].append(diff_type) + + numeric_diff_results = { + "ESO": entry.eso_diffs, + "MTR": entry.mtr_diffs, + "SSZ": entry.ssz_diffs, + "ZSZ": entry.zsz_diffs, + "JSON": entry.json_diffs, + } + for diff_type, diffs in numeric_diff_results.items(): + if diffs is None: + continue + if diffs.diff_type == 'Big Diffs': + has_diffs = True + this_file_diffs.append(f"{diff_type} Big Diffs") + self.diffs_by_type[f"{diff_type} Big Diffs"].append(idf) + self.diffs_by_idf[idf].append(f"{diff_type} Big Diffs") + elif diffs.diff_type == 'Small Diffs': + has_diffs = True + this_file_diffs.append(f"{diff_type} Small Diffs") + self.diffs_by_type[f"{diff_type} Small Diffs"].append(idf) + self.diffs_by_idf[idf].append(f"{diff_type} Small Diffs") + + if entry.table_diffs: + if entry.table_diffs.big_diff_count > 0: + has_diffs = True + this_file_diffs.append("Table Big Diffs") + self.diffs_by_type["Table Big Diffs"].append(idf) + self.diffs_by_idf[idf].append("Table Big Diffs") + elif entry.table_diffs.small_diff_count > 0: + has_diffs = True + this_file_diffs.append("Table Small Diffs") + self.diffs_by_type["Table Small Diffs"].append(idf) + self.diffs_by_idf[idf].append("Table Small Diffs") + if entry.table_diffs.string_diff_count > 1: # There's always one...the time stamp + has_diffs = True + this_file_diffs.append("Table String Diffs") + self.diffs_by_type["Table String Diffs"].append(idf) + self.diffs_by_idf[idf].append("Table String Diffs") + + return entry, has_diffs + + @staticmethod + def single_diff_html(contents: str) -> str: + return f""" + + + + + + + + +
+   
+    {contents}
+   
+  
+ + +""" + + @staticmethod + def regression_row_in_single_test_case_html(diff_file_name: str) -> str: + return f""" + + {diff_file_name} + download + view + """ + + @staticmethod + def single_test_case_html(contents: str) -> str: + return f""" + + + + + + + + + + + + + + + + +{contents} +
filename
+ +""" + + def bundle_root_index_html(self, header_info: list[str]) -> str: + # set up header table + header_content = "" + for hi in header_info: + header_content += f"""
  • {hi}
  • \n""" + + # set up diff summary listings + num_no_diff = len(self.root_index_files_no_diff) + nds = 's' if num_no_diff == 0 or num_no_diff > 1 else '' + no_diff_content = "" + for nd in self.root_index_files_no_diff: + no_diff_content += f"""
  • {nd}
  • \n""" + num_diff = len(self.root_index_files_diffs) + ds = 's' if num_diff == 0 or num_diff > 1 else '' + diff_content = "" + for d in self.root_index_files_diffs: + diff_content += f"""{d}\n""" + + # set up diff type listing + diff_type_keys = sorted(self.diffs_by_type.keys()) + num_diff_types = len(diff_type_keys) + dt = 's' if num_diff_types == 0 or num_diff_types > 1 else '' + diff_type_content = "" + if num_diff_types > 0: + for k in diff_type_keys: + nice_type_key = k.lower().replace(' ', '') + diffs_this_type = self.diffs_by_type[k] + num_files_this_type = len(diffs_this_type) + dtt = 's' if num_diff_types == 0 or num_diff_types > 1 else '' + this_diff_type_list = "" + for idf in diffs_this_type: + this_diff_type_list += f"""{idf}\n""" + diff_type_content += f""" +
    +
    + +
    +
    +
      +{this_diff_type_list} +
    +
    +
    +
    +
    """ + + # set up runtime results table + run_time_rows_text = "" + sum_base_seconds = 0 + sum_branch_seconds = 0 + sorted_idf_keys = sorted(self.summary_results.keys()) + for idf in sorted_idf_keys: + summary = self.summary_results[idf] + case_1_success = summary.simulation_status_case1 == EndErrSummary.STATUS_SUCCESS + case_2_success = summary.simulation_status_case2 == EndErrSummary.STATUS_SUCCESS + if case_1_success: + base_time = summary.run_time_seconds_case1 + else: + base_time = "N/A" + if case_1_success: + branch_time = summary.run_time_seconds_case2 + else: + branch_time = "N/A" + if case_1_success and case_2_success: + sum_base_seconds += base_time + sum_branch_seconds += branch_time + + run_time_rows_text += f"""{idf}{base_time}{branch_time}""" + run_time_rows_text += f"""Runtime Total (Successes){sum_base_seconds:.1f}{sum_branch_seconds:.1f}""" + + return f""" + + + + + + + + + + +
    + +

    EnergyPlus Regressions

    + +
    +
    + + +
    +
    + +
    + +

    Summary by File

    + +
    +
    + +
    +
    +
      +{no_diff_content} +
    +
    +
    +
    +
    + +
    +
    + +
    +
    +
      +{diff_content} +
    +
    +
    +
    +
    + +
    + +

    Summary by Diff Type

    + +
    +
    + +
    +
    +
      +{diff_type_content} +
    +
    +
    +
    +
    + +
    + +

    Run Times

    + +
    +
    + +
    +
    + + + + + + +{run_time_rows_text} +
    FilenameBase Case Runtime (seconds)Branch Case Runtime (seconds)
    +
    +
    +
    +
    + +
    + + +""" + + def generate_markdown_summary(self, bundle_root: Path): + diff_lines = "" + for diff_type, idfs in self.diffs_by_type.items(): + diff_lines += f" - {diff_type}: {len(idfs)}\n" + content = f""" +
    + Regression Summary + +{diff_lines} +
    """ + (bundle_root / 'summary.md').write_text(content) + + def check_all_regressions(self, base_testfiles: Path, mod_testfiles: Path, bundle_root: Path) -> bool: + any_diffs = False + bundle_root.mkdir(exist_ok=True) + entries = sorted(base_testfiles.iterdir()) + for entry_num, baseline in enumerate(entries): + if not baseline.is_dir(): + continue + if baseline.name == 'CMakeFiles': # add more ignore dirs here + continue + modified = mod_testfiles / baseline.name + if not modified.exists(): + continue # TODO: Should we warn that it is missing? + entry, diffs = self.single_file_regressions(baseline, modified) + if diffs: + self.root_index_files_diffs.append(baseline.name) + any_diffs = True + potential_diff_files = baseline.glob("*.*.*") # TODO: Could try to get this from the regression tool + target_dir_for_this_file_diffs = bundle_root / baseline.name + if potential_diff_files: + if target_dir_for_this_file_diffs.exists(): + rmtree(target_dir_for_this_file_diffs) + target_dir_for_this_file_diffs.mkdir() + index_contents_this_file = "" + for potential_diff_file in potential_diff_files: + copy(potential_diff_file, target_dir_for_this_file_diffs) + diff_file_with_html = target_dir_for_this_file_diffs / (potential_diff_file.name + '.html') + if potential_diff_file.name.endswith('.htm'): + # already a html file, just upload the raw contents but renamed as ...htm.html + copy(potential_diff_file, diff_file_with_html) + else: + # it's not an HTML file, wrap it inside an HTML wrapper in a temp file and send it + contents = potential_diff_file.read_text() + wrapped_contents = self.single_diff_html(contents) + diff_file_with_html.write_text(wrapped_contents) + index_contents_this_file += self.regression_row_in_single_test_case_html(potential_diff_file.name) + index_file = target_dir_for_this_file_diffs / 'index.html' + index_this_file = self.single_test_case_html(index_contents_this_file) + index_file.write_text(index_this_file) + else: + self.root_index_files_no_diff.append(baseline.name) + so_far = ' Diffs! ' if any_diffs else 'No diffs' + if entry_num % 40 == 0: + print(f"On file #{entry_num}/{len(entries)} ({baseline.name}), Diff status so far: {so_far}") + meta_data = [ + f"Regression time stamp in UTC: {datetime.now(UTC)}", + f"Regression time stamp in Central Time: {datetime.now(ZoneInfo('America/Chicago'))}", + f"Number of input files evaluated: {self.num_idf_inspected}", + ] + bundle_root_index_file_path = bundle_root / 'index.html' + bundle_root_index_content = self.bundle_root_index_html(meta_data) + bundle_root_index_file_path.write_text(bundle_root_index_content) + print() + print(f"* Files with Diffs *:\n{"\n ".join(self.root_index_files_diffs)}\n") + print(f"* Diffs by File *:\n{json.dumps(self.diffs_by_idf, indent=2, sort_keys=True)}\n") + print(f"* Diffs by Type *:\n{json.dumps(self.diffs_by_type, indent=2, sort_keys=True)}\n") + if any_diffs: + self.generate_markdown_summary(bundle_root) + # print("::warning title=Regressions::Diffs Detected") + return any_diffs + + +if __name__ == "__main__": # pragma: no cover - testing function, not the __main__ entry point + + if len(sys.argv) != 4: + print("syntax: %s base_dir mod_dir regression_dir" % sys.argv[0]) + sys.exit(1) + arg_base_dir = Path(sys.argv[1]) + arg_mod_dir = Path(sys.argv[2]) + arg_regression_dir = Path(sys.argv[3]) + rm = RegressionManager() + response = rm.check_all_regressions(arg_base_dir, arg_mod_dir, arg_regression_dir) + sys.exit(1 if response else 0) diff --git a/src/EnergyPlus/DataSystemVariables.cc b/src/EnergyPlus/DataSystemVariables.cc index def6ab83745..dd2100e87d6 100644 --- a/src/EnergyPlus/DataSystemVariables.cc +++ b/src/EnergyPlus/DataSystemVariables.cc @@ -111,17 +111,7 @@ namespace DataSystemVariables { constexpr const char * cDisplayInputInAuditEnvVar("DISPLAYINPUTINAUDIT"); // environmental variable that enables the echoing of the input file into the audit file - // DERIVED TYPE DEFINITIONS - // na - - // INTERFACE BLOCK SPECIFICATIONS - // na - - // MODULE VARIABLE DECLARATIONS: - - // Shading methods - - // Functions + constexpr const char *ciForceTimeStepEnvVar("CI_FORCE_TIME_STEP"); // environment var forcing 30 minute time steps on CI for efficiency fs::path CheckForActualFilePath(EnergyPlusData &state, fs::path const &originalInputFilePath, // path (or filename only) as input for object @@ -311,6 +301,9 @@ namespace DataSystemVariables { get_environment_variable(cDisplayInputInAuditEnvVar, cEnvValue); if (!cEnvValue.empty()) state.dataGlobal->DisplayInputInAudit = env_var_on(cEnvValue); // Yes or True + + get_environment_variable(ciForceTimeStepEnvVar, cEnvValue); + if (!cEnvValue.empty()) state.dataSysVars->ciForceTimeStep = env_var_on(cEnvValue); // Yes or True } } // namespace DataSystemVariables diff --git a/src/EnergyPlus/DataSystemVariables.hh b/src/EnergyPlus/DataSystemVariables.hh index 5d4eedc1c54..f1d57148038 100644 --- a/src/EnergyPlus/DataSystemVariables.hh +++ b/src/EnergyPlus/DataSystemVariables.hh @@ -150,6 +150,7 @@ struct SystemVarsData : BaseGlobalStruct int NumberIntRadThreads = 1; int iNominalTotSurfaces = 0; bool Threading = false; + bool ciForceTimeStep = false; void init_state([[maybe_unused]] EnergyPlusData &state) override { diff --git a/src/EnergyPlus/SimulationManager.cc b/src/EnergyPlus/SimulationManager.cc index f28b5f9843a..4f5808e0f65 100644 --- a/src/EnergyPlus/SimulationManager.cc +++ b/src/EnergyPlus/SimulationManager.cc @@ -765,6 +765,9 @@ namespace SimulationManager { state.dataIPShortCut->cAlphaFieldNames, state.dataIPShortCut->cNumericFieldNames); state.dataGlobal->NumOfTimeStepInHour = Number(1); + if (state.dataSysVars->ciForceTimeStep) { + state.dataGlobal->NumOfTimeStepInHour = 2; // Force 30 minute time steps on CI + } if (state.dataGlobal->NumOfTimeStepInHour <= 0 || state.dataGlobal->NumOfTimeStepInHour > 60) { Alphas(1) = fmt::to_string(state.dataGlobal->NumOfTimeStepInHour); ShowWarningError(state, format("{}: Requested number ({}) invalid, Defaulted to 4", CurrentModuleObject, Alphas(1))); diff --git a/third_party/.gitignore b/third_party/.gitignore index e1afacd951f..f684023a3d0 100644 --- a/third_party/.gitignore +++ b/third_party/.gitignore @@ -32,6 +32,7 @@ btwxt/.gitmodules btwxt/vendor/fmt btwxt/vendor/courierr/vendor/fmt btwxt/vendor/courierr/.gitmodules +btwxt/vendor/courierr/vendor/googletest libtk205/.gitmodules # unused repo directories/files diff --git a/tst/EnergyPlus/unit/CommandLineInterface.unit.cc b/tst/EnergyPlus/unit/CommandLineInterface.unit.cc index d9d23566260..7fb32b7e246 100644 --- a/tst/EnergyPlus/unit/CommandLineInterface.unit.cc +++ b/tst/EnergyPlus/unit/CommandLineInterface.unit.cc @@ -385,7 +385,7 @@ TEST_F(CommandLineInterfaceFixture, runReadVars) } } -TEST_F(CommandLineInterfaceFixture, numThread) +TEST_F(CommandLineInterfaceFixture, DISABLED_numThread) { struct TestCase { diff --git a/tst/EnergyPlus/unit/Timer.unit.cc b/tst/EnergyPlus/unit/Timer.unit.cc index 6817974a17f..ac1878fc137 100644 --- a/tst/EnergyPlus/unit/Timer.unit.cc +++ b/tst/EnergyPlus/unit/Timer.unit.cc @@ -60,7 +60,7 @@ using namespace EnergyPlus; -TEST_F(EnergyPlusFixture, Timer_ticktock) +TEST_F(EnergyPlusFixture, DISABLED_Timer_ticktock) { constexpr std::chrono::milliseconds::rep sleep_time_ms = 100; @@ -70,7 +70,7 @@ TEST_F(EnergyPlusFixture, Timer_ticktock) std::this_thread::sleep_for(std::chrono::milliseconds(sleep_time_ms)); t.tock(); // In some occurrences CI is reporting slightly above than 100 values, probably system was quite busy at that time, - // but we don't want to have the test failing occassionally + // but we don't want to have the test failing occasionally EXPECT_GE(t.duration().count(), sleep_time_ms); EXPECT_LT(t.duration().count(), sleep_time_ms * 2); EXPECT_GE(t.elapsedSeconds(), sleep_time_s); From 6dfd01e050a211a813f717bfc6554a44869d63ff Mon Sep 17 00:00:00 2001 From: Edwin Lee Date: Wed, 4 Sep 2024 11:06:47 -0500 Subject: [PATCH 3/3] Fix epjson build dir, re-enable code integrity --- ...d_test_code_integrity.yml.txt => test_code_integrity.yml} | 0 .github/workflows/test_epjson.yml | 5 ++++- 2 files changed, 4 insertions(+), 1 deletion(-) rename .github/workflows/{disabled_test_code_integrity.yml.txt => test_code_integrity.yml} (100%) diff --git a/.github/workflows/disabled_test_code_integrity.yml.txt b/.github/workflows/test_code_integrity.yml similarity index 100% rename from .github/workflows/disabled_test_code_integrity.yml.txt rename to .github/workflows/test_code_integrity.yml diff --git a/.github/workflows/test_epjson.yml b/.github/workflows/test_epjson.yml index 12aebf2c2bc..bfd093c8947 100644 --- a/.github/workflows/test_epjson.yml +++ b/.github/workflows/test_epjson.yml @@ -27,8 +27,11 @@ jobs: shell: bash run: pip install pytest pytest-timeout + - name: Create Build Directory + run: cmake -E make_directory ./build/ + - name: Configure CMake - working-directory: ./build # TODO: Wait...it will automatically create it? + working-directory: ./build run: cmake -G "Visual Studio 17 2022" -A x64 ../ - name: Test epjson