From 0c6ef391e7de56529f0ba9affaf85e931c8a96d0 Mon Sep 17 00:00:00 2001 From: Jochem Smit Date: Thu, 19 Oct 2023 15:13:11 +0200 Subject: [PATCH 1/6] refactor tests and add CI testing --- .github/workflows/pytest.yaml | 34 ++++++ tests/test_hdxrate.py | 200 ++++++++++++++++++++++++---------- 2 files changed, 176 insertions(+), 58 deletions(-) create mode 100644 .github/workflows/pytest.yaml diff --git a/.github/workflows/pytest.yaml b/.github/workflows/pytest.yaml new file mode 100644 index 0000000..a61fb5e --- /dev/null +++ b/.github/workflows/pytest.yaml @@ -0,0 +1,34 @@ +name: Run Tests + +on: + push: + branches: + - main + pull_request: + branches: + - main + +jobs: + test: + strategy: + matrix: + python-version: [3.6, 3.7, 3.8, 3.9, 3.10, 3.11] + + runs-on: ubuntu-latest + + steps: + - name: Checkout code + uses: actions/checkout@v4 + + - name: Set up Python + uses: actions/setup-python@v4 + with: + python-version: ${{ matrix.python-version }} + + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install -r requirements.txt + + - name: Run tests + run: pytest \ No newline at end of file diff --git a/tests/test_hdxrate.py b/tests/test_hdxrate.py index bda5161..0ae7a03 100644 --- a/tests/test_hdxrate.py +++ b/tests/test_hdxrate.py @@ -1,5 +1,6 @@ """Tests for `hdxrate` package.""" +from __future__ import annotations import numpy as np from hdxrate import k_int_from_sequence from hdxrate.hdxrate import get_side_chain_dictionary @@ -8,63 +9,146 @@ from itertools import combinations from operator import add +import pytest + pth = Path(__file__).parent -class TestHDXrate(object): - """Tests for `hdxrate` package.""" - - @classmethod - def setup_class(cls): - cls.seq1 = list('AAAWADEAA') - - k_reference = {'D': 3.87, 'E': 4.33, 'H': 7.0} # DH - dict = get_side_chain_dictionary(278, 8, k_reference) - one_letter = [k for k in dict.keys() if len(k) == 1] - cls.seq2 = reduce(add, [a + b for a, b in combinations(one_letter, 2)]) - - cls.seq2 = np.genfromtxt(pth / 'sequence.txt', dtype='U') - - def test_seq1(self): - # HD rates - pH_read = 6.6 - temp = 279 - rates = k_int_from_sequence(self.seq1, temp, pH_read, reference='poly', wildcard='X') * 60 - # Reference rates obtained from Englander group xls sheet - ref_rates = np.array([np.inf, 1.29939811E+03, 3.11703908E+01, 1.21266892E+01, 2.41959255E+01, 3.95805093E+01, - 1.63948783E+01, 2.25232682E+01, 4.94028674E-01]) - assert np.allclose(rates, ref_rates) - - # DH rates - pH_read = 7.0 - temp = 278 - rates = k_int_from_sequence(self.seq1, temp, pH_read, exchange_type='DH', reference='poly') * 60 - # Reference rates obtained from Englander group xls sheet - ref_rates = np.array([np.inf, 5.831286065E+03, 1.398828104E+02, 5.442072826E+01, 1.085836280E+02, - 1.764790873E+02, 7.219785837E+01, 9.955072660E+01, 2.216999376E+00]) - assert np.allclose(rates, ref_rates) - - #HH rates - pH_read = 7.0 - temp = 278 - rates = k_int_from_sequence(self.seq1, temp, pH_read, exchange_type='HH', reference='poly', wildcard='X') * 60 - # Reference rates obtained from Englander group xls sheet - ref_rates = np.array([np.inf, 7.01071144E+03, 1.68175255E+02, 6.54277663E+01, 1.30545556E+02, 2.12183978E+02, - 8.68187188E+01, 1.19715146E+02, 2.66540425E+00]) - assert np.allclose(rates, ref_rates) - - def test_seq2(self): - reference_rates = np.genfromtxt(pth / 'exchange_rates_xls.txt', skip_header=2, - delimiter='\t', filling_values=0.) - - reference_rates[0][:] = np.inf - - #HD exchange - rates = k_int_from_sequence(self.seq2, 279, 7 - 0.4, exchange_type='HD') * 60 - assert np.allclose(rates, reference_rates[:, 0], rtol=0.1, equal_nan=True) - - # - rates = k_int_from_sequence(self.seq2, 279, 7, exchange_type='DH') * 60 - assert np.allclose(rates, reference_rates[:, 1], rtol=0.1, equal_nan=True) - - rates = k_int_from_sequence(self.seq2, 279, 7, exchange_type='HH') * 60 - assert np.allclose(rates, reference_rates[:, 2], rtol=0.1, equal_nan=True) \ No newline at end of file + +@pytest.fixture() +def seq1() -> list[str]: + return list("AAAWADEAA") + + +@pytest.fixture() +def seq2() -> list[str]: + """sequence two a sequence of the pairwise combination of all side chains""" + k_reference = {"D": 3.87, "E": 4.33, "H": 7.0} # DH + chains_dict = get_side_chain_dictionary(278, 8, k_reference) + one_letter = [k for k in chains_dict.keys() if len(k) == 1] + seq2 = reduce(add, [a + b for a, b in combinations(one_letter, 2)]) + + return list(seq2) + + +def test_seq1_HD(seq1): + # HD rates + pH_read = 6.6 + temp = 279 + rates = ( + k_int_from_sequence(seq1, temp, pH_read, reference="poly", wildcard="X") * 60 + ) + # Reference rates obtained from Englander group xls sheet + ref_rates = np.array( + [ + np.inf, + 1.29939811e03, + 3.11703908e01, + 1.21266892e01, + 2.41959255e01, + 3.95805093e01, + 1.63948783e01, + 2.25232682e01, + 4.94028674e-01, + ] + ) + assert np.allclose(rates, ref_rates) + + +def test_seq1_DH(seq1): + # DH rates + pH_read = 7.0 + temp = 278 + rates = ( + k_int_from_sequence(seq1, temp, pH_read, exchange_type="DH", reference="poly") + * 60 + ) + # check repeated calls give the same result + rep_rates = ( + k_int_from_sequence(seq1, temp, pH_read, exchange_type="DH", reference="poly") + * 60 + ) + # Reference rates obtained from Englander group xls sheet + ref_rates = np.array( + [ + np.inf, + 5.831286065e03, + 1.398828104e02, + 5.442072826e01, + 1.085836280e02, + 1.764790873e02, + 7.219785837e01, + 9.955072660e01, + 2.216999376e00, + ] + ) + assert np.allclose(rates, ref_rates) + assert np.allclose(rates, rep_rates) + + +def test_seq1_HH(seq1): + # HH rates + pH_read = 7.0 + temp = 278 + rates = ( + k_int_from_sequence( + seq1, + temp, + pH_read, + exchange_type="HH", + reference="poly", + wildcard="X", + ) + * 60 + ) + + rep_rates = ( + k_int_from_sequence( + seq1, + temp, + pH_read, + exchange_type="HH", + reference="poly", + wildcard="X", + ) + * 60 + ) + + # Reference rates obtained from Englander group xls sheet + ref_rates = np.array( + [ + np.inf, + 7.01071144e03, + 1.68175255e02, + 6.54277663e01, + 1.30545556e02, + 2.12183978e02, + 8.68187188e01, + 1.19715146e02, + 2.66540425e00, + ] + ) + assert np.allclose(rates, ref_rates) + assert np.allclose(rates, rep_rates) + + +def test_seq2(seq2): + reference_rates = np.genfromtxt( + pth / "exchange_rates_xls.txt", + skip_header=2, + delimiter="\t", + filling_values=0.0, + ) + + reference_rates[0][:] = np.inf + + # HD exchange + rates = k_int_from_sequence(seq2, 279, 7 - 0.4, exchange_type="HD") * 60 + assert np.allclose(rates, reference_rates[:, 0], rtol=0.1, equal_nan=True) + + # DH exchange + rates = k_int_from_sequence(seq2, 279, 7, exchange_type="DH") * 60 + assert np.allclose(rates, reference_rates[:, 1], rtol=0.1, equal_nan=True) + + # HH exchange + rates = k_int_from_sequence(seq2, 279, 7, exchange_type="HH") * 60 + assert np.allclose(rates, reference_rates[:, 2], rtol=0.1, equal_nan=True) From dd516d102f764f8b767b4fc9ab8b46d3a7f8c9ca Mon Sep 17 00:00:00 2001 From: Jochem Smit Date: Thu, 19 Oct 2023 15:16:46 +0200 Subject: [PATCH 2/6] trigger on master --- .github/workflows/pytest.yaml | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/.github/workflows/pytest.yaml b/.github/workflows/pytest.yaml index a61fb5e..d169d51 100644 --- a/.github/workflows/pytest.yaml +++ b/.github/workflows/pytest.yaml @@ -3,10 +3,9 @@ name: Run Tests on: push: branches: - - main + - master pull_request: - branches: - - main + jobs: test: From 8bd6925128ba16f964e0db1d84b38ae1b15d03f0 Mon Sep 17 00:00:00 2001 From: Jochem Smit Date: Thu, 19 Oct 2023 15:19:28 +0200 Subject: [PATCH 3/6] pytest missing --- .github/workflows/pytest.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/pytest.yaml b/.github/workflows/pytest.yaml index d169d51..3a3d2f5 100644 --- a/.github/workflows/pytest.yaml +++ b/.github/workflows/pytest.yaml @@ -28,6 +28,7 @@ jobs: run: | python -m pip install --upgrade pip pip install -r requirements.txt + pip install pytest - name: Run tests run: pytest \ No newline at end of file From 5a1ec446cd7e7dcc07b77d5ced415f6328eecc26 Mon Sep 17 00:00:00 2001 From: Jochem Smit Date: Thu, 19 Oct 2023 15:23:42 +0200 Subject: [PATCH 4/6] ye oldy versions only --- .github/workflows/pytest.yaml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/pytest.yaml b/.github/workflows/pytest.yaml index 3a3d2f5..25993cd 100644 --- a/.github/workflows/pytest.yaml +++ b/.github/workflows/pytest.yaml @@ -11,9 +11,9 @@ jobs: test: strategy: matrix: - python-version: [3.6, 3.7, 3.8, 3.9, 3.10, 3.11] + python-version: [3.6, 3.7, 3.8, 3.9] - runs-on: ubuntu-latest + runs-on: ubuntu-20.04 steps: - name: Checkout code From 8333b63132272c19d739160307307f72f6c33982 Mon Sep 17 00:00:00 2001 From: Jochem Smit Date: Thu, 19 Oct 2023 15:24:59 +0200 Subject: [PATCH 5/6] install the project --- .github/workflows/pytest.yaml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/pytest.yaml b/.github/workflows/pytest.yaml index 25993cd..111063b 100644 --- a/.github/workflows/pytest.yaml +++ b/.github/workflows/pytest.yaml @@ -28,6 +28,7 @@ jobs: run: | python -m pip install --upgrade pip pip install -r requirements.txt + pip install . pip install pytest - name: Run tests From d087ade5dc98b69b799d263f3cd54497cbd0f9e5 Mon Sep 17 00:00:00 2001 From: Jochem Smit Date: Thu, 19 Oct 2023 15:27:06 +0200 Subject: [PATCH 6/6] there is no future in python 3.6 --- tests/test_hdxrate.py | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/tests/test_hdxrate.py b/tests/test_hdxrate.py index 0ae7a03..9a40694 100644 --- a/tests/test_hdxrate.py +++ b/tests/test_hdxrate.py @@ -1,6 +1,5 @@ """Tests for `hdxrate` package.""" -from __future__ import annotations import numpy as np from hdxrate import k_int_from_sequence from hdxrate.hdxrate import get_side_chain_dictionary @@ -15,12 +14,12 @@ @pytest.fixture() -def seq1() -> list[str]: +def seq1(): return list("AAAWADEAA") @pytest.fixture() -def seq2() -> list[str]: +def seq2(): """sequence two a sequence of the pairwise combination of all side chains""" k_reference = {"D": 3.87, "E": 4.33, "H": 7.0} # DH chains_dict = get_side_chain_dictionary(278, 8, k_reference)