Compare commits

..

No commits in common. "main" and "v0.1.0" have entirely different histories.
main ... v0.1.0

15 changed files with 301 additions and 471 deletions

4
.gitignore vendored
View File

@ -19,7 +19,6 @@ dist/
eggs/ eggs/
lib/ lib/
lib64/ lib64/
lib64
parts/ parts/
sdist/ sdist/
var/ var/
@ -29,7 +28,6 @@ venv/
*.egg-info/ *.egg-info/
.installed.cfg .installed.cfg
*.egg *.egg
pyvenv.cfg
# Installer logs # Installer logs
pip-log.txt pip-log.txt
@ -81,5 +79,3 @@ docs/_build/
# Rust files # Rust files
src/main.rs src/main.rs
blog.md

View File

@ -1,9 +0,0 @@
# `v0.2.0`
- Bumped maturin version to `1.7.8`
- Bumped PyO3 version to `0.23.3`
- Changed from `init()` syntax to direct `tic()` and `toc()` calls. `init()` syntax still supported.
- Multiple timing operations now possible by assigning output of `tic()` and passing it to `toc()`
- Added PyO3 features to support Windows cross-compilation
# `v0.1.0`
- Initial release

189
Cargo.lock generated
View File

@ -1,6 +1,6 @@
# This file is automatically @generated by Cargo. # This file is automatically @generated by Cargo.
# It is not intended for manual editing. # It is not intended for manual editing.
version = 4 version = 3
[[package]] [[package]]
name = "autocfg" name = "autocfg"
@ -9,13 +9,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa"
[[package]] [[package]]
name = "cc" name = "bitflags"
version = "1.1.13" version = "1.3.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "72db2f7947ecee9b03b510377e8bb9077afa27176fdbff55c51027e976fdcc48" checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
dependencies = [
"shlex",
]
[[package]] [[package]]
name = "cfg-if" name = "cfg-if"
@ -23,17 +20,11 @@ version = "1.0.0"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
[[package]]
name = "heck"
version = "0.5.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea"
[[package]] [[package]]
name = "indoc" name = "indoc"
version = "2.0.5" version = "1.0.9"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b248f5224d1d606005e02c97f5aa4e88eeb230488bcc03bc9ca4d7991399f2b5" checksum = "bfa799dd5ed20a7e349f3b4639aa80d74549c81716d9ec4f994c9b5815598306"
[[package]] [[package]]
name = "libc" name = "libc"
@ -41,6 +32,16 @@ version = "0.2.149"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a08173bc88b7955d1b3145aa561539096c421ac8debde8cbc3612ec635fee29b" checksum = "a08173bc88b7955d1b3145aa561539096c421ac8debde8cbc3612ec635fee29b"
[[package]]
name = "lock_api"
version = "0.4.11"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "3c168f8615b12bc01f9c17e2eb0cc07dcae1940121185446edc3744920e8ef45"
dependencies = [
"autocfg",
"scopeguard",
]
[[package]] [[package]]
name = "memoffset" name = "memoffset"
version = "0.9.0" version = "0.9.0"
@ -57,32 +58,48 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d" checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d"
[[package]] [[package]]
name = "portable-atomic" name = "parking_lot"
version = "1.10.0" version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "280dc24453071f1b63954171985a0b0d30058d287960968b9b2aca264c8d4ee6" checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f"
dependencies = [
"lock_api",
"parking_lot_core",
]
[[package]]
name = "parking_lot_core"
version = "0.9.9"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4c42a9226546d68acdd9c0a280d17ce19bfe27a46bf68784e4066115788d008e"
dependencies = [
"cfg-if",
"libc",
"redox_syscall",
"smallvec",
"windows-targets",
]
[[package]] [[package]]
name = "proc-macro2" name = "proc-macro2"
version = "1.0.92" version = "1.0.69"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "37d3544b3f2748c54e147655edb5025752e2303145b5aefb3c3ea2c78b973bb0" checksum = "134c189feb4956b20f6f547d2cf727d4c0fe06722b20a0eec87ed445a97f92da"
dependencies = [ dependencies = [
"unicode-ident", "unicode-ident",
] ]
[[package]] [[package]]
name = "pyo3" name = "pyo3"
version = "0.23.3" version = "0.19.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "e484fd2c8b4cb67ab05a318f1fd6fa8f199fcc30819f08f07d200809dba26c15" checksum = "e681a6cfdc4adcc93b4d3cf993749a4552018ee0a9b65fc0ccfad74352c72a38"
dependencies = [ dependencies = [
"cfg-if", "cfg-if",
"indoc", "indoc",
"libc", "libc",
"memoffset", "memoffset",
"once_cell", "parking_lot",
"portable-atomic",
"pyo3-build-config", "pyo3-build-config",
"pyo3-ffi", "pyo3-ffi",
"pyo3-macros", "pyo3-macros",
@ -91,20 +108,19 @@ dependencies = [
[[package]] [[package]]
name = "pyo3-build-config" name = "pyo3-build-config"
version = "0.23.3" version = "0.19.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "dc0e0469a84f208e20044b98965e1561028180219e35352a2afaf2b942beff3b" checksum = "076c73d0bc438f7a4ef6fdd0c3bb4732149136abd952b110ac93e4edb13a6ba5"
dependencies = [ dependencies = [
"once_cell", "once_cell",
"python3-dll-a",
"target-lexicon", "target-lexicon",
] ]
[[package]] [[package]]
name = "pyo3-ffi" name = "pyo3-ffi"
version = "0.23.3" version = "0.19.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "eb1547a7f9966f6f1a0f0227564a9945fe36b90da5a93b3933fc3dc03fae372d" checksum = "e53cee42e77ebe256066ba8aa77eff722b3bb91f3419177cf4cd0f304d3284d9"
dependencies = [ dependencies = [
"libc", "libc",
"pyo3-build-config", "pyo3-build-config",
@ -112,9 +128,9 @@ dependencies = [
[[package]] [[package]]
name = "pyo3-macros" name = "pyo3-macros"
version = "0.23.3" version = "0.19.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "fdb6da8ec6fa5cedd1626c886fc8749bdcbb09424a86461eb8cdf096b7c33257" checksum = "dfeb4c99597e136528c6dd7d5e3de5434d1ceaf487436a3f03b2d56b6fc9efd1"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"pyo3-macros-backend", "pyo3-macros-backend",
@ -124,46 +140,50 @@ dependencies = [
[[package]] [[package]]
name = "pyo3-macros-backend" name = "pyo3-macros-backend"
version = "0.23.3" version = "0.19.2"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "38a385202ff5a92791168b1136afae5059d3ac118457bb7bc304c197c2d33e7d" checksum = "947dc12175c254889edc0c02e399476c2f652b4b9ebd123aa655c224de259536"
dependencies = [ dependencies = [
"heck",
"proc-macro2", "proc-macro2",
"pyo3-build-config",
"quote", "quote",
"syn", "syn",
] ]
[[package]]
name = "python3-dll-a"
version = "0.2.11"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9b9e268ee1be609e93a13eb06839f68f67e5fe0fb4049834d261c2d5091c1b6d"
dependencies = [
"cc",
]
[[package]] [[package]]
name = "quote" name = "quote"
version = "1.0.37" version = "1.0.33"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af" checksum = "5267fca4496028628a95160fc423a33e8b2e6af8a5302579e322e4b520293cae"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
] ]
[[package]] [[package]]
name = "shlex" name = "redox_syscall"
version = "1.3.0" version = "0.4.1"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa"
dependencies = [
"bitflags",
]
[[package]]
name = "scopeguard"
version = "1.2.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
[[package]]
name = "smallvec"
version = "1.11.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "942b4a808e05215192e39f4ab80813e599068285906cc91aa64f923db842bd5a"
[[package]] [[package]]
name = "syn" name = "syn"
version = "2.0.90" version = "1.0.109"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "919d3b74a5dd0ccd15aeb8f93e7006bd9e14c295087c9896a110f490752bcf31" checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237"
dependencies = [ dependencies = [
"proc-macro2", "proc-macro2",
"quote", "quote",
@ -172,13 +192,13 @@ dependencies = [
[[package]] [[package]]
name = "target-lexicon" name = "target-lexicon"
version = "0.12.16" version = "0.12.12"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1" checksum = "14c39fd04924ca3a864207c66fc2cd7d22d7c016007f9ce846cbb9326331930a"
[[package]] [[package]]
name = "tictoc" name = "tictoc"
version = "0.2.0" version = "0.1.0"
dependencies = [ dependencies = [
"pyo3", "pyo3",
] ]
@ -191,6 +211,63 @@ checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b"
[[package]] [[package]]
name = "unindent" name = "unindent"
version = "0.2.3" version = "0.1.11"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "c7de7d73e1754487cb58364ee906a499937a0dfabd86bcb980fa99ec8c8fa2ce" checksum = "e1766d682d402817b5ac4490b3c3002d91dfa0d22812f341609f97b08757359c"
[[package]]
name = "windows-targets"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c"
dependencies = [
"windows_aarch64_gnullvm",
"windows_aarch64_msvc",
"windows_i686_gnu",
"windows_i686_msvc",
"windows_x86_64_gnu",
"windows_x86_64_gnullvm",
"windows_x86_64_msvc",
]
[[package]]
name = "windows_aarch64_gnullvm"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8"
[[package]]
name = "windows_aarch64_msvc"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc"
[[package]]
name = "windows_i686_gnu"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e"
[[package]]
name = "windows_i686_msvc"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406"
[[package]]
name = "windows_x86_64_gnu"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e"
[[package]]
name = "windows_x86_64_gnullvm"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc"
[[package]]
name = "windows_x86_64_msvc"
version = "0.48.5"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538"

View File

@ -1,6 +1,6 @@
[package] [package]
name = "tictoc" name = "tictoc"
version = "0.2.0" version = "0.1.0"
edition = "2021" edition = "2021"
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
@ -9,4 +9,4 @@ name = "tictoc"
crate-type = ["cdylib"] crate-type = ["cdylib"]
[dependencies] [dependencies]
pyo3 = { version = "0.23.3", features = ["extension-module", "generate-import-lib"] } pyo3 = "0.19.0"

View File

@ -1,7 +0,0 @@
Copyright (c) 2023-2024 Andrew Conlin
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

View File

@ -1,6 +1,10 @@
![A logo with the word tictoc followed by a stopwatch emoji](./.docs/logoLightMode.png#gh-light-mode-only)
![A logo with the word tictoc followed by a stopwatch emoji](./.docs/logoDarkMode.png#gh-dark-mode-only)
# Fast, simple and accurate Python timing. Written in Rust. # Fast, simple and accurate Python timing. Written in Rust.
[![PyPI Downloads](https://static.pepy.tech/badge/tictoc/month)](https://pepy.tech/projects/tictoc) [![PyPI Downloads](https://static.pepy.tech/badge/tictoc)](https://pepy.tech/projects/tictoc) ![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 ## Installation
Install with [pip](https://pypi.org/project/pip). Install with [pip](https://pypi.org/project/pip).
@ -9,76 +13,40 @@ $ python -m pip install tictoc
``` ```
## Usage ## Usage
Import. Import and initialise. **The module must be initialised to be used!**
```python ```python
from tictoc import tic,toc import tictoc
t = tictoc.init()
``` ```
Begin timing with `tic()`, and stop with `toc()`. Begin timing with `tic()`, and stop with `toc()`.
```python ```python
tic() t.tic()
# some code # some code
toc() t.toc()
``` ```
A call to `tic()` can be followed with multiple `toc()` calls. Each will print the time elapsed since the most recent `tic()` call. When `toc` is called, the results are saved. They can be accessed with the following syntax:
```python ```python
tic() t.results.{unit}
time.sleep(3)
toc()
# >>> The elapsed time was 3.000132333 seconds.
time.sleep(3)
toc()
# >>> The elapsed time was 6.000383124 seconds.
```
For more complex timing operations, you can assign the output of `tic()` and pass it as an input to `toc()`.
> [!NOTE]
> This syntax cannot be used interchangeably with the default syntax above. Any call to `tic()` resets the global timer.
```python
firstTic = tic()
time.sleep(3)
secondTic = tic()
time.sleep(1)
toc(firstTic)
# >>> The elapsed time was 4.000317251 seconds.
time.sleep(3)
toc(secondTic)
# >>> The elapsed time was 4.000312568 seconds.
```
Any call to `toc()` will print the elapsed time in seconds. You can save the results with full precision by assigning the output of `toc()`.
```python
tic()
# some code
results = toc()
``` ```
The available units are: The available units are:
```python ```python
results.nanos # u128 t.results.nanos # u128
results.micros # u128 t.results.micros # u128
results.millis # u128 t.results.millis # u128
results.seconds # f64 t.results.seconds # f64
``` ```
## Full example ## Full example
```python ```python
import time import time
from tictoc import tic,toc
tic() # start timing import tictoc
t = tictoc.init()
t.tic() # start timing
time.sleep(3) # sleep for 3 seconds time.sleep(3) # sleep for 3 seconds
toc() # stop timing t.toc() # stop timing
# >>> The elapsed time was 3.000132333 seconds.
firstTic = tic() print(t.results.seconds)
time.sleep(3) # >>> 3.000457715
secondTic = tic()
time.sleep(1)
toc(firstTic)
# >>> The elapsed time was 4.000317251 seconds.
time.sleep(3)
toc(secondTic)
# >>> The elapsed time was 4.000312568 seconds.
tic()
results = toc()
print(results.nanos)
# >>> 2825
``` ```

View File

@ -1,35 +0,0 @@
# All OSes build for CPython 3.8, CPython 3.9, CPython 3.10, CPython 3.11, CPython 3.12, CPython 3.13, CPython 3.13t, PyPy 3.9, PyPy 3.10
### Source distribution
maturin sdist
### Linux
# - aarch64-unknown-linux-gnu
# - i686-unknown-linux-gnu
# - x86_64-unknown-linux-gnu
# Add Rust targets
rustup target add aarch64-unknown-linux-gnu i686-unknown-linux-gnu x86_64-unknown-linux-gnu
# Build wheels
maturin build --release --target aarch64-unknown-linux-gnu --compatibility manylinux2014 --auditwheel repair --find-interpreter --zig --quiet
maturin build --release --target i686-unknown-linux-gnu --compatibility manylinux2014 --auditwheel repair --find-interpreter --zig --quiet
maturin build --release --target x86_64-unknown-linux-gnu --compatibility manylinux2014 --auditwheel repair --find-interpreter --zig --quiet
### macOS
# - aarch64-apple-darwin
# - x86_64-apple-darwin
# Add Rust targets
rustup target add aarch64-apple-darwin x86_64-apple-darwin
# Build wheels
maturin build --release --target aarch64-apple-darwin --compatibility manylinux2014 --auditwheel repair --find-interpreter --zig --quiet
maturin build --release --target x86_64-apple-darwin --compatibility manylinux2014 --auditwheel repair --find-interpreter --zig --quiet
### Windows
# - x86_64-pc-windows-msvc
# Add Rust target
rustup target add x86_64-pc-windows-msvc
# Install LLVM for llvm-dlltool
sudo apt install llvm -y
# Force use of cargo-xwin for building
export MATURIN_USE_XWIN=1
# Build wheels
maturin build --release --target x86_64-pc-windows-msvc --compatibility manylinux2014 --auditwheel repair --find-interpreter --quiet

View File

@ -4,24 +4,13 @@ build-backend = "maturin"
[project] [project]
name = "tictoc" name = "tictoc"
authors = [
{name = "Andrew Conlin", email = "andrew@andrewconl.in"},
]
description = "Fast, simple and accurate Python timing. Written in Rust."
readme = "README.md"
license = {file = "LICENSE.md"}
requires-python = ">=3.8" requires-python = ">=3.8"
classifiers = [ classifiers = [
"Programming Language :: Rust", "Programming Language :: Rust",
"Programming Language :: Python :: Implementation :: CPython", "Programming Language :: Python :: Implementation :: CPython",
"Programming Language :: Python :: Implementation :: PyPy", "Programming Language :: Python :: Implementation :: PyPy",
"License :: OSI Approved :: MIT License",
] ]
dynamic = ["version"] dynamic = ["version"]
[project.urls]
Repository = "https://git.andrewconl.in/andrew/tictoc.git"
Changelog = "https://git.andrewconl.in/andrew/tictoc/src/branch/main/CHANGELOG.md"
[tool.maturin] [tool.maturin]
features = ["pyo3/extension-module"] features = ["pyo3/extension-module"]

View File

@ -3,7 +3,7 @@ charset-normalizer==3.3.2
coverage==7.3.2 coverage==7.3.2
idna==3.4 idna==3.4
iniconfig==2.0.0 iniconfig==2.0.0
maturin==1.7.8 maturin==1.3.1
packaging==23.2 packaging==23.2
pluggy==1.3.0 pluggy==1.3.0
pytest==7.4.3 pytest==7.4.3

25
run.py
View File

@ -1,22 +1,11 @@
import time import time
from tictoc import tic,toc
tic() # start timing import tictoc
time.sleep(3) # sleep for 3 seconds
toc() # stop timing
# >>> The elapsed time was 3.000132333 seconds.
firstTic = tic() t = tictoc.init()
time.sleep(3)
secondTic = tic()
time.sleep(1)
toc(firstTic)
# >>> The elapsed time was 4.000317251 seconds.
time.sleep(3)
toc(secondTic)
# >>> The elapsed time was 4.000312568 seconds.
tic() t.tic() # start timing
results = toc() time.sleep(3) # sleep for 3 seconds
print(results.nanos) t.toc() # stop timing
# >>> 2825
print(t.results.seconds)

View File

@ -1,133 +1,92 @@
use pyo3::prelude::*; use pyo3::prelude::*;
use std::time::Instant;
use pyo3::exceptions::PyException;
#[pymodule] #[pyclass]
mod tictoc { #[derive(Clone)]
use super::*; struct Results {
use std::time::Instant; #[pyo3(get)]
use pyo3::exceptions::PyException; nanos: u128,
#[pyo3(get)]
micros: u128,
#[pyo3(get)]
millis: u128,
#[pyo3(get)]
seconds: f64,
}
#[pyclass(name = "results")] #[pyclass(module = "tictoc", name = "init")]
#[derive(Clone, Debug, PartialEq)] struct Init {
struct Results { time: Instant,
#[pyo3(get)] #[pyo3(get)]
nanos: u128, results: Results,
#[pyo3(get)] status: bool,
micros: u128, }
#[pyo3(get)]
millis: u128,
#[pyo3(get)]
seconds: f64,
}
#[derive(Clone)] #[pymethods]
#[pyclass(name = "init")] impl Init {
pub struct Init { #[new]
time: Instant, fn new() -> Self {
#[pyo3(get)] let res = Results {
results: Results, nanos: 0,
status: bool, micros: 0,
} millis: 0,
seconds: 0.0,
#[pymethods] };
impl Init { Init {
#[new] time: Instant::now(),
fn new() -> Self { results: res,
let res = Results { status: false,
nanos: 0,
micros: 0,
millis: 0,
seconds: 0.0,
};
Init {
time: Instant::now(),
results: res,
status: false,
}
}
fn tic(&mut self) -> PyResult<Init> {
self.time = Instant::now();
self.status = true;
Ok(Init::new())
}
#[pyo3(signature = (tic=None))]
fn toc(&mut self, tic: Option<Init>) -> PyResult<Results> {
let elapsed_time = match tic {
Some(ref tic) => tic.time.elapsed(),
None => self.time.elapsed(),
};
let status = match tic {
Some(ref _tic) => true,
None => self.status,
};
if status == false {
Err(PyException::new_err("tic() must be called before toc()"))
} else {
self.results = Results {
nanos: elapsed_time.as_nanos(),
micros: elapsed_time.as_micros(),
millis: elapsed_time.as_millis(),
seconds: elapsed_time.as_secs_f64(),
};
println!("The elapsed time was {} seconds.",self.results.seconds);
Ok(self.results.clone())
}
} }
} }
#[test] fn tic(&mut self) {
fn test_new() { self.time = Instant::now();
let init = Init::new(); self.status = true;
assert_eq!(init.results.nanos,0);
} }
#[test] fn toc(&mut self) -> PyResult<()> {
fn test_tic() { if self.status == false {
let mut init = Init::new(); Err(PyException::new_err("tic() must be called before toc()"))
let time1 = init.time; } else {
let _ = init.tic(); let elapsed_time = self.time.elapsed();
let time2 = init.time; self.results = Results {
assert!(time2 > time1) nanos: elapsed_time.as_nanos(),
} micros: elapsed_time.as_micros(),
millis: elapsed_time.as_millis(),
#[test] seconds: elapsed_time.as_secs_f64(),
fn test_toc() { };
let mut init = Init::new(); Ok(())
let _ = init.tic(); }
println!("{}","test");
let _ = init.toc(None).unwrap();
assert!(init.results.nanos > 0);
}
#[test]
fn test_passing_tic_to_toc() {
let mut init = Init::new();
let tic_obj = init.tic().unwrap();
println!("{}","test");
let results = init.toc(Some(tic_obj)).unwrap();
assert!(init.results.nanos > 0);
assert_eq!(init.results,results)
}
#[test]
fn test_multiple_calls() {
let mut init = Init::new();
let first_tic = init.tic().unwrap();
println!("{}","test");
let second_tic = init.tic().unwrap();
println!("{}","test");
let results2 = init.toc(Some(second_tic)).unwrap();
let results = init.toc(Some(first_tic)).unwrap();
assert!(results.nanos > results2.nanos);
}
#[test]
fn test_toc_before_tic() {
let mut init = Init::new();
//assert!(init.toc(None).is_err())
pyo3::prepare_freethreaded_python();
let e = init.toc(None).unwrap_err();
assert_eq!(e.to_string(),"Exception: tic() must be called before toc()")
} }
} }
#[pymodule]
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)
}

View File

@ -1,41 +1,43 @@
import pytest import pytest
from tictoc import tic,toc import tictoc
import time import time
import math import math
@pytest.fixture
def t():
t = tictoc.init()
return t
class testFunctionality: class testFunctionality:
def testBasic(self): def testBasic(self, t):
tic() t.tic()
print("test") print("test")
results = toc() t.toc()
assert results.seconds > 0 assert t.results.seconds > 0
def testMultipleGlobalCalls(self): def testOverwrite(self, t):
tic() t.tic()
print("test") print("test")
results = toc() t.toc()
firstResult = t.results.seconds
print("test2") print("test2")
results2 = toc() t.toc()
secondResult = t.results.seconds
assert results.seconds < results2.seconds assert firstResult < secondResult
def testMultipleCalls(self):
first = tic()
print("test")
second = tic()
print("test2")
secondResult = toc(second).seconds
firstResult = toc(first).seconds
assert firstResult > secondResult
class testInvalid: class testInvalid:
def testNonTicInputForToc(self): def testNoInit(self):
with pytest.raises(Exception): with pytest.raises(Exception):
tic() t.tic()
print("test")
toc(1) def testTocBeforeTic(self, t):
with pytest.raises(Exception):
t.toc()
@pytest.mark.parametrize("sleepTime", [0.05, 0.5, 1]) @pytest.mark.parametrize("sleepTime", [0.05, 0.5, 1])
class testAccuracy: class testAccuracy:
@ -43,47 +45,42 @@ class testAccuracy:
def tol(self): def tol(self):
return 0.0006 return 0.0006
def testSingleCall(self, sleepTime, tol): def testSingleCall(self, t, sleepTime, tol):
tic() t.tic()
time.sleep(sleepTime) time.sleep(sleepTime)
results = toc() t.toc()
assert (results.seconds < sleepTime+tol) assert (t.results.seconds > sleepTime) & (
t.results.seconds < (t.results.seconds + tol)
)
def testMultipleGlobalCalls(self, sleepTime, tol): def testMultipleCalls(self, t, sleepTime, tol):
tic() t.tic()
time.sleep(sleepTime) time.sleep(sleepTime)
toc() t.toc()
time.sleep(sleepTime) time.sleep(sleepTime)
results = toc() t.toc()
assert (results.seconds < (sleepTime * 2)+tol) assert (t.results.seconds > sleepTime * 2) & (
t.results.seconds < (t.results.seconds + tol)
)
def testMultipleCalls(self, sleepTime, tol):
first = tic()
time.sleep(sleepTime)
second = tic()
time.sleep(sleepTime)
results2 = toc(second)
results = toc(first)
assert (results.seconds < (sleepTime * 2)+tol)
assert (results2.seconds < sleepTime+tol)
class testConsistency: class testConsistency:
def testMicros(self): def testMicros(self, t):
tic() t.tic()
print("test") print("test")
results = toc() t.toc()
assert results.micros == (math.floor(results.nanos * pow(10, -3))) assert t.results.micros == (math.floor(t.results.nanos * pow(10, -3)))
def testMillis(self): def testMillis(self, t):
tic() t.tic()
print("test") print("test")
results = toc() t.toc()
assert results.millis == (math.floor(results.nanos * pow(10, -6))) assert t.results.millis == (math.floor(t.results.nanos * pow(10, -6)))
def testSeconds(self): def testSeconds(self, t):
tic() t.tic()
print("test") print("test")
results = toc() t.toc()
assert results.seconds == round( assert t.results.seconds == round(
(results.nanos * pow(10, -9)), 9 (t.results.nanos * pow(10, -9)), 9
) # f64 vs u128, hence the round ) # f64 vs u128, hence the round

View File

@ -1,78 +0,0 @@
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()
@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+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)+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

View File

@ -1,7 +0,0 @@
import pytest
import tictoc
def testTocBeforeTic():
with pytest.raises(Exception):
t = tictoc.init()
t.toc()

View File

@ -1,9 +0,0 @@
from .tictoc import *
__doc__ = tictoc.__doc__
if hasattr(tictoc, "__all__"):
__all__ = tictoc.__all__
t = tictoc.init();
tic = t.tic;
toc = t.toc;