From 2470ca941c374fe55ffdd359eab2abf722a7eff3 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Mon, 5 Feb 2024 11:16:29 -0600 Subject: [PATCH 01/11] Use bundled package in deployment tests --- .github/workflows/CI.yml | 11 +-- .github/workflows/beta_rc.yaml | 2 +- .github/workflows/conda.yml | 62 +++++++--------- .github/workflows/examples.yml | 30 ++------ .github/workflows/lint.yml | 2 +- devtools/conda-envs/conda.yaml | 9 --- devtools/conda-envs/conda_oe.yaml | 11 --- openff/toolkit/_tests/conftest.py | 94 ------------------------ openff/toolkit/_tests/test_examples.py | 34 ++++++--- openff/toolkit/_tests/test_forcefield.py | 7 +- openff/toolkit/_tests/test_links.py | 22 ++++-- pytest.ini | 4 + 12 files changed, 79 insertions(+), 209 deletions(-) delete mode 100644 devtools/conda-envs/conda.yaml delete mode 100644 devtools/conda-envs/conda_oe.yaml create mode 100644 pytest.ini diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 108edf8ed..736eb5fb8 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -1,14 +1,6 @@ name: CI on: - push: - branches: - - "main" - - "maintenance/.*" - pull_request: - branches: - - "main" - - "maintenance/.*" schedule: # Nightly tests run on main by default: # Scheduled workflows run on the latest commit on the default or base branch. @@ -41,7 +33,6 @@ jobs: env: OE_LICENSE: ${{ github.workspace }}/oe_license.txt - PACKAGE: openff PYTEST_ARGS: -r fE --tb=short -nauto COV: --cov=openff/toolkit --cov-config=setup.cfg --cov-append --cov-report=xml @@ -151,7 +142,7 @@ jobs: PYTEST_ARGS+=" --ignore=openff/toolkit/_tests/test_examples.py" PYTEST_ARGS+=" --ignore=openff/toolkit/_tests/test_links.py" if [[ "$GITHUB_EVENT_NAME" == "schedule" ]]; then - PYTEST_ARGS+=" --runslow" + PYTEST_ARGS+=" -m 'slow or not slow'" fi python -m pytest $PYTEST_ARGS $COV diff --git a/.github/workflows/beta_rc.yaml b/.github/workflows/beta_rc.yaml index 45dab90de..f01f3e3fb 100644 --- a/.github/workflows/beta_rc.yaml +++ b/.github/workflows/beta_rc.yaml @@ -77,7 +77,7 @@ jobs: run: | PYTEST_ARGS+=" --ignore=openff/toolkit/_tests/test_examples.py" PYTEST_ARGS+=" --ignore=openff/toolkit/_tests/test_links.py" - PYTEST_ARGS+=" --runslow" + PYTEST_ARGS+=" -m 'slow or not slow'" pytest $PYTEST_ARGS $COV - name: Run code snippets in docs diff --git a/.github/workflows/conda.yml b/.github/workflows/conda.yml index 4e0d33b1d..32d358b59 100644 --- a/.github/workflows/conda.yml +++ b/.github/workflows/conda.yml @@ -1,6 +1,7 @@ name: Conda latest on: + push: release: types: - released @@ -20,59 +21,46 @@ jobs: strategy: fail-fast: false matrix: - os: [ubuntu-latest, macos-latest] - python-version: ["3.9", "3.10"] + os: [ubuntu-latest] + python-version: ["3.11"] openeye: ["true", "false"] env: - CI_OS: ${{ matrix.os }} OPENEYE: ${{ matrix.openeye }} - PYVER: ${{ matrix.python-version }} OE_LICENSE: ${{ github.workspace }}/oe_license.txt - PACKAGE: openff-toolkit + PYTEST_ARGS: -r fE --tb=short -n logical --durations=20 steps: - uses: actions/checkout@v4 - name: Vanilla install from conda uses: mamba-org/setup-micromamba@v1 - if: ${{ matrix.openeye == 'false' }} with: - environment-file: devtools/conda-envs/conda.yaml - create-args: >- - python=${{ matrix.python-version }} - - name: Install from conda with OpenEye - uses: mamba-org/setup-micromamba@v1 - if: ${{ matrix.openeye == 'true' }} - with: - environment-file: devtools/conda-envs/conda_oe.yaml + environment-name: latest-deployment create-args: >- python=${{ matrix.python-version }} + openff-toolkit-examples + smirnoff-plugins=2024 + pytest-xdist + pytest-rerunfailures - - name: Additional info about the build - run: | - uname -a - df -h - ulimit -a + - name: Install OpenEye Toolkits + if: ${{ matrix.openeye == 'true' }} + run: micromamba install openeye-toolkits -c openeye - name: Make oe_license.txt file from GH org secret "OE_LICENSE" + if: ${{ matrix.openeye == 'true' }} env: OE_LICENSE_TEXT: ${{ secrets.OE_LICENSE }} run: | echo "${OE_LICENSE_TEXT}" > ${OE_LICENSE} - name: Environment Information - run: | - conda info - conda list + run: micromamba info && micromamba list - name: Check installed toolkits run: | - # Checkout the state of the repo as of the last release (including RCs) export LATEST_TAG=$(git ls-remote --tags https://github.com/openforcefield/openff-toolkit.git | cut -f2 | grep -E "([0-9]+)\.([0-9]+)\.([0-9]+)$" | sort --version-sort | tail -1 | sed 's/refs\/tags\///') - git fetch --tags - git checkout tags/$LATEST_TAG - git log -1 | cat if [[ "$OPENEYE" == true ]]; then python -c "from openff.toolkit.utils.toolkits import OPENEYE_AVAILABLE; assert OPENEYE_AVAILABLE, 'OpenEye unavailable'" @@ -86,9 +74,6 @@ jobs: - name: Check that correct OFFTK version was installed run: | - # Go up one directory to ensure that we don't just load the OFFTK from the checked-out repo - cd ../ - export LATEST_TAG=$(git ls-remote --tags https://github.com/openforcefield/openff-toolkit.git | cut -f2 | grep -E "([0-9]+)\.([0-9]+)\.([0-9]+)$" | sort --version-sort | tail -1 | sed 's/refs\/tags\///') export FOUND_VER=$(python -c "import openff.toolkit; print(openff.toolkit.__version__)") @@ -102,20 +87,23 @@ jobs: exit 1 fi - cd openff-toolkit - - name: Test the package run: | - python -m pip install utilities/test_plugins - pwd - ls + # Act like we're testing with this patch deployed ... + python -m pip install . utilities/test_plugins/ if [[ "$OPENEYE" == true ]]; then python -c "import openff.toolkit; print(openff.toolkit.__file__)" python -c "import openeye; print(openeye.oechem.OEChemIsLicensed())" fi - PYTEST_ARGS=" -r fE --tb=short --runslow openff/toolkit/_tests/conftest.py" - PYTEST_ARGS+=" --ignore=openff/toolkit/_tests/test_links.py" - pytest $PYTEST_ARGS openff + PYTEST_ARGS+=" --ignore-glob='*_links.py'" + PYTEST_ARGS+=" --ignore-glob='*_examples.py'" + PYTEST_ARGS+=" --ignore-glob='*_nagl.py'" + + env + + python -m pytest $PYTEST_ARGS \ + --pyargs "openff.toolkit" \ + -m "slow or not slow" diff --git a/.github/workflows/examples.yml b/.github/workflows/examples.yml index d03bc2ec2..d2fc904d4 100644 --- a/.github/workflows/examples.yml +++ b/.github/workflows/examples.yml @@ -1,14 +1,6 @@ name: Examples on: - push: - branches: - - "main" - - "maintenance/.+" - pull_request: - branches: - - "main" - - "maintenance/.+" schedule: - cron: "0 0 * * *" @@ -66,24 +58,13 @@ jobs: create-args: >- python=${{ matrix.python-version }} - - name: Additional info about the build - run: | - uname -a - df -h - ulimit -a - - name: Make oe_license.txt file from GH org secret "OE_LICENSE" env: OE_LICENSE_TEXT: ${{ secrets.OE_LICENSE }} - run: | - echo "${OE_LICENSE_TEXT}" > ${OE_LICENSE} + run: echo "${OE_LICENSE_TEXT}" > ${OE_LICENSE} - name: Install package - run: | - # Maybe remove the packaged openff-toolkit, installed as a dependency of openmmforcefields - # and/or Interchange - micromamba remove --force openff-toolkit-base - python -m pip install . + run: python -m pip install . - name: Remove undesired toolkits run: | @@ -92,7 +73,7 @@ jobs: # so don't remove it. if [ ! -z "${{ env.PACKAGES_TO_REMOVE }}" ]; then for cpkg in ${{ env.PACKAGES_TO_REMOVE }}; do - if [[ $(conda list | grep $cpkg) ]]; then micromamba remove --force $cpkg --yes ; fi + if [[ $(micromamba list | grep $cpkg) ]]; then micromamba remove --force $cpkg --yes ; fi done fi @@ -119,10 +100,11 @@ jobs: python -c "from openff.toolkit.utils.toolkits import ${TK}_AVAILABLE; assert not ${TK}_AVAILABLE, '${TK} available'" done fi + - name: Environment Information run: | - conda info - conda list + micromamba info + micromamba list - name: Run example scripts run: | diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index 089bfb92d..f09d48fb3 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -1,7 +1,7 @@ name: lint on: - push: + release: jobs: diff --git a/devtools/conda-envs/conda.yaml b/devtools/conda-envs/conda.yaml deleted file mode 100644 index ec34886aa..000000000 --- a/devtools/conda-envs/conda.yaml +++ /dev/null @@ -1,9 +0,0 @@ -name: latest-deployment -channels: - - conda-forge -dependencies: - # Base depends - - openff-toolkit-examples - # Tests - - pytest=7.4 - - pytest-rerunfailures diff --git a/devtools/conda-envs/conda_oe.yaml b/devtools/conda-envs/conda_oe.yaml deleted file mode 100644 index d379d4cb2..000000000 --- a/devtools/conda-envs/conda_oe.yaml +++ /dev/null @@ -1,11 +0,0 @@ -name: latest-deployment -channels: - - conda-forge - - openeye -dependencies: - # Base depends - - openff-toolkit-examples - # Tests - - openeye-toolkits - - pytest=7.4 - - pytest-rerunfailures diff --git a/openff/toolkit/_tests/conftest.py b/openff/toolkit/_tests/conftest.py index 7fd3d0796..9233d7965 100644 --- a/openff/toolkit/_tests/conftest.py +++ b/openff/toolkit/_tests/conftest.py @@ -1,15 +1,9 @@ """ Configuration file for pytest. - -This adds the following command line options. -- runslow: Run tests marked as slow (default is False). - """ import logging -import pytest - logger = logging.getLogger(__name__) # Ensure QCPortal is imported before any OpenEye modules, see @@ -20,16 +14,6 @@ pass -def pytest_configure(config): - """ - Initialization hook to register custom markers without a pytest.ini - More info: https://docs.pytest.org/en/latest/reference.html#initialization-hooks - """ - config.addinivalue_line( - "markers", "slow: marks tests as slow (deselect with `-m 'not slow'`)" - ) - - def untar_full_alkethoh_and_freesolv_set(): """When running slow tests, we unpack the full AlkEthOH and FreeSolv test sets in advance to speed things up. @@ -48,81 +32,3 @@ def untar_full_alkethoh_and_freesolv_set(): tarfile_path = os.path.join(molecule_dir_path, tarfile_name) with tarfile.open(tarfile_path, "r:gz") as tar: tar.extractall(path=molecule_dir_path) - - -def pytest_addoption(parser): - """Add the pytest command line option --runslow and --failwip. - - If --runslow is not given, tests marked with pytest.mark.slow are - skipped. - - If --failwip is not given, tests marked with pytest.mark.wip are - xfailed. - - Parameters - ---------- - parser : argparsing.parser - The parser used by pytest to process arguments - - Returns - ------- - None - - """ - - # Loaded pytest plugins define their own arguments, and in certain cases - # our options use the same name. Although this can define two different - # behaviors for the same argument, the two options below are fairly - # unambiguous in their interpretation, so we skip any errors and allow - # the testing to proceed - - try: - parser.addoption( - "--runslow", action="store_true", default=False, help="run slow tests" - ) - except ValueError: - logger.warning( - "Option --runslow already added elsewhere (from a plugin possibly?). Skipping..." - ) - - try: - parser.addoption( - "--failwip", - action="store_true", - default=False, - help="fail work in progress tests", - ) - except ValueError: - logger.warning( - "Option --failwip already added elsewhere (from a plugin possibly?). Skipping..." - ) - - -def pytest_collection_modifyitems(config, items): - if config.getoption("runslow"): - # If --runslow is given, we don't have to mark items for skipping, - # but we need to extract the whole AlkEthOH and FreeSolv sets (see - # test_forcefield::test_alkethoh/freesolv_parameters_assignment). - untar_full_alkethoh_and_freesolv_set() - else: - # Mark for skipping all items marked as slow. - skip_slow = pytest.mark.skip( - reason="specify --runslow pytest option to run this test." - ) - for item in items: - if "slow" in item.keywords: - item.add_marker(skip_slow) - - # Mark work-in-progress tests for xfail. - if not config.getoption("failwip"): - xfail_wip_reason = ( - "This is a work in progress test. Specify " - "--failwip pytest option to make this test fail." - ) - for item in items: - if "wip" in item.keywords: - # Augment original reason. - reason = xfail_wip_reason + item.get_closest_marker("wip").kwargs.get( - "reason", "" - ) - item.add_marker(pytest.mark.xfail(reason=reason)) diff --git a/openff/toolkit/_tests/test_examples.py b/openff/toolkit/_tests/test_examples.py index bed901cd2..64c8a5fbc 100644 --- a/openff/toolkit/_tests/test_examples.py +++ b/openff/toolkit/_tests/test_examples.py @@ -2,7 +2,6 @@ Test that the examples in the repo run without errors. """ -import os import pathlib import re import subprocess @@ -38,19 +37,19 @@ def run_script_str(script_str): """ with tempfile.TemporaryDirectory() as tmp_dir: - temp_file_path = os.path.join(tmp_dir, "temp.py") + temp_file_path = (pathlib.Path(tmp_dir) / "temp.py").as_posix() # Create temporary python script. with open(temp_file_path, "w") as f: f.write(script_str) # Run the Python script. try: run_script_file(temp_file_path) - except: # noqa + except Exception as error: script_str = textwrap.indent(script_str, " ") - raise Exception(f"The following script failed:\n{script_str}") + raise Exception(f"The following script failed:\n{script_str}") from error -def find_example_scripts(): +def find_example_scripts() -> list[str]: """Find all Python scripts, excluding Jupyter notebooks, in the examples folder. Returns @@ -58,11 +57,16 @@ def find_example_scripts(): example_file_paths : list[str] List of full paths to python scripts to execute. """ - examples_dir_path = ROOT_DIR_PATH.joinpath("examples") + if "site-packages" in __file__: + # This test file is being collected from the installed package, which + # does not provide the examples folder in the same location + return list() + + examples_dir_path = pathlib.Path(__file__).parents[3] / "examples" # Examples that require RDKit rdkit_examples = { - examples_dir_path.joinpath("conformer_energies/conformer_energies.py"), + examples_dir_path / "conformer_energies/conformer_energies.py", } example_file_paths = [] @@ -75,7 +79,7 @@ def find_example_scripts(): return example_file_paths -def find_readme_examples(): +def find_readme_examples() -> list[str]: """Yield the Python scripts in the main README.md file. Returns @@ -83,9 +87,19 @@ def find_readme_examples(): readme_examples : list[str] The list of Python scripts included in the README.md files. """ - readme_path = ROOT_DIR_PATH.joinpath("README.md") - with open(readme_path, "r") as f: + if "site-packages" in __file__: + # This test file is being collected from the installed package, which + # does not provide the README file. + # Note that there will likely be a mis-bundled file + # $CONDA_PREFIX/lib/python3.x/site-packages/README.md, but this is not + # the toolkit's README file! + return list() + + readme_file_path = pathlib.Path(__file__).parents[3] / "README.md" + + with open(readme_file_path, "r") as f: readme_content = f.read() + return re.findall("```python(.*?)```", readme_content, flags=re.DOTALL) diff --git a/openff/toolkit/_tests/test_forcefield.py b/openff/toolkit/_tests/test_forcefield.py index e7b836491..1ab4582f0 100644 --- a/openff/toolkit/_tests/test_forcefield.py +++ b/openff/toolkit/_tests/test_forcefield.py @@ -1347,7 +1347,7 @@ def test_parameterize_large_system( force_field, ): """Test parameterizing a large system of several distinct molecules. - This test is very slow, so it is only run if the --runslow option is provided to pytest. + This test is very slow, so it is only run if the slow marker option is provided to pytest. """ box_file_path = get_data_file_path( os.path.join("systems", "packmol_boxes", box) @@ -1886,9 +1886,8 @@ def test_handlers_tracked_if_already_loaded(self): plugins = load_handler_plugins() - assert ( - len(plugins) > 0 - ), "Test assumes that some ParameterHandler plugins are available" + if len(plugins) == 0: + pytest.skip("Test assumes that some ParameterHandler plugins are available") assert ForceField(load_plugins=False)._plugin_parameter_handler_classes == [] assert ForceField(load_plugins=True)._plugin_parameter_handler_classes == [ diff --git a/openff/toolkit/_tests/test_links.py b/openff/toolkit/_tests/test_links.py index 262632df6..8f53b33f3 100644 --- a/openff/toolkit/_tests/test_links.py +++ b/openff/toolkit/_tests/test_links.py @@ -1,15 +1,11 @@ -import os +import pathlib import re from urllib.request import Request, urlopen import pytest -ROOT_DIR_PATH = os.path.join( - os.path.dirname(os.path.realpath(__file__)), "..", "..", ".." -) - -def find_readme_links(): +def find_readme_links() -> list[str]: """Yield all the links in the main README.md file. Returns @@ -17,9 +13,19 @@ def find_readme_links(): readme_examples : list[str] The list of links included in the README.md file. """ - readme_file_path = os.path.join(ROOT_DIR_PATH, "README.md") - with open(readme_file_path, "r") as f: + if "site-packages" in __file__: + # This test file is being collected from the installed package, which + # does not provide the README file. + # Note that there will likely be a mis-bundled file + # $CONDA_PREFIX/lib/python3.x/site-packages/README.md, but this is not + # the toolkit's README file! + return list() + + readme_file_path = pathlib.Path(__file__).parents[3] / "README.md" + + with open(readme_file_path.as_posix(), "r") as f: readme_content = f.read() + return re.findall("http[s]?://(?:[0-9a-zA-Z]|[-/.%:_])+", readme_content) diff --git a/pytest.ini b/pytest.ini new file mode 100644 index 000000000..a997e0e7f --- /dev/null +++ b/pytest.ini @@ -0,0 +1,4 @@ +[pytest] +markers = + slow: marks tests as slow (deselect with '-m "not slow"') +addopts = -m "not slow" From 821a8bbad4f1f0835d62112df1c95da0ce63f749 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Thu, 15 Feb 2024 10:05:36 -0600 Subject: [PATCH 02/11] Skip version check --- .github/workflows/conda.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/conda.yml b/.github/workflows/conda.yml index 32d358b59..93325e404 100644 --- a/.github/workflows/conda.yml +++ b/.github/workflows/conda.yml @@ -84,7 +84,7 @@ jobs: if [[ $LATEST_TAG != $FOUND_VER ]]; then echo "Version mismatch" - exit 1 + # exit 1 fi - name: Test the package From 5ad99d8ae87c005349f7248d0e4b801d4658e2e7 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Thu, 15 Feb 2024 11:02:06 -0600 Subject: [PATCH 03/11] Remove some legacy fixtures --- openff/toolkit/_tests/test_toolkit_io.py | 28 +++++++----------------- 1 file changed, 8 insertions(+), 20 deletions(-) diff --git a/openff/toolkit/_tests/test_toolkit_io.py b/openff/toolkit/_tests/test_toolkit_io.py index fe20df0ce..97dc43186 100644 --- a/openff/toolkit/_tests/test_toolkit_io.py +++ b/openff/toolkit/_tests/test_toolkit_io.py @@ -3,7 +3,6 @@ """ -import os import pathlib import sys import tempfile @@ -944,14 +943,6 @@ def test_from_file_obj_smi_supports_stringio(self): assert mol.name == "CHEMBL113" -@pytest.fixture(scope="class") -def tmpdir(request): - request.cls.tmpdir = tmpdir = tempfile.TemporaryDirectory() - with tmpdir: - yield - request.cls.tmpdir = None - - def assert_is_ethanol_sdf(f): assert f.readline() == "ethanol\n" # title line f.readline() # ignore next two lines @@ -973,12 +964,9 @@ def assert_is_ethanol_smiles(smiles): class BaseToFileIO: - def get_tmpfile(self, name): - return os.path.join(self.tmpdir.name, name) - @pytest.mark.parametrize("format_name", ["SDF", "sdf", "sDf", "mol", "MOL"]) - def test_to_file_sdf(self, format_name): - filename = self.get_tmpfile("abc.xyz") + def test_to_file_sdf(self, format_name, tmp_path): + filename = tmp_path / "abc.xyz" self.toolkit_wrapper.to_file(ETHANOL, filename, format_name) with open(filename) as f: assert_is_ethanol_sdf(f) @@ -999,8 +987,8 @@ def test_to_file_obj_sdf_with_bytesio(self): self.toolkit_wrapper.to_file_obj(ETHANOL, f, "sdf") @pytest.mark.parametrize("format_name", ["SMI", "smi", "sMi"]) - def test_to_file_smi(self, format_name): - filename = self.get_tmpfile("abc.xyz") + def test_to_file_smi(self, format_name, tmp_path): + filename = tmp_path / "abc.xyz" self.toolkit_wrapper.to_file(ETHANOL, filename, format_name) with open(filename) as f: assert_is_ethanol_smi(f) @@ -1026,19 +1014,19 @@ def test_to_file_qwe_format_raises_exception(self): self.toolkit_wrapper.to_file(ETHANOL, fileobj.name, "QWE") @pytest.mark.parametrize("format_name", ["smi", "sdf", "mol"]) - def test_to_file_when_the_file_does_not_exist(self, format_name): - filename = self.get_tmpfile("does/not/exist.smi") + def test_to_file_when_the_file_does_not_exist(self, format_name, tmp_path): + filename = tmp_path / "does/not/exist.smi" with pytest.raises(OSError): self.toolkit_wrapper.to_file(ETHANOL, filename, format_name) -@pytest.mark.usefixtures("init_toolkit", "tmpdir") +@pytest.mark.usefixtures("init_toolkit") @requires_openeye class TestOpenEyeToolkitToFileIO(BaseToFileIO): toolkit_wrapper_class = OpenEyeToolkitWrapper -@pytest.mark.usefixtures("init_toolkit", "tmpdir") +@pytest.mark.usefixtures("init_toolkit") @requires_rdkit class TestRDKitToolkitToFileIO(BaseToFileIO): toolkit_wrapper_class = RDKitToolkitWrapper From 4772e1130ccc25fe1217e780accbd93b4abab3b9 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Thu, 20 Jun 2024 09:34:08 -0500 Subject: [PATCH 04/11] Update deployment test environments --- .github/workflows/conda.yml | 2 +- devtools/conda-envs/conda.yaml | 2 +- devtools/conda-envs/conda_oe.yaml | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/conda.yml b/.github/workflows/conda.yml index 5618306b3..566cb722e 100644 --- a/.github/workflows/conda.yml +++ b/.github/workflows/conda.yml @@ -106,4 +106,4 @@ jobs: python -m pytest $PYTEST_ARGS \ --pyargs "openff.toolkit" \ - -m "slow or not slow" + -m "slow or not slow" -x diff --git a/devtools/conda-envs/conda.yaml b/devtools/conda-envs/conda.yaml index 0c64bc009..3193ed308 100644 --- a/devtools/conda-envs/conda.yaml +++ b/devtools/conda-envs/conda.yaml @@ -5,7 +5,7 @@ dependencies: # Base depends - openff-toolkit-examples # Tests - - pytest=7.4 + - pytest=8 - pytest-rerunfailures - nbval - parmed=3 diff --git a/devtools/conda-envs/conda_oe.yaml b/devtools/conda-envs/conda_oe.yaml index ad491db0a..ad6f94b1f 100644 --- a/devtools/conda-envs/conda_oe.yaml +++ b/devtools/conda-envs/conda_oe.yaml @@ -7,7 +7,7 @@ dependencies: - openff-toolkit-examples # Tests - openeye-toolkits - - pytest=7.4 + - pytest=8 - pytest-rerunfailures - nbval - parmed=3 From 295189c40d71082aedac93009615eb461206af2b Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Fri, 3 Jan 2025 15:55:14 -0600 Subject: [PATCH 05/11] Turn on CI --- .github/workflows/CI.yml | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 26a591be6..cc70069a9 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -1,10 +1,11 @@ name: CI on: + push: + branches: [ $default-branch ] + pull_request: + branches: [ $default-branch ] schedule: - # Nightly tests run on main by default: - # Scheduled workflows run on the latest commit on the default or base branch. - # (from https://help.github.com/en/actions/reference/events-that-trigger-workflows#scheduled-events-schedule) - cron: "21 0 * * *" defaults: From c9aa0b4390fab7d1b8ec1c4edadf904834db1531 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Fri, 3 Jan 2025 15:57:04 -0600 Subject: [PATCH 06/11] Clean up --- .github/workflows/conda.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/conda.yml b/.github/workflows/conda.yml index e7a38661b..c9f4a78d5 100644 --- a/.github/workflows/conda.yml +++ b/.github/workflows/conda.yml @@ -32,6 +32,8 @@ jobs: steps: - uses: actions/checkout@v4 + with: + fetch-depth: 0 - name: Vanilla install from conda uses: mamba-org/setup-micromamba@v2 From 6c9f0890324e91906fafc20c3bcea345702226bb Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Fri, 3 Jan 2025 16:08:43 -0600 Subject: [PATCH 07/11] Clean up --- .github/workflows/CI.yml | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index cc70069a9..60011451d 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -2,9 +2,10 @@ name: CI on: push: - branches: [ $default-branch ] + branches: + - main pull_request: - branches: [ $default-branch ] + - main schedule: - cron: "21 0 * * *" From db266c9e025a0c285f89d276bc3eb49d34ebf091 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Fri, 3 Jan 2025 16:12:39 -0600 Subject: [PATCH 08/11] Clean up --- .github/workflows/CI.yml | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 60011451d..7c7d05709 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -3,9 +3,10 @@ name: CI on: push: branches: - - main + - "main" pull_request: - - main + branches: + - "main" schedule: - cron: "21 0 * * *" From 88e0a1234eaf73eee75783ed38617942153a4cc7 Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Mon, 6 Jan 2025 09:04:18 -0600 Subject: [PATCH 09/11] Update docexample --- openff/toolkit/topology/topology.py | 9 ++------- 1 file changed, 2 insertions(+), 7 deletions(-) diff --git a/openff/toolkit/topology/topology.py b/openff/toolkit/topology/topology.py index 0f09020c5..cc8b09b55 100644 --- a/openff/toolkit/topology/topology.py +++ b/openff/toolkit/topology/topology.py @@ -1688,12 +1688,7 @@ def from_pdb( (24, 23, 29, 36) (32, 31, 40, 42) >>> [*top.hierarchy_iterator("residues")] - [HierarchyElement ('A', '1', ' ', 'ACE') of iterator 'residues' containing 6 atom(s), - HierarchyElement ('A', '2', ' ', 'SER') of iterator 'residues' containing 11 atom(s), - HierarchyElement ('A', '3', ' ', 'NME') of iterator 'residues' containing 6 atom(s), - HierarchyElement ('B', '1', ' ', 'ACE') of iterator 'residues' containing 6 atom(s), - HierarchyElement ('B', '2', ' ', 'CYS') of iterator 'residues' containing 11 atom(s), - HierarchyElement ('B', '3', ' ', 'NME') of iterator 'residues' containing 6 atom(s)] + [HierarchyElement ('A', '1', ' ', 'ACE') of iterator 'residues' containing 6 atom(s), HierarchyElement ('A', '2', ' ', 'SER') of iterator 'residues' containing 11 atom(s), HierarchyElement ('A', '3', ' ', 'NME') of iterator 'residues' containing 6 atom(s), HierarchyElement ('B', '1', ' ', 'ACE') of iterator 'residues' containing 6 atom(s), HierarchyElement ('B', '2', ' ', 'CYS') of iterator 'residues' containing 11 atom(s), HierarchyElement ('B', '3', ' ', 'NME') of iterator 'residues' containing 6 atom(s)] Polymer systems can also be supported if ``_custom_substructures`` are given as a ``dict[str, list[str]]``, where the keys are unique atom @@ -1716,7 +1711,7 @@ def from_pdb( ... get_data_file_path("systems/test_systems/PE.pdb"), ... _custom_substructures=PE_substructs, ... ) - """ + """ # noqa: E501 import io import json From 91070711599c1387b4e149db517ea35be8ae385a Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Mon, 6 Jan 2025 09:09:47 -0600 Subject: [PATCH 10/11] Run all doctests --- .github/workflows/CI.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 7c7d05709..55e9b99f0 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -160,7 +160,7 @@ jobs: if: ${{ matrix.rdkit == true && matrix.openeye == true }} run: | pytest openff \ - -v -x -n logical --no-cov --doctest-modules \ + -v -n logical --no-cov --doctest-modules \ --ignore-glob='openff/toolkit/_tests*' \ --ignore=openff/toolkit/data/ \ --ignore=openff/toolkit/utils/utils.py From ed686ec0bbbb9a869387630607537e408fad796a Mon Sep 17 00:00:00 2001 From: "Matthew W. Thompson" Date: Mon, 6 Jan 2025 11:30:29 -0600 Subject: [PATCH 11/11] Update doctests --- .../typing/engines/smirnoff/parameters.py | 27 ++++++++----------- pytest.ini | 2 ++ 2 files changed, 13 insertions(+), 16 deletions(-) diff --git a/openff/toolkit/typing/engines/smirnoff/parameters.py b/openff/toolkit/typing/engines/smirnoff/parameters.py index 24a2637d0..7440588bc 100644 --- a/openff/toolkit/typing/engines/smirnoff/parameters.py +++ b/openff/toolkit/typing/engines/smirnoff/parameters.py @@ -291,11 +291,10 @@ class ParameterAttribute: >>> my_par.attr_quantity = '1.0 * nanometer' >>> my_par.attr_quantity - >>> my_par.attr_quantity = 3.0 + >>> my_par.attr_quantity = 3.0 # doctest: +ELLIPSIS Traceback (most recent call last): ... - openff.toolkit.utils.exceptions.IncompatibleUnitError: - attr_quantity=3.0 dimensionless should have units of angstrom + openff.toolkit.utils.exceptions.IncompatibleUnitError: attr_quantity=3.0 dimensionless should have units of angstrom You can attach a custom converter to an attribute. @@ -334,7 +333,7 @@ class ParameterAttribute: ... TypeError: Cannot convert '4.0' to float - """ + """ # noqa: E501 UNDEFINED = UNDEFINED """Marker type for an undeclared default parameter.""" @@ -734,21 +733,18 @@ class _ParameterAttributeHandler: While assigning incompatible units is forbidden. - >>> my_par.k = 3.0 * unit.gram + >>> my_par.k = 3.0 * unit.gram # doctest: +ELLIPSIS Traceback (most recent call last): ... - openff.toolkit.utils.exceptions.IncompatibleUnitError: - k=3.0 gram should have units of kilocalorie / angstrom ** 2 / mole + openff.toolkit.utils.exceptions.IncompatibleUnitError: k=3.0 gram should have units of kilocalorie / angstrom ** 2 / mole On top of type checking, the constructor implemented in ``_ParameterAttributeHandler`` checks if some required parameters are not given. - >>> ParameterTypeOrHandler(length=3.0*unit.nanometer) + >>> ParameterTypeOrHandler(length=3.0*unit.nanometer) # doctest: +ELLIPSIS Traceback (most recent call last): ... - openff.toolkit.utils.exceptions.SMIRNOFFSpecError: - require the following missing - parameters: ['k']. Defined kwargs are ['length'] + openff.toolkit.utils.exceptions.SMIRNOFFSpecError: require the following missing parameters: ['k']. Defined kwargs are ['length'] Each attribute can be made optional by specifying a default value, and you can attach a converter function by passing a callable as an @@ -812,7 +808,7 @@ class _ParameterAttributeHandler: >>> my_par.periodicity [1, 6] - """ + """ # noqa: E501 def __init__(self, allow_cosmetic_attributes=False, **kwargs): """ @@ -1697,11 +1693,10 @@ class ParameterType(_ParameterAttributeHandler): ... ) >>> my_par.length - >>> my_par.k = 3.0 * unit.gram + >>> my_par.k = 3.0 * unit.gram # doctest: +ELLIPSIS Traceback (most recent call last): ... - openff.toolkit.utils.exceptions.IncompatibleUnitError: - k=3.0 gram should have units of kilocalorie / angstrom ** 2 / mole + openff.toolkit.utils.exceptions.IncompatibleUnitError: k=3.0 gram should have units of kilocalorie / angstrom ** 2 / mole Each attribute can be made optional by specifying a default value, and you can attach a converter function by passing a callable as an @@ -1769,7 +1764,7 @@ class ParameterType(_ParameterAttributeHandler): >>> my_par.periodicity [1, 6] - """ + """ # noqa: E501 # The string mapping to this ParameterType in a SMIRNOFF data source _ELEMENT_NAME: Optional[str] = None diff --git a/pytest.ini b/pytest.ini index a997e0e7f..3cafada44 100644 --- a/pytest.ini +++ b/pytest.ini @@ -2,3 +2,5 @@ markers = slow: marks tests as slow (deselect with '-m "not slow"') addopts = -m "not slow" +filterwarnings = + ignore:Molecule.from_pdb_and_smiles.*deprecated in favor of .*from_pdb