Compare commits
19 Commits
6343fbdd87
...
7d70bbe5de
Author | SHA1 | Date | |
---|---|---|---|
7d70bbe5de | |||
982ef8b813 | |||
a3b819ded5 | |||
08d918675d | |||
2219a148fd | |||
932247bb7c | |||
c54e001cd7 | |||
a01349bf18 | |||
47f7beb14c | |||
2b03ba598b | |||
4a0c0de3cb | |||
5b2a23d06a | |||
2a3087ad35 | |||
bec55635f9 | |||
a5475a460b | |||
3c4978bef8 | |||
c47dd577f5 | |||
6a4e818144 | |||
3f32df0bb6 |
2
.gitignore
vendored
2
.gitignore
vendored
@ -19,6 +19,7 @@ dist/
|
||||
eggs/
|
||||
lib/
|
||||
lib64/
|
||||
lib64
|
||||
parts/
|
||||
sdist/
|
||||
var/
|
||||
@ -28,6 +29,7 @@ venv/
|
||||
*.egg-info/
|
||||
.installed.cfg
|
||||
*.egg
|
||||
pyvenv.cfg
|
||||
|
||||
# Installer logs
|
||||
pip-log.txt
|
||||
|
9
CHANGELOG.md
Normal file
9
CHANGELOG.md
Normal file
@ -0,0 +1,9 @@
|
||||
# `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
|
193
Cargo.lock
generated
193
Cargo.lock
generated
@ -1,6 +1,6 @@
|
||||
# This file is automatically @generated by Cargo.
|
||||
# It is not intended for manual editing.
|
||||
version = 3
|
||||
version = 4
|
||||
|
||||
[[package]]
|
||||
name = "autocfg"
|
||||
@ -9,10 +9,13 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa"
|
||||
|
||||
[[package]]
|
||||
name = "bitflags"
|
||||
version = "1.3.2"
|
||||
name = "cc"
|
||||
version = "1.1.13"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
|
||||
checksum = "72db2f7947ecee9b03b510377e8bb9077afa27176fdbff55c51027e976fdcc48"
|
||||
dependencies = [
|
||||
"shlex",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "cfg-if"
|
||||
@ -21,10 +24,16 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
|
||||
|
||||
[[package]]
|
||||
name = "indoc"
|
||||
version = "1.0.9"
|
||||
name = "heck"
|
||||
version = "0.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "bfa799dd5ed20a7e349f3b4639aa80d74549c81716d9ec4f994c9b5815598306"
|
||||
checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea"
|
||||
|
||||
[[package]]
|
||||
name = "indoc"
|
||||
version = "2.0.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b248f5224d1d606005e02c97f5aa4e88eeb230488bcc03bc9ca4d7991399f2b5"
|
||||
|
||||
[[package]]
|
||||
name = "libc"
|
||||
@ -32,16 +41,6 @@ version = "0.2.149"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
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]]
|
||||
name = "memoffset"
|
||||
version = "0.9.0"
|
||||
@ -58,48 +57,32 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d"
|
||||
|
||||
[[package]]
|
||||
name = "parking_lot"
|
||||
version = "0.12.1"
|
||||
name = "portable-atomic"
|
||||
version = "1.10.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
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",
|
||||
]
|
||||
checksum = "280dc24453071f1b63954171985a0b0d30058d287960968b9b2aca264c8d4ee6"
|
||||
|
||||
[[package]]
|
||||
name = "proc-macro2"
|
||||
version = "1.0.69"
|
||||
version = "1.0.92"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "134c189feb4956b20f6f547d2cf727d4c0fe06722b20a0eec87ed445a97f92da"
|
||||
checksum = "37d3544b3f2748c54e147655edb5025752e2303145b5aefb3c3ea2c78b973bb0"
|
||||
dependencies = [
|
||||
"unicode-ident",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "pyo3"
|
||||
version = "0.19.2"
|
||||
version = "0.23.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e681a6cfdc4adcc93b4d3cf993749a4552018ee0a9b65fc0ccfad74352c72a38"
|
||||
checksum = "e484fd2c8b4cb67ab05a318f1fd6fa8f199fcc30819f08f07d200809dba26c15"
|
||||
dependencies = [
|
||||
"cfg-if",
|
||||
"indoc",
|
||||
"libc",
|
||||
"memoffset",
|
||||
"parking_lot",
|
||||
"once_cell",
|
||||
"portable-atomic",
|
||||
"pyo3-build-config",
|
||||
"pyo3-ffi",
|
||||
"pyo3-macros",
|
||||
@ -108,19 +91,20 @@ dependencies = [
|
||||
|
||||
[[package]]
|
||||
name = "pyo3-build-config"
|
||||
version = "0.19.2"
|
||||
version = "0.23.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "076c73d0bc438f7a4ef6fdd0c3bb4732149136abd952b110ac93e4edb13a6ba5"
|
||||
checksum = "dc0e0469a84f208e20044b98965e1561028180219e35352a2afaf2b942beff3b"
|
||||
dependencies = [
|
||||
"once_cell",
|
||||
"python3-dll-a",
|
||||
"target-lexicon",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "pyo3-ffi"
|
||||
version = "0.19.2"
|
||||
version = "0.23.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e53cee42e77ebe256066ba8aa77eff722b3bb91f3419177cf4cd0f304d3284d9"
|
||||
checksum = "eb1547a7f9966f6f1a0f0227564a9945fe36b90da5a93b3933fc3dc03fae372d"
|
||||
dependencies = [
|
||||
"libc",
|
||||
"pyo3-build-config",
|
||||
@ -128,9 +112,9 @@ dependencies = [
|
||||
|
||||
[[package]]
|
||||
name = "pyo3-macros"
|
||||
version = "0.19.2"
|
||||
version = "0.23.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "dfeb4c99597e136528c6dd7d5e3de5434d1ceaf487436a3f03b2d56b6fc9efd1"
|
||||
checksum = "fdb6da8ec6fa5cedd1626c886fc8749bdcbb09424a86461eb8cdf096b7c33257"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"pyo3-macros-backend",
|
||||
@ -140,50 +124,46 @@ dependencies = [
|
||||
|
||||
[[package]]
|
||||
name = "pyo3-macros-backend"
|
||||
version = "0.19.2"
|
||||
version = "0.23.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "947dc12175c254889edc0c02e399476c2f652b4b9ebd123aa655c224de259536"
|
||||
checksum = "38a385202ff5a92791168b1136afae5059d3ac118457bb7bc304c197c2d33e7d"
|
||||
dependencies = [
|
||||
"heck",
|
||||
"proc-macro2",
|
||||
"pyo3-build-config",
|
||||
"quote",
|
||||
"syn",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "quote"
|
||||
version = "1.0.33"
|
||||
name = "python3-dll-a"
|
||||
version = "0.2.11"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5267fca4496028628a95160fc423a33e8b2e6af8a5302579e322e4b520293cae"
|
||||
checksum = "9b9e268ee1be609e93a13eb06839f68f67e5fe0fb4049834d261c2d5091c1b6d"
|
||||
dependencies = [
|
||||
"cc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "quote"
|
||||
version = "1.0.37"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "redox_syscall"
|
||||
version = "0.4.1"
|
||||
name = "shlex"
|
||||
version = "1.3.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
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"
|
||||
checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64"
|
||||
|
||||
[[package]]
|
||||
name = "syn"
|
||||
version = "1.0.109"
|
||||
version = "2.0.90"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237"
|
||||
checksum = "919d3b74a5dd0ccd15aeb8f93e7006bd9e14c295087c9896a110f490752bcf31"
|
||||
dependencies = [
|
||||
"proc-macro2",
|
||||
"quote",
|
||||
@ -192,13 +172,13 @@ dependencies = [
|
||||
|
||||
[[package]]
|
||||
name = "target-lexicon"
|
||||
version = "0.12.12"
|
||||
version = "0.12.16"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "14c39fd04924ca3a864207c66fc2cd7d22d7c016007f9ce846cbb9326331930a"
|
||||
checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1"
|
||||
|
||||
[[package]]
|
||||
name = "tictoc"
|
||||
version = "0.1.0"
|
||||
version = "0.2.0"
|
||||
dependencies = [
|
||||
"pyo3",
|
||||
]
|
||||
@ -211,63 +191,6 @@ checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b"
|
||||
|
||||
[[package]]
|
||||
name = "unindent"
|
||||
version = "0.1.11"
|
||||
version = "0.2.3"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
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"
|
||||
checksum = "c7de7d73e1754487cb58364ee906a499937a0dfabd86bcb980fa99ec8c8fa2ce"
|
||||
|
@ -1,6 +1,6 @@
|
||||
[package]
|
||||
name = "tictoc"
|
||||
version = "0.1.0"
|
||||
version = "0.2.0"
|
||||
edition = "2021"
|
||||
|
||||
# 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"]
|
||||
|
||||
[dependencies]
|
||||
pyo3 = "0.19.0"
|
||||
pyo3 = { version = "0.23.3", features = ["extension-module", "generate-import-lib"] }
|
||||
|
7
LICENSE.md
Normal file
7
LICENSE.md
Normal file
@ -0,0 +1,7 @@
|
||||
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.
|
77
README.md
77
README.md
@ -1,10 +1,5 @@
|
||||

|
||||

|
||||
|
||||
# Fast, simple and accurate Python timing. Written in Rust.
|
||||
|
||||

|
||||

|
||||

|
||||
|
||||
## Installation
|
||||
@ -14,40 +9,76 @@ $ python -m pip install tictoc
|
||||
```
|
||||
|
||||
## Usage
|
||||
Import and initialise. **The module must be initialised to be used!**
|
||||
Import.
|
||||
```python
|
||||
import tictoc
|
||||
t = tictoc.init()
|
||||
from tictoc import tic,toc
|
||||
```
|
||||
Begin timing with `tic()`, and stop with `toc()`.
|
||||
```python
|
||||
t.tic()
|
||||
tic()
|
||||
# some code
|
||||
t.toc()
|
||||
toc()
|
||||
```
|
||||
When `toc` is called, the results are saved. They can be accessed with the following syntax:
|
||||
A call to `tic()` can be followed with multiple `toc()` calls. Each will print the time elapsed since the most recent `tic()` call.
|
||||
```python
|
||||
t.results.{unit}
|
||||
tic()
|
||||
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:
|
||||
```python
|
||||
t.results.nanos # u128
|
||||
t.results.micros # u128
|
||||
t.results.millis # u128
|
||||
t.results.seconds # f64
|
||||
results.nanos # u128
|
||||
results.micros # u128
|
||||
results.millis # u128
|
||||
results.seconds # f64
|
||||
```
|
||||
|
||||
## Full example
|
||||
```python
|
||||
import time
|
||||
from tictoc import tic,toc
|
||||
|
||||
import tictoc
|
||||
t = tictoc.init()
|
||||
|
||||
t.tic() # start timing
|
||||
tic() # start timing
|
||||
time.sleep(3) # sleep for 3 seconds
|
||||
t.toc() # stop timing
|
||||
toc() # stop timing
|
||||
# >>> The elapsed time was 3.000132333 seconds.
|
||||
|
||||
print(t.results.seconds)
|
||||
# >>> 3.000457715
|
||||
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.
|
||||
|
||||
tic()
|
||||
results = toc()
|
||||
print(results.nanos)
|
||||
# >>> 2825
|
||||
```
|
||||
|
35
build.sh
Executable file
35
build.sh
Executable file
@ -0,0 +1,35 @@
|
||||
# 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
|
@ -4,13 +4,24 @@ build-backend = "maturin"
|
||||
|
||||
[project]
|
||||
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"
|
||||
classifiers = [
|
||||
"Programming Language :: Rust",
|
||||
"Programming Language :: Python :: Implementation :: CPython",
|
||||
"Programming Language :: Python :: Implementation :: PyPy",
|
||||
"License :: OSI Approved :: MIT License",
|
||||
]
|
||||
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]
|
||||
features = ["pyo3/extension-module"]
|
||||
|
@ -3,7 +3,7 @@ charset-normalizer==3.3.2
|
||||
coverage==7.3.2
|
||||
idna==3.4
|
||||
iniconfig==2.0.0
|
||||
maturin==1.3.1
|
||||
maturin==1.7.8
|
||||
packaging==23.2
|
||||
pluggy==1.3.0
|
||||
pytest==7.4.3
|
||||
|
25
run.py
25
run.py
@ -1,11 +1,22 @@
|
||||
import time
|
||||
from tictoc import tic,toc
|
||||
|
||||
import tictoc
|
||||
tic() # start timing
|
||||
time.sleep(3) # sleep for 3 seconds
|
||||
toc() # stop timing
|
||||
# >>> The elapsed time was 3.000132333 seconds.
|
||||
|
||||
t = tictoc.init()
|
||||
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.
|
||||
|
||||
t.tic() # start timing
|
||||
time.sleep(3) # sleep for 3 seconds
|
||||
t.toc() # stop timing
|
||||
|
||||
print(t.results.seconds)
|
||||
tic()
|
||||
results = toc()
|
||||
print(results.nanos)
|
||||
# >>> 2825
|
||||
|
211
src/lib.rs
211
src/lib.rs
@ -1,92 +1,133 @@
|
||||
use pyo3::prelude::*;
|
||||
use std::time::Instant;
|
||||
use pyo3::exceptions::PyException;
|
||||
|
||||
#[pyclass]
|
||||
#[derive(Clone)]
|
||||
struct Results {
|
||||
#[pyo3(get)]
|
||||
nanos: u128,
|
||||
#[pyo3(get)]
|
||||
micros: u128,
|
||||
#[pyo3(get)]
|
||||
millis: u128,
|
||||
#[pyo3(get)]
|
||||
seconds: f64,
|
||||
}
|
||||
|
||||
#[pyclass(module = "tictoc", name = "init")]
|
||||
struct Init {
|
||||
time: Instant,
|
||||
#[pyo3(get)]
|
||||
results: Results,
|
||||
status: bool,
|
||||
}
|
||||
|
||||
#[pymethods]
|
||||
impl Init {
|
||||
#[new]
|
||||
fn new() -> Self {
|
||||
let res = Results {
|
||||
nanos: 0,
|
||||
micros: 0,
|
||||
millis: 0,
|
||||
seconds: 0.0,
|
||||
};
|
||||
Init {
|
||||
time: Instant::now(),
|
||||
results: res,
|
||||
status: false,
|
||||
}
|
||||
}
|
||||
|
||||
fn tic(&mut self) {
|
||||
self.time = Instant::now();
|
||||
self.status = true;
|
||||
}
|
||||
|
||||
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(())
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[pymodule]
|
||||
fn tictoc(_py: Python<'_>, m: &PyModule) -> PyResult<()> {
|
||||
m.add_class::<Init>()?;
|
||||
Ok(())
|
||||
}
|
||||
mod tictoc {
|
||||
use super::*;
|
||||
use std::time::Instant;
|
||||
use pyo3::exceptions::PyException;
|
||||
|
||||
#[pyclass(name = "results")]
|
||||
#[derive(Clone, Debug, PartialEq)]
|
||||
struct Results {
|
||||
#[pyo3(get)]
|
||||
nanos: u128,
|
||||
#[pyo3(get)]
|
||||
micros: u128,
|
||||
#[pyo3(get)]
|
||||
millis: u128,
|
||||
#[pyo3(get)]
|
||||
seconds: f64,
|
||||
}
|
||||
|
||||
#[derive(Clone)]
|
||||
#[pyclass(name = "init")]
|
||||
pub struct Init {
|
||||
time: Instant,
|
||||
#[pyo3(get)]
|
||||
results: Results,
|
||||
status: bool,
|
||||
}
|
||||
|
||||
#[pymethods]
|
||||
impl Init {
|
||||
#[new]
|
||||
fn new() -> Self {
|
||||
let res = Results {
|
||||
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 test_new() {
|
||||
let init = Init::new();
|
||||
assert_eq!(init.results.nanos,0);
|
||||
}
|
||||
#[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;
|
||||
let _ = init.tic();
|
||||
let time2 = init.time;
|
||||
assert!(time2 > time1)
|
||||
}
|
||||
|
||||
#[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();
|
||||
let _ = init.tic();
|
||||
println!("{}","test");
|
||||
let _ = init.toc(None).unwrap();
|
||||
assert!(init.results.nanos > 0);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_toc() {
|
||||
let mut init = Init::new();
|
||||
init.tic();
|
||||
println!("{}","test");
|
||||
let _ = init.toc();
|
||||
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()")
|
||||
}
|
||||
}
|
||||
|
107
tests/test.py
107
tests/test.py
@ -1,43 +1,41 @@
|
||||
import pytest
|
||||
import tictoc
|
||||
from tictoc import tic,toc
|
||||
import time
|
||||
import math
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def t():
|
||||
t = tictoc.init()
|
||||
return t
|
||||
|
||||
|
||||
class testFunctionality:
|
||||
def testBasic(self, t):
|
||||
t.tic()
|
||||
def testBasic(self):
|
||||
tic()
|
||||
print("test")
|
||||
t.toc()
|
||||
assert t.results.seconds > 0
|
||||
results = toc()
|
||||
assert results.seconds > 0
|
||||
|
||||
def testOverwrite(self, t):
|
||||
t.tic()
|
||||
def testMultipleGlobalCalls(self):
|
||||
tic()
|
||||
print("test")
|
||||
t.toc()
|
||||
firstResult = t.results.seconds
|
||||
results = toc()
|
||||
print("test2")
|
||||
t.toc()
|
||||
secondResult = t.results.seconds
|
||||
results2 = toc()
|
||||
|
||||
assert firstResult < secondResult
|
||||
assert results.seconds < results2.seconds
|
||||
|
||||
def testMultipleCalls(self):
|
||||
first = tic()
|
||||
print("test")
|
||||
second = tic()
|
||||
print("test2")
|
||||
secondResult = toc(second).seconds
|
||||
firstResult = toc(first).seconds
|
||||
|
||||
assert firstResult > secondResult
|
||||
|
||||
|
||||
class testInvalid:
|
||||
def testNoInit(self):
|
||||
def testNonTicInputForToc(self):
|
||||
with pytest.raises(Exception):
|
||||
t.tic()
|
||||
|
||||
def testTocBeforeTic(self, t):
|
||||
with pytest.raises(Exception):
|
||||
t.toc()
|
||||
|
||||
tic()
|
||||
print("test")
|
||||
toc(1)
|
||||
|
||||
@pytest.mark.parametrize("sleepTime", [0.05, 0.5, 1])
|
||||
class testAccuracy:
|
||||
@ -45,42 +43,47 @@ class testAccuracy:
|
||||
def tol(self):
|
||||
return 0.0006
|
||||
|
||||
def testSingleCall(self, t, sleepTime, tol):
|
||||
t.tic()
|
||||
def testSingleCall(self, sleepTime, tol):
|
||||
tic()
|
||||
time.sleep(sleepTime)
|
||||
t.toc()
|
||||
assert (t.results.seconds > sleepTime) & (
|
||||
t.results.seconds < (t.results.seconds + tol)
|
||||
)
|
||||
results = toc()
|
||||
assert (results.seconds < sleepTime+tol)
|
||||
|
||||
def testMultipleCalls(self, t, sleepTime, tol):
|
||||
t.tic()
|
||||
def testMultipleGlobalCalls(self, sleepTime, tol):
|
||||
tic()
|
||||
time.sleep(sleepTime)
|
||||
t.toc()
|
||||
toc()
|
||||
time.sleep(sleepTime)
|
||||
t.toc()
|
||||
assert (t.results.seconds > sleepTime * 2) & (
|
||||
t.results.seconds < (t.results.seconds + tol)
|
||||
)
|
||||
results = toc()
|
||||
assert (results.seconds < (sleepTime * 2)+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:
|
||||
def testMicros(self, t):
|
||||
t.tic()
|
||||
def testMicros(self):
|
||||
tic()
|
||||
print("test")
|
||||
t.toc()
|
||||
assert t.results.micros == (math.floor(t.results.nanos * pow(10, -3)))
|
||||
results = toc()
|
||||
assert results.micros == (math.floor(results.nanos * pow(10, -3)))
|
||||
|
||||
def testMillis(self, t):
|
||||
t.tic()
|
||||
def testMillis(self):
|
||||
tic()
|
||||
print("test")
|
||||
t.toc()
|
||||
assert t.results.millis == (math.floor(t.results.nanos * pow(10, -6)))
|
||||
results = toc()
|
||||
assert results.millis == (math.floor(results.nanos * pow(10, -6)))
|
||||
|
||||
def testSeconds(self, t):
|
||||
t.tic()
|
||||
def testSeconds(self):
|
||||
tic()
|
||||
print("test")
|
||||
t.toc()
|
||||
assert t.results.seconds == round(
|
||||
(t.results.nanos * pow(10, -9)), 9
|
||||
results = toc()
|
||||
assert results.seconds == round(
|
||||
(results.nanos * pow(10, -9)), 9
|
||||
) # f64 vs u128, hence the round
|
||||
|
78
tests/testInitSyntax.py
Normal file
78
tests/testInitSyntax.py
Normal file
@ -0,0 +1,78 @@
|
||||
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
|
7
tests/testTocBeforeTic.py
Normal file
7
tests/testTocBeforeTic.py
Normal file
@ -0,0 +1,7 @@
|
||||
import pytest
|
||||
import tictoc
|
||||
|
||||
def testTocBeforeTic():
|
||||
with pytest.raises(Exception):
|
||||
t = tictoc.init()
|
||||
t.toc()
|
9
tictoc/__init__.py
Normal file
9
tictoc/__init__.py
Normal file
@ -0,0 +1,9 @@
|
||||
from .tictoc import *
|
||||
|
||||
__doc__ = tictoc.__doc__
|
||||
if hasattr(tictoc, "__all__"):
|
||||
__all__ = tictoc.__all__
|
||||
|
||||
t = tictoc.init();
|
||||
tic = t.tic;
|
||||
toc = t.toc;
|
Loading…
x
Reference in New Issue
Block a user