From 25d0092918434208c831eb65b21f383cfdbb44a2 Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Mon, 2 Dec 2024 23:33:32 +0100 Subject: [PATCH 1/7] Skip tests with missing requirements --- examples/read-csv/main.py | 7 ------- examples/read-csv/readcsv.py | 5 +++++ storages/python/tests-python/test_image.py | 8 ++++---- 3 files changed, 9 insertions(+), 11 deletions(-) diff --git a/examples/read-csv/main.py b/examples/read-csv/main.py index c7961a81f..cbebd35a9 100644 --- a/examples/read-csv/main.py +++ b/examples/read-csv/main.py @@ -1,8 +1 @@ -try: - import pandas # noqa: F401 - import tables # noqa: F401 -except ImportError: - import sys - sys.exit(44) # skip this test if pandas is not available - import readcsv # noqa: F401 diff --git a/examples/read-csv/readcsv.py b/examples/read-csv/readcsv.py index 8b1a9e545..687aa74ea 100644 --- a/examples/read-csv/readcsv.py +++ b/examples/read-csv/readcsv.py @@ -1,6 +1,11 @@ from pathlib import Path import dlite +from dlite import importskip + +importskip("pandas") +importskip("tables") +importskip("yaml") # Set up some paths diff --git a/storages/python/tests-python/test_image.py b/storages/python/tests-python/test_image.py index a27bf7922..0092c7fbb 100644 --- a/storages/python/tests-python/test_image.py +++ b/storages/python/tests-python/test_image.py @@ -2,12 +2,12 @@ from pathlib import Path import numpy as np + import dlite +from dlite.testutils import importskip -try: - import skimage -except ImportError: - sys.exit(44) # skip test +importskip("scipy") +importskip("skimage") thisdir = Path(__file__).absolute().parent From 6de725a448e1fa3dbba066de4c6e38b438ee98ff Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Tue, 3 Dec 2024 00:47:20 +0100 Subject: [PATCH 2/7] Corrected import path --- examples/read-csv/readcsv.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/read-csv/readcsv.py b/examples/read-csv/readcsv.py index 687aa74ea..1ad5cd223 100644 --- a/examples/read-csv/readcsv.py +++ b/examples/read-csv/readcsv.py @@ -1,7 +1,7 @@ from pathlib import Path import dlite -from dlite import importskip +from dlite.testutils import importskip importskip("pandas") importskip("tables") From 4f83104565ca8bd3f824df975aa8850d106c0b0c Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Fri, 13 Dec 2024 12:10:32 +0100 Subject: [PATCH 3/7] Updated importskip to not skip tests with missing dependency on github CI --- .github/workflows/ci_tests.yml | 10 ++---- bindings/python/tests/CMakeLists.txt | 2 ++ bindings/python/testutils.py | 34 +++++++++++++++++-- doc/user_guide/environment_variables.md | 5 +++ examples/CMakeLists.txt | 3 ++ examples/mappings/oteexample.py | 19 ++++++----- .../tests-python/test_mongodb-atlas_python.py | 5 +++ .../tests-python/test_mongodb_python.py | 6 ++-- .../test_postgresql_storage_python.py | 10 +++--- storages/python/tests-python/test_redis.py | 2 +- 10 files changed, 69 insertions(+), 27 deletions(-) diff --git a/.github/workflows/ci_tests.yml b/.github/workflows/ci_tests.yml index 6a10df7a0..cce45ce9d 100644 --- a/.github/workflows/ci_tests.yml +++ b/.github/workflows/ci_tests.yml @@ -63,14 +63,6 @@ jobs: run: | python3 -c 'import numpy as np; print(np.get_include())' - - name: Workaround - install CMake 3.25.2 since 3.26.0 doesn't work - # Normally we would install cmake with the package manager, but - # ubuntu 20.04 doesn't seems to keep older versions of cmake around... - # Fortunately, there exists a pip package - run: | - python3 -m pip install cmake==3.25.2 - cmake --version - - name: List installed Python packages run: | uname -a @@ -78,6 +70,8 @@ jobs: pip freeze - name: Configure + env: + DLITE_IMPORTSKIP_EXITCODE: 1 run: | Python3_ROOT=$(python3 -c 'import sys; print(sys.exec_prefix)') \ CFLAGS='-Wno-missing-field-initializers' \ diff --git a/bindings/python/tests/CMakeLists.txt b/bindings/python/tests/CMakeLists.txt index a5d0a7e2a..d72b873ee 100644 --- a/bindings/python/tests/CMakeLists.txt +++ b/bindings/python/tests/CMakeLists.txt @@ -54,6 +54,8 @@ foreach(test ${tests}) ENVIRONMENT "DLITE_USE_BUILD_ROOT=YES") # Skip tests that exit with return code 44 + set_property(TEST ${name} APPEND PROPERTY + ENVIRONMENT "DLITE_IMPORTSKIP_EXITCODE=$ENV{DLITE_IMPORTSKIP_EXITCODE}") set_property(TEST ${name} PROPERTY SKIP_RETURN_CODE 44) endforeach() diff --git a/bindings/python/testutils.py b/bindings/python/testutils.py index 4cc6fefa3..6c0db355e 100644 --- a/bindings/python/testutils.py +++ b/bindings/python/testutils.py @@ -1,5 +1,6 @@ """Some utilities for testing.""" import importlib +import os import socket import sys @@ -71,15 +72,44 @@ def importcheck(module_name, package=None): return None -def importskip(module_name, package=None, exitcode=44): +def importskip(module_name, package=None, exitcode=44, + env_exitcode="DLITE_IMPORTSKIP_EXITCODE"): """Import and return the requested module. Calls `sys.exit()` with given exitcode if the module cannot be imported. + + Arguments: + module_name: Name of module to try to import. + package: Optional package name that the module might reside in. + exitcode: The default exit code of `sys.exit()` if the module cannot + be imported. + env_exitcode: Name of environment variable containing an exitcode + to call `sys.exit()` with if the module cannot be imported. + This overrides `exitcode`. + + Notes: + If you want to run the tests and get an error if a module + cannot be imported, set environment variable + `DLITE_IMPORTSKIP_EXITCODE=1` before running the tests (if you + run with ctest, set `DLITE_IMPORTSKIP_EXITCODE` at configure + time). + + For packages that depends on external services like postgresql, + call `importskip()` with `env_exitcode=None` to skip the test + regardless of `DLITE_IMPORTSKIP_EXITCODE` is set. + """ try: return importlib.import_module(module_name, package=package) except ModuleNotFoundError as exc: - print(f"{exc}: skipping test", file=sys.stderr) + if env_exitcode and env_exitcode in os.environ: + try: + exitcode = int(os.environ[env_exitcode]) + except: + pass + else: + print(f"{exc}: skipping test", file=sys.stderr) + sys.exit(exitcode) diff --git a/doc/user_guide/environment_variables.md b/doc/user_guide/environment_variables.md index 8ed012d9b..400d9cee4 100644 --- a/doc/user_guide/environment_variables.md +++ b/doc/user_guide/environment_variables.md @@ -96,6 +96,11 @@ DLITE_USE_BUILD_ROOT is set). - a glob pattern (/path/to/*.json) In the two last cases, the file extension must match the driver name. + - **DLITE_IMPORTSKIP_EXITCODE**: Exit code from tests that fails to + load a python module with `importskip()`. Define this to 1 at + configure time if you want the tests to not be skipped if they + cannot import a needed module. + Environment variables for controlling error handling ---------------------------------------------------- diff --git a/examples/CMakeLists.txt b/examples/CMakeLists.txt index b13d48b6d..e032d4ddc 100644 --- a/examples/CMakeLists.txt +++ b/examples/CMakeLists.txt @@ -107,6 +107,9 @@ if(FORCE_EXAMPLES OR EXISTS ${CMAKE_INSTALL_PREFIX}/share/dlite/examples) endif() set_property(TEST ${name} APPEND PROPERTY ENVIRONMENT "DLITE_USE_BUILD_ROOT=YES") + set_property(TEST ${name} APPEND PROPERTY + ENVIRONMENT "DLITE_IMPORTSKIP_EXITCODE=$ENV{DLITE_IMPORTSKIP_EXITCODE}") + set_property(TEST ${name} PROPERTY SKIP_RETURN_CODE 44) endforeach() diff --git a/examples/mappings/oteexample.py b/examples/mappings/oteexample.py index ec08857b4..b871bdcf8 100644 --- a/examples/mappings/oteexample.py +++ b/examples/mappings/oteexample.py @@ -1,16 +1,17 @@ """Mapping example using OTELib.""" from pathlib import Path -try: - from tripper import EMMO, MAP, Triplestore - from otelib import OTEClient - import oteapi_dlite # To check that it is installed -except ModuleNotFoundError as exc: - print(f"Skipping OTE example because of missing module: {exc}") - import sys - sys.exit(44) # Exit code 44 -> skip test because of missing dependencies - import dlite +from dlite.testutils import importskip + +importskip("tripper") +from tripper import EMMO, MAP, Triplestore + +importskip("otelib") +from otelib import OTEClient + +oteapi_dlite = importskip("oteapi_dlite", env_exitcode=None) + # Paths diff --git a/storages/python/tests-python/test_mongodb-atlas_python.py b/storages/python/tests-python/test_mongodb-atlas_python.py index b3b639db4..621b7b9d5 100644 --- a/storages/python/tests-python/test_mongodb-atlas_python.py +++ b/storages/python/tests-python/test_mongodb-atlas_python.py @@ -1,7 +1,12 @@ import sys import os from pathlib import Path + import dlite +from dlite.testutils import importskip + +importskip("pymongo", env_exitcode=None) + # Get the current file path current_file = Path(__file__).resolve() diff --git a/storages/python/tests-python/test_mongodb_python.py b/storages/python/tests-python/test_mongodb_python.py index 723e131ee..064d9f06f 100644 --- a/storages/python/tests-python/test_mongodb_python.py +++ b/storages/python/tests-python/test_mongodb_python.py @@ -7,8 +7,10 @@ from dlite.options import Options from dlite.testutils import importskip -importskip("pymongo") -mongomock = importskip("mongomock") +importskip("pymongo", env_exitcode=None) +mongomock = importskip("mongomock", env_exitcode=None) + +from dlite.testutils import importskip @mongomock.patch(servers=(('localhost', 27017),)) diff --git a/storages/python/tests-python/test_postgresql_storage_python.py b/storages/python/tests-python/test_postgresql_storage_python.py index 83be9530c..61989974f 100644 --- a/storages/python/tests-python/test_postgresql_storage_python.py +++ b/storages/python/tests-python/test_postgresql_storage_python.py @@ -5,14 +5,14 @@ from pathlib import Path sys.dont_write_bytecode = True -try: - import psycopg2 -except ImportError: - sys.exit(44) -from psycopg2 import sql import dlite from dlite.utils import instance_from_dict +from dlite.testutils import importskip + +psycopg2 = importskip("psycopg2", env_exitcode=None) +from psycopg2 import sql + from run_python_storage_tests import print_test_exception diff --git a/storages/python/tests-python/test_redis.py b/storages/python/tests-python/test_redis.py index e2ff6a706..662a3b826 100644 --- a/storages/python/tests-python/test_redis.py +++ b/storages/python/tests-python/test_redis.py @@ -7,7 +7,7 @@ import dlite from dlite.testutils import importskip, serverskip -importskip("redis") # skip this test if redis is not available +importskip("redis", env_exitcode=None) # skip this test if redis is not available serverskip("localhost", 6379) # skip test if redis is down From ffe34e1bce73c6ce27592fd2d15dee88d1b50981 Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Fri, 13 Dec 2024 12:22:42 +0100 Subject: [PATCH 4/7] Debugging with a specific test for mappings --- .github/workflows/ci_tests.yml | 6 ++++++ examples/mappings/oteexample.py | 2 +- 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/.github/workflows/ci_tests.yml b/.github/workflows/ci_tests.yml index cce45ce9d..36c51fe79 100644 --- a/.github/workflows/ci_tests.yml +++ b/.github/workflows/ci_tests.yml @@ -85,6 +85,12 @@ jobs: run: cmake --install . working-directory: build + - name: Test mappings + env: + DLITE_PYDEBUG: "" + working-directory: build + run: ctest -V -R mappings + - name: Test env: DLITE_PYDEBUG: "" diff --git a/examples/mappings/oteexample.py b/examples/mappings/oteexample.py index b871bdcf8..c75987dfa 100644 --- a/examples/mappings/oteexample.py +++ b/examples/mappings/oteexample.py @@ -7,7 +7,7 @@ importskip("tripper") from tripper import EMMO, MAP, Triplestore -importskip("otelib") +importskip("otelib", env_exitcode=None) from otelib import OTEClient oteapi_dlite = importskip("oteapi_dlite", env_exitcode=None) From 1d10d6b8d6357c532fd469c36df957116973a073 Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Fri, 13 Dec 2024 12:39:06 +0100 Subject: [PATCH 5/7] Added wheel as cmake target --- .github/workflows/ci_tests.yml | 6 ------ CMakeLists.txt | 8 ++++++++ storages/python/tests-python/CMakeLists.txt | 2 ++ 3 files changed, 10 insertions(+), 6 deletions(-) diff --git a/.github/workflows/ci_tests.yml b/.github/workflows/ci_tests.yml index 36c51fe79..cce45ce9d 100644 --- a/.github/workflows/ci_tests.yml +++ b/.github/workflows/ci_tests.yml @@ -85,12 +85,6 @@ jobs: run: cmake --install . working-directory: build - - name: Test mappings - env: - DLITE_PYDEBUG: "" - working-directory: build - run: ctest -V -R mappings - - name: Test env: DLITE_PYDEBUG: "" diff --git a/CMakeLists.txt b/CMakeLists.txt index b3fc817cb..5229d15ac 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -336,6 +336,13 @@ if(WITH_PYTHON) # endif() # message(STATUS "ENV{Python3_LIBRARY}: $ENV{Python3_LIBRARY}") + add_custom_target(wheel + COMMAND ${Python3_EXECUTABLE} -m pip wheel -w dist + ${dlite_SOURCE_DIR}/python + DEPENDS ${dlite} + COMMENT "Build Python wheels" + ) + message(STATUS "CMAKE_INSTALL_PREFIX = ${CMAKE_INSTALL_PREFIX}") message(STATUS "Python3_prefix = ${Python3_prefix}") @@ -942,6 +949,7 @@ configure_package_config_file( PATH_VARS ${DLITE_CONFIG_VARS} ) + ################################################################# # Install diff --git a/storages/python/tests-python/CMakeLists.txt b/storages/python/tests-python/CMakeLists.txt index 7cc78d8be..862cfff8d 100644 --- a/storages/python/tests-python/CMakeLists.txt +++ b/storages/python/tests-python/CMakeLists.txt @@ -78,6 +78,8 @@ foreach(test ${python-tests}) set_property(TEST ${test} APPEND PROPERTY ENVIRONMENT "LD_LIBRARY_PATH=${dlite_LD_LIBRARY_PATH_NATIVE}") endif() + set_property(TEST ${name} APPEND PROPERTY + ENVIRONMENT "DLITE_IMPORTSKIP_EXITCODE=$ENV{DLITE_IMPORTSKIP_EXITCODE}") # Skip tests that exit with return code 44 set_property(TEST ${test} PROPERTY SKIP_RETURN_CODE 44) From 7fcb8e0df4aef46f165b3e6443e3cf71289eb6bd Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Sat, 14 Dec 2024 15:38:56 +0100 Subject: [PATCH 6/7] Update bindings/python/testutils.py Co-authored-by: Francesca L. Bleken <48128015+francescalb@users.noreply.github.com> --- bindings/python/testutils.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/bindings/python/testutils.py b/bindings/python/testutils.py index 6c0db355e..1e356e3a5 100644 --- a/bindings/python/testutils.py +++ b/bindings/python/testutils.py @@ -94,9 +94,9 @@ def importskip(module_name, package=None, exitcode=44, run with ctest, set `DLITE_IMPORTSKIP_EXITCODE` at configure time). - For packages that depends on external services like postgresql, + For packages that depend on external services like postgresql, call `importskip()` with `env_exitcode=None` to skip the test - regardless of `DLITE_IMPORTSKIP_EXITCODE` is set. + regardless of whether `DLITE_IMPORTSKIP_EXITCODE` is set or not. """ try: From 3a0e29979f2f0946929f402c089734b80c43ac86 Mon Sep 17 00:00:00 2001 From: Jesper Friis Date: Sat, 14 Dec 2024 15:39:10 +0100 Subject: [PATCH 7/7] Update doc/user_guide/environment_variables.md Co-authored-by: Francesca L. Bleken <48128015+francescalb@users.noreply.github.com> --- doc/user_guide/environment_variables.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/doc/user_guide/environment_variables.md b/doc/user_guide/environment_variables.md index 400d9cee4..3c1584a02 100644 --- a/doc/user_guide/environment_variables.md +++ b/doc/user_guide/environment_variables.md @@ -96,9 +96,9 @@ DLITE_USE_BUILD_ROOT is set). - a glob pattern (/path/to/*.json) In the two last cases, the file extension must match the driver name. - - **DLITE_IMPORTSKIP_EXITCODE**: Exit code from tests that fails to + - **DLITE_IMPORTSKIP_EXITCODE**: Exit code from tests that fail to load a python module with `importskip()`. Define this to 1 at - configure time if you want the tests to not be skipped if they + configure time if you want the tests not to be skipped if they cannot import a needed module.