diff --git a/.github/workflows/python.yml b/.github/workflows/python.yml new file mode 100644 index 0000000..2ee5ad0 --- /dev/null +++ b/.github/workflows/python.yml @@ -0,0 +1,35 @@ +name: Python + +on: + push: + branches: [ "dev" ] + +permissions: + contents: read + +jobs: + build: + + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + - name: Set up Python 3.11.2 + uses: actions/setup-python@v3 + with: + python-version: "3.11.2" + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install flake8 pytest + if [ -f requirements.txt ]; then pip install -r requirements.txt; fi + pip install . + - name: Lint with flake8 + run: | + # stop the build if there are Python syntax errors or undefined names + flake8 . --count --select=E9,F63,F7,F82 --show-source --statistics + # exit-zero treats all errors as warnings. The GitHub editor is 127 chars wide + flake8 . --count --exit-zero --max-complexity=10 --max-line-length=127 --statistics + - name: Test with pytest + run: | + pytest -v tests diff --git a/.github/workflows/rust.yml b/.github/workflows/rust.yml new file mode 100644 index 0000000..83af3ea --- /dev/null +++ b/.github/workflows/rust.yml @@ -0,0 +1,23 @@ +name: Rust + +on: + push: + branches: [ "dev" ] + +permissions: + contents: read + +env: + CARGO_TERM_COLOR: always + +jobs: + build: + + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v3 + - name: Build + run: cargo build --verbose + - name: Run tests + run: cargo test --verbose diff --git a/.gitignore b/.gitignore index 9845fe0..7e88271 100644 --- a/.gitignore +++ b/.gitignore @@ -72,7 +72,7 @@ docs/_build/ .python-version # GitHub Actions -.github/ +#.github/ # pyenv .env/ diff --git a/README.md b/README.md index c56982d..081e1ac 100644 --- a/README.md +++ b/README.md @@ -3,8 +3,11 @@ # Fast, simple and accurate Python timing. Written in Rust. +![badge](https://github.com/andrwcnln/tictoc-py/actions/workflows/python.yml/badge.svg) +![badge](https://github.com/andrwcnln/tictoc-py/actions/workflows/rust.yml/badge.svg) + ## Installation -Install with [pip](https://pypi.org/project/pip): +Install with [pip](https://pypi.org/project/pip). ```bash $ python -m pip install tictoc ``` diff --git a/requirements.txt b/requirements.txt index 2b076d0..db9c234 100644 --- a/requirements.txt +++ b/requirements.txt @@ -1,6 +1,11 @@ certifi==2023.7.22 charset-normalizer==3.3.2 +coverage==7.3.2 idna==3.4 +iniconfig==2.0.0 maturin==1.3.1 +packaging==23.2 +pluggy==1.3.0 +pytest==7.4.3 requests==2.31.0 urllib3==2.0.7 diff --git a/src/lib.rs b/src/lib.rs index d3cf904..4974904 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,5 +1,6 @@ use pyo3::prelude::*; use std::time::Instant; +use pyo3::exceptions::PyException; #[pyclass] #[derive(Clone)] @@ -19,6 +20,7 @@ struct Init { time: Instant, #[pyo3(get)] results: Results, + status: bool, } #[pymethods] @@ -34,21 +36,28 @@ impl Init { Init { time: Instant::now(), results: res, + status: false, } } fn tic(&mut self) { - self.time = Instant::now() + self.time = Instant::now(); + self.status = true; } - fn toc(&mut self) { - let elapsed_time = self.time.elapsed(); - self.results = Results { - nanos: elapsed_time.as_nanos(), - micros: elapsed_time.as_micros(), - millis: elapsed_time.as_millis(), - seconds: elapsed_time.as_secs_f64(), - }; + fn toc(&mut self) -> PyResult<()> { + if self.status == false { + Err(PyException::new_err("tic() must be called before toc()")) + } else { + let elapsed_time = self.time.elapsed(); + self.results = Results { + nanos: elapsed_time.as_nanos(), + micros: elapsed_time.as_micros(), + millis: elapsed_time.as_millis(), + seconds: elapsed_time.as_secs_f64(), + }; + Ok(()) + } } } @@ -57,3 +66,27 @@ fn tictoc(_py: Python<'_>, m: &PyModule) -> PyResult<()> { m.add_class::()?; Ok(()) } + +#[test] +fn test_new() { + let init = Init::new(); + assert_eq!(init.results.nanos,0); +} + +#[test] +fn test_tic() { + let mut init = Init::new(); + let time1 = init.time; + init.tic(); + let time2 = init.time; + assert!(time2 > time1) +} + +#[test] +fn test_toc() { + let mut init = Init::new(); + init.tic(); + println!("{}","test"); + let _ = init.toc(); + assert!(init.results.nanos > 0) +} diff --git a/tests/pytest.ini b/tests/pytest.ini new file mode 100644 index 0000000..2779ef8 --- /dev/null +++ b/tests/pytest.ini @@ -0,0 +1,4 @@ +[pytest] +python_files = test*.py +python_classes = test +python_functions = test* diff --git a/tests/test.py b/tests/test.py new file mode 100644 index 0000000..b019fd2 --- /dev/null +++ b/tests/test.py @@ -0,0 +1,86 @@ +import pytest +import tictoc +import time +import math + + +@pytest.fixture +def t(): + t = tictoc.init() + return t + + +class testFunctionality: + def testBasic(self, t): + t.tic() + print("test") + t.toc() + assert t.results.seconds > 0 + + def testOverwrite(self, t): + t.tic() + print("test") + t.toc() + firstResult = t.results.seconds + print("test2") + t.toc() + secondResult = t.results.seconds + + assert firstResult < secondResult + + +class testInvalid: + def testNoInit(self): + with pytest.raises(Exception): + t.tic() + + def testTocBeforeTic(self, t): + with pytest.raises(Exception): + t.toc() + + +@pytest.mark.parametrize("sleepTime", [0.05, 0.5, 1]) +class testAccuracy: + @pytest.fixture(scope="class") + def tol(self): + return 0.0006 + + def testSingleCall(self, t, sleepTime, tol): + t.tic() + time.sleep(sleepTime) + t.toc() + assert (t.results.seconds > sleepTime) & ( + t.results.seconds < (t.results.seconds + tol) + ) + + def testMultipleCalls(self, t, sleepTime, tol): + t.tic() + time.sleep(sleepTime) + t.toc() + time.sleep(sleepTime) + t.toc() + assert (t.results.seconds > sleepTime * 2) & ( + t.results.seconds < (t.results.seconds + tol) + ) + + +class testConsistency: + def testMicros(self, t): + t.tic() + print("test") + t.toc() + assert t.results.micros == (math.floor(t.results.nanos * pow(10, -3))) + + def testMillis(self, t): + t.tic() + print("test") + t.toc() + assert t.results.millis == (math.floor(t.results.nanos * pow(10, -6))) + + def testSeconds(self, t): + t.tic() + print("test") + t.toc() + assert t.results.seconds == round( + (t.results.nanos * pow(10, -9)), 9 + ) # f64 vs u128, hence the round