Skip to content

Commit bb39905

Browse files
authored
Merge pull request #58 from ACCESS-NRI/davide/integration_tests_pytest
Set pytest integration tests
2 parents 63ab4a8 + deb4432 commit bb39905

File tree

4 files changed

+239
-4
lines changed

4 files changed

+239
-4
lines changed

.conda/env_dev.yml

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
channels:
2+
- accessnri
3+
- conda-forge
4+
- coecms
5+
- nodefaults
6+
7+
dependencies:
8+
- mule
9+
- numpy <= 1.23.4 # https://stackoverflow.com/a/75148219/21024780
10+
- scitools-iris
11+
- xarray
12+
- versioneer
13+
- pytest
14+
- pytest-cov
15+
- pytest-xdist
16+
- hypothesis

.gitignore

-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
test_data/
22
.coverage
3-
*.sh
43
__pycache__/
54
*.py[cod]
65
.ruff_cache/

README.md

+46-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,48 @@
1-
[![pre-commit](https://img.shields.io/badge/pre--commit-enabled-brightgreen?logo=pre-commit)](https://github.com/pre-commit/pre-commit)
2-
31
# replace_landsurface
42

5-
This repository contains Python scripts for use by Regional Nesting Suites to replace specific land surface fields in static input data.
3+
## About
4+
5+
`replace_landsurface` is a `Python` utility to be used within ACCESS-NRI versions of the Regional Nesting Suites to replace specific land surface initial/boundary conditions.
6+
7+
8+
## Development/Testing instructions
9+
For development/testing, it is recommended to install `replace_landsurface` as a development package within a `micromamba`/`conda` testing environment.
10+
11+
### Clone replace_landsurface GitHub repo
12+
```
13+
git clone [email protected]:ACCESS-NRI/replace_landsurface.git
14+
```
15+
16+
### Create a micromamba/conda testing environment
17+
> [!TIP]
18+
> In the following instructions `micromamba` can be replaced with `conda`.
19+
20+
```
21+
cd replace_landsurface
22+
micromamba env create -n replace_landsurface_dev --file .conda/env_dev.yml
23+
micromamba activate replace_landsurface_dev
24+
```
25+
26+
### Install replace_landsurface as a development package
27+
```
28+
pip install --no-deps --no-build-isolation -e .
29+
```
30+
31+
### Running the tests
32+
33+
The test suite currently includes only integration tests.
34+
35+
To manually run the tests, from the `replace_landsurface` directory, you can:
36+
37+
1. Activate your [micromamba/conda testing environment](#create-a-micromamba-conda-testing-environment)
38+
2. Run the following command:
39+
```
40+
pytest -n 4
41+
```
42+
43+
> [!TIP]
44+
> The `-n 4` option is a [pytest-xdist](https://pytest-xdist.readthedocs.io/en/stable/) option to run the tests in parallel across 4 different workers.
45+
46+
> [!IMPORTANT]
47+
> Integration tests are designed to be run on `Gadi`.
48+
> If you run tests on a local machine, the integration tests will be skipped.

tests/integration/test_integration.py

+177
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,177 @@
1+
import contextlib
2+
import filecmp
3+
import os
4+
import shutil
5+
import socket
6+
from unittest.mock import patch
7+
import pytest
8+
9+
# If not on Gadi, skip the tests because the test data is not available
10+
GADI_HOSTNAME = "gadi.nci.org.au"
11+
hostname = socket.gethostname()
12+
# Marker to skip tests if not on Gadi
13+
skip_marker = pytest.mark.skipif(
14+
not hostname.endswith(GADI_HOSTNAME),
15+
reason=f"Skipping integration tests because they cannot be executed on {hostname}.\n"
16+
"Integration tests are specifically designed to run on Gadi (gadi.nci.org.au).",
17+
)
18+
# Marker to suppress warnings
19+
warning_marker = pytest.mark.filterwarnings("ignore::Warning")
20+
# Apply the markers to all tests in this file
21+
pytestmark = [skip_marker, warning_marker]
22+
23+
############################################
24+
## === Integration tests setup === ##
25+
############################################
26+
TEST_DATA_DIR = "/g/data/vk83/testing/data/replace_landsurface/integration_tests"
27+
INPUT_DIR = os.path.join(TEST_DATA_DIR, "input_data")
28+
OUTPUT_DIR = os.path.join(TEST_DATA_DIR, "expected_outputs")
29+
DRIVING_DATA_DIR = os.path.join(TEST_DATA_DIR, "driving_data")
30+
# Set the ROSE_DATA environment variable to the driving data directory
31+
os.environ["ROSE_DATA"] = str(DRIVING_DATA_DIR)
32+
from replace_landsurface import hres_ic, hres_eccb # importing here because we need to set the ROSE_DATA env variable before importing # noqa
33+
34+
35+
############################################
36+
## === Integration tests === ##
37+
############################################
38+
def get_test_args(num, start, _type):
39+
test_dir = f"test_{num}"
40+
_hres_ic = (
41+
os.path.join(INPUT_DIR, test_dir, "hres_ic")
42+
if _type == "astart"
43+
else "NOT_USED"
44+
)
45+
return [
46+
"script_name",
47+
"--mask",
48+
os.path.join(INPUT_DIR, test_dir, "mask"),
49+
"--file",
50+
os.path.join(INPUT_DIR, test_dir, "file" + ".tmp"),
51+
"--start",
52+
start,
53+
"--type",
54+
_type,
55+
"--hres_ic",
56+
_hres_ic,
57+
]
58+
59+
60+
def get_error_msg(num, output, expected_output):
61+
return f"Test {num}: Test output '{output}' does not match the expected output '{expected_output}'!"
62+
63+
64+
@pytest.fixture
65+
def mock_sys_argv():
66+
@contextlib.contextmanager
67+
def _mock_sys_argv(num, start, _type):
68+
with patch("sys.argv", get_test_args(num, start, _type)):
69+
yield mock_sys_argv
70+
71+
return _mock_sys_argv
72+
73+
74+
@pytest.fixture(scope="module")
75+
def working_dir(tmp_path_factory):
76+
return tmp_path_factory.mktemp("replace_landsurface_integration_tests")
77+
78+
79+
@pytest.fixture()
80+
def get_output_path(working_dir):
81+
def _get_output_path(num):
82+
return os.path.join(working_dir, f"output_{num}")
83+
84+
return _get_output_path
85+
86+
87+
@pytest.fixture()
88+
def get_expected_output_path():
89+
def _get_expected_output_path(num):
90+
return os.path.join(OUTPUT_DIR, f"output_{num}")
91+
92+
return _get_expected_output_path
93+
94+
@pytest.fixture(scope="module")
95+
def original_shutil_move():
96+
return shutil.move
97+
98+
99+
@pytest.fixture()
100+
def new_shutil_move(original_shutil_move, get_output_path):
101+
def _new_shutil_move(num):
102+
def _wrapper(src, dst, **kwargs):
103+
output_path = get_output_path(num)
104+
return original_shutil_move(src=src, dst=output_path, **kwargs)
105+
106+
return _wrapper
107+
108+
return _new_shutil_move
109+
110+
111+
@pytest.mark.parametrize(
112+
"num, start, _type",
113+
[
114+
(1, "202202260000", "era5land"),
115+
(2, "202008090000", "barra"),
116+
(3, "202112310000", "astart"),
117+
],
118+
ids=[
119+
"hres_ic_era5land",
120+
"hres_ic_barra",
121+
"hres_ic_astart",
122+
],
123+
)
124+
def test_hres_ic(
125+
new_shutil_move,
126+
get_output_path,
127+
get_expected_output_path,
128+
mock_sys_argv,
129+
num,
130+
start,
131+
_type,
132+
):
133+
"""
134+
Test the hres_ic entry point
135+
"""
136+
with mock_sys_argv(num, start, _type):
137+
with patch("shutil.move", side_effect=new_shutil_move(num)):
138+
hres_ic.main()
139+
output = get_output_path(num)
140+
expected_output = get_expected_output_path(num)
141+
# Compare the output file with the expected output
142+
assert filecmp.cmp(output, expected_output), get_error_msg(
143+
num, output, expected_output
144+
)
145+
146+
@pytest.mark.parametrize(
147+
"num, start, _type",
148+
[
149+
(4, "202305040000", "era5land"),
150+
# (5, "202403050000", "barra"),
151+
],
152+
ids=[
153+
"hres_eccb_era5land",
154+
# "hres_eccb_barra",
155+
],
156+
)
157+
def test_hres_eccb(
158+
new_shutil_move,
159+
get_output_path,
160+
get_expected_output_path,
161+
mock_sys_argv,
162+
num,
163+
start,
164+
_type,
165+
):
166+
"""
167+
Test the hres_eccb entry point
168+
"""
169+
with mock_sys_argv(num, start, _type):
170+
with patch("shutil.move", side_effect=new_shutil_move(num)):
171+
hres_eccb.main()
172+
output = get_output_path(num)
173+
expected_output = get_expected_output_path(num)
174+
# Compare the output file with the expected output
175+
assert filecmp.cmp(output, expected_output), get_error_msg(
176+
num, output, expected_output
177+
)

0 commit comments

Comments
 (0)