[20/11/23] Merge dev changes to main
This commit is contained in:
commit
e38bbac712
35
.github/workflows/python.yml
vendored
Normal file
35
.github/workflows/python.yml
vendored
Normal file
@ -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
|
23
.github/workflows/rust.yml
vendored
Normal file
23
.github/workflows/rust.yml
vendored
Normal file
@ -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
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -72,7 +72,7 @@ docs/_build/
|
||||
.python-version
|
||||
|
||||
# GitHub Actions
|
||||
.github/
|
||||
#.github/
|
||||
|
||||
# pyenv
|
||||
.env/
|
||||
|
@ -3,8 +3,11 @@
|
||||
|
||||
# Fast, simple and accurate Python timing. Written in Rust.
|
||||
|
||||

|
||||

|
||||
|
||||
## Installation
|
||||
Install with [pip](https://pypi.org/project/pip):
|
||||
Install with [pip](https://pypi.org/project/pip).
|
||||
```bash
|
||||
$ python -m pip install tictoc
|
||||
```
|
||||
|
@ -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
|
||||
|
37
src/lib.rs
37
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,14 +36,19 @@ 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) {
|
||||
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(),
|
||||
@ -49,6 +56,8 @@ impl Init {
|
||||
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::<Init>()?;
|
||||
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)
|
||||
}
|
||||
|
4
tests/pytest.ini
Normal file
4
tests/pytest.ini
Normal file
@ -0,0 +1,4 @@
|
||||
[pytest]
|
||||
python_files = test*.py
|
||||
python_classes = test
|
||||
python_functions = test*
|
86
tests/test.py
Normal file
86
tests/test.py
Normal file
@ -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
|
Loading…
x
Reference in New Issue
Block a user