Skip to main content

A fast Lomb-Scargle periodogram. It's nifty, and uses a NUFFT.

Project description

nifty-ls

A fast Lomb-Scargle periodogram. It's nifty, and uses a NUFFT!

PyPI Tests pre-commit.ci status Jenkins Tests

[!WARNING] This project is in a pre-release stage and will likely undergo breaking changes during development. Some of the instructions in the README are aspirational.

Overview

The Lomb-Scargle periodogram, used for identifying periodicity in irregularly-spaced observations, is useful but computationally expensive. However, it can be phrased mathematically as a pair of non-uniform FFTs (NUFFTs). This allows us to leverage Flatiron Institute's finufft package, which is really fast! It also enables GPU (CUDA) support and is several orders of magnitude more accurate than Astropy's Lomb Scargle with default settings for many regions of parameter space.

Background

The Press & Rybicki (1989) method for Lomb-Scargle poses the computation as four weighted trigonometric sums that are solved with a pair of FFTs by "extirpolation" to an equi-spaced grid. Specifically, the sums are of the form:

\begin{align}
S_k &= \sum_{j=1}^M h_j \sin(2 \pi f_k t_j), \\
C_k &= \sum_{j=1}^M h_j \cos(2 \pi f_k t_j),
\end{align}

where the $k$ subscript runs from 0 to $N$, the number of frequency bins, $f_k$ is the cyclic frequency of bin $k$, $t_j$ are the observation times (of which there are $M$), and $h_j$ are the weights.

The key observation for our purposes is that this is exactly what a non-uniform FFT computes! Specifically, a "type-1" (non-uniform to uniform) complex NUFFT in the finufft convention computes:

g_k = \sum_{j=1}^M h_j e^{i k t_j}.

The complex and real parts of this transform are Press & Rybicki's $S_k$ and $C_k$, with some adjustment for cyclic/angular frequencies, domain of $k$, real vs. complex transform, etc. finufft has a particularly fast and accurate spreading kernel ("exponential of semicircle") that it uses instead of Press & Rybicki's extirpolation.

There is some pre- and post-processing of $S_k$ and $C_k$ to compute the periodogram, which can become the bottleneck because finufft is so fast. This package also optimizes and parallelizes those computations.

Installation

From PyPI

For CPU support:

$ pip install nifty-ls

For GPU (CUDA) support:

$ pip install nifty-ls[cuda]

The default is to install with CUDA 12 support; one can use nifty-ls[cuda11] instead for CUDA 11 support (installs cupy-cuda11x).

From source

First, clone the repo and cd to the repo root:

$ git clone https://www.github.com/flatironinstitute/nifty-ls
$ cd nifty-ls

Then, to install with CPU support:

$ pip install .

To install with GPU (CUDA) support:

$ pip install .[cuda]

or .[cuda11] for CUDA 11.

For development (with automatic rebuilds enabled by default in pyproject.toml):

$ pip install nanobind scikit-build-core
$ pip install -e .[test] --no-build-isolation

Developers may also be interested in setting these keys in pyproject.toml:

[tool.scikit-build]
cmake.build-type = "Debug"
cmake.verbose = true
install.strip = false

For best performance

You may wish to compile and install finufft and cufinufft yourself so they will be built with optimizations for your hardware. To do so, first install nifty-ls, then follow the Python installation instructions for finufft and cufinufft, configuring the libraries as desired.

nifty-ls can likewise be built from source following the instructions above for best performance, but most of the heavy computations are offloaded to (cu)finufft, so the performance benefit is minimal.

⚠️ MacOS ARM users (M1/M2/etc): due to an OpenMP library incompatibility, the nifty-ls "C++ helpers" are not parallelized in the Mac ARM builds on PyPI. This is not expected to have a big impact on performance,as the core finufft computation will still be parallelized. Building both finufft and nifty-ls from source is a possible workaround.

Usage

From Astropy

Importing nifty_ls makes nifty-ls available via method="fastnifty" in Astropy's LombScargle module. The name is prefixed with "fast" as it's part of the fast family of methods that assume a regularly-spaced frequency grid.

import nifty_ls
from astropy.timeseries import LombScargle
frequency, power = LombScargle(t, y, method="fastnifty").autopower()

To use the CUDA (cufinufft) backend, pass the appropriate argument via method_kws:

frequency, power = LombScargle(t, y, method="fastnifty", method_kws=dict(backend="cufinufft")).autopower()

In many cases, accelerating your periodogram is as simple as setting the method in your Astropy Lomb Scargle code! More advanced usage, such as computing multiple periodograms in parallel, should go directly through the nifty-ls interface.

From nifty-ls (native interface)

nifty-ls has its own interface that offers more flexibility than the Astropy interface for batched periodograms.

Single periodograms

A single periodogram can be computed through nifty-ls as:

import nifty_ls
# with automatic frequency grid:
nifty_res = nifty_ls.lombscargle(t, y, dy)

# with user-specified frequency grid:
nifty_res = nifty_ls.lombscargle(t, y, dy, fmin=0.1, fmax=10, Nf=10**6)

Batched Periodograms

Batched periodograms (multiple objects with the same observation times) can be computed as:

import nifty_ls
import numpy as np

N_t = 100
N_obj = 10
Nf = 200

rng = np.random.default_rng()
t = rng.random(N_t)
freqs = rng.random(N_obj).reshape(-1,1)
y_batch = np.sin(freqs * t)
dy_batch = rng.random(y.shape)

batched = nifty_ls.lombscargle(t, y_batch, dy_batch, Nf=Nf)
print(batched['power'].shape)  # (10, 200)

Note that this computes multiple periodograms simultaneously on a set of time series with the same observation times. This approach is particularly efficient for short time series, and/or when using the GPU.

Support for batching multiple time series with distinct observation times is not currently implemented, but is planned.

Limitations

The code only supports frequency grids with fixed spacing; however, finufft does support type 3 NUFFTs (non-uniform to non-uniform), which would enable arbitrary frequency grids. It's not clear how useful this is, so it hasn't been implemented, but please open a GitHub issue if this is of interest to you.

Performance

Using 16 cores of an Intel Icelake CPU and a NVIDIA A100 GPU, we obtain the following performance. First, we'll look at results from a single periodogram (i.e. unbatched):

benchmarks

In this case, finufft is 5x faster (11x with threads) than Astropy for large transforms, and 2x faster for (very) small transforms. Small transforms improve futher relative to Astropy with more frequency bins. (Dynamic multi-threaded dispatch of transforms is planned as a future feature which will especially benefit small $N$.)

cufinufft is 200x faster than Astropy for large $N$! The performance plateaus towards small $N$, mostly due to the overhead of sending data to the GPU and fetching the result. (Concurrent job execution on the GPU is another planned feature, which will especially help small $N$.)

The following demonstrates "batch mode", in which 10 periodograms are computed from 10 different time series with the same observation times:

batched benchmarks

Here, the finufft single-threaded advantage is consistently 6x across problem sizes, while the multi-threaded advantage is up to 30x for large transforms.

The 200x advantage of the GPU extends to even smaller $N$ in this case, since we're sending and receiving more data at once.

We see that both multi-threaded finufft and cufinufft particularly benefit from batched transforms, as this exposes more parallelism and amortizes fixed latencies.

We use FFTW_MEASURE for finufft in these benchmarks, which improves performance a few tens of percents.

Multi-threading hurts the performance of small problem sizes; the default behavior of nifty-ls is to use fewer threads in such cases. The "multi-threaded" line uses between 1 and 16 threads.

On the CPU, nifty-ls gets its performance not only through its use of finufft, but also by offloading the pre- and post-processing steps to compiled extensions. The extensions enable us to do much more processing element-wise, rather than array-wise. In other words, they enable "kernel fusion" (to borrow a term from GPU computing), increasing the compute density.

Accuracy

While we compared performance with Astropy's fast method, this isn't quite fair. nifty-ls is much more accurate than Astropy fast! Astropy fast uses Press & Rybicki's extirpolation approximation, trading accuracy for speed, but thanks to finufft, nifty-ls can have both.

In the figure below, we plot the median periodogram error in circles and the 99th percentile error in triangles for astropy, finufft, and cufinufft for a range of $N$ (and default $N_F \approx 12N$).

The astropy result is presented for two cases: a nominal case and a "worst case". Internally, astropy uses an FFT grid whose size is the next power of 2 above the target oversampling rate. Each jump to a new power of 2 typically yields an increase in accuracy. The "worst case", therefore, is the highest frequency that does not yield such a jump.

Errors of $\mathcal{O}(10%)$ or greater are not uncommon with worst-case evaluations. Errors of $\mathcal{O}(1%)$ or greater are common in typical evaluations. nifty-ls is conservatively 6 orders of magnitude more accurate.

The reference result in the above figure comes from the "phase winding" method, which uses trigonometric identities to avoid expensive sin and cos evaluations. One can also use astropy's fast method as a reference with exact evaluation enabled via use_fft=False. One finds the same result, but the phase winding is a few orders of magnitude faster (but still not competitive with finufft).

To summarize, we see that nifty-ls is highly accurate while also giving high performance.

float32 vs float64

While 32-bit floats provide a substantial speedup for finufft and cufinufft, we generally don't recommend their use for Lomb-Scargle. The reason is the challenging condition number of the problem. The condition number is the response in the output to a small perturbation in the input—in other words, the derivative. It can easily be shown that the derivative of a NUFFT with respect to the non-uniform points is proportional to $N$, the transform length (i.e. the number of modes). In other words, errors in the observation times are amplified by $\mathcal{O}(N)$. Since float32 has a relative error of $\mathcal{O}(10^{-7})$, transforms of length $10^5$ already suffer $\mathcal{O}(1%)$ error. Therefore, we focus on float64 in nifty-ls, but float32 is also natively supported by all backends for adventurous users.

The condition number is also a likely contributor to the mild upward trend in error versus $N$ in the above figure, at least for finufft/cufinufft. With a relative error of $\mathcal{O}(10^{-16})$ for float64 and a transform length of $\mathcal{O}(10^{6})$, the minimum error is $\mathcal{O}(10^{-10})$.

Testing

First, install from source (pip install .[test]). Then, from the repo root, run:

$ pytest

The tests are defined in the tests/ directory, and include a mini-benchmark of nifty-ls and Astropy, shown below:

$ pytest
======================================================== test session starts =========================================================
platform linux -- Python 3.10.13, pytest-8.1.1, pluggy-1.4.0
benchmark: 4.0.0 (defaults: timer=time.perf_counter disable_gc=True min_rounds=5 min_time=0.000005 max_time=1.0 calibration_precision=10 warmup=False warmup_iterations=100000)
rootdir: /mnt/home/lgarrison/nifty-ls
configfile: pyproject.toml
plugins: benchmark-4.0.0, asdf-2.15.0, anyio-3.6.2, hypothesis-6.23.1
collected 36 items                                                                                                                   

tests/test_ls.py ......................                                                                                        [ 61%]
tests/test_perf.py ..............                                                                                              [100%]


----------------------------------------- benchmark 'Nf=1000': 5 tests ----------------------------------------
Name (time in ms)                       Min                Mean            StdDev            Rounds  Iterations
---------------------------------------------------------------------------------------------------------------
test_batched[finufft-1000]           6.8418 (1.0)        7.1821 (1.0)      0.1831 (1.32)         43           1
test_batched[cufinufft-1000]         7.7027 (1.13)       8.6634 (1.21)     0.9555 (6.89)         74           1
test_unbatched[finufft-1000]       110.7541 (16.19)    111.0603 (15.46)    0.1387 (1.0)          10           1
test_unbatched[astropy-1000]       441.2313 (64.49)    441.9655 (61.54)    1.0732 (7.74)          5           1
test_unbatched[cufinufft-1000]     488.2630 (71.36)    496.0788 (69.07)    6.1908 (44.63)         5           1
---------------------------------------------------------------------------------------------------------------

--------------------------------- benchmark 'Nf=10000': 3 tests ----------------------------------
Name (time in ms)            Min              Mean            StdDev            Rounds  Iterations
--------------------------------------------------------------------------------------------------
test[finufft-10000]       1.8481 (1.0)      1.8709 (1.0)      0.0347 (1.75)        507           1
test[cufinufft-10000]     5.1269 (2.77)     5.2052 (2.78)     0.3313 (16.72)       117           1
test[astropy-10000]       8.1725 (4.42)     8.2176 (4.39)     0.0198 (1.0)         113           1
--------------------------------------------------------------------------------------------------

----------------------------------- benchmark 'Nf=100000': 3 tests ----------------------------------
Name (time in ms)              Min               Mean            StdDev            Rounds  Iterations
-----------------------------------------------------------------------------------------------------
test[cufinufft-100000]      5.8566 (1.0)       6.0411 (1.0)      0.7407 (10.61)       159           1
test[finufft-100000]        6.9766 (1.19)      7.1816 (1.19)     0.0748 (1.07)        132           1
test[astropy-100000]       47.9246 (8.18)     48.0828 (7.96)     0.0698 (1.0)          19           1
-----------------------------------------------------------------------------------------------------

------------------------------------- benchmark 'Nf=1000000': 3 tests --------------------------------------
Name (time in ms)                  Min                  Mean            StdDev            Rounds  Iterations
------------------------------------------------------------------------------------------------------------
test[cufinufft-1000000]         8.0038 (1.0)          8.5193 (1.0)      1.3245 (1.62)         84           1
test[finufft-1000000]          74.9239 (9.36)        76.5690 (8.99)     0.8196 (1.0)          10           1
test[astropy-1000000]       1,430.4282 (178.72)   1,434.7986 (168.42)   5.5234 (6.74)          5           1
------------------------------------------------------------------------------------------------------------

Legend:
  Outliers: 1 Standard Deviation from Mean; 1.5 IQR (InterQuartile Range) from 1st Quartile and 3rd Quartile.
  OPS: Operations Per Second, computed as 1 / Mean
======================================================== 36 passed in 30.81s =========================================================

The results were obtained using 16 cores of an Intel Icelake CPU and 1 NVIDIA A100 GPU. The ratio of the runtime relative to the fastest are shown in parentheses. You may obtain very different performance on your platform! The slowest Astropy results in particular may depend on the Numpy distribution you have installed and its trig function performance.

Authors

nifty-ls was originally implemented by Lehman Garrison based on work done by Dan Foreman-Mackey in the dfm/nufft-ls repo, with consulting from Alex Barnett.

Acknowledgements

nifty-ls builds directly on top of the excellent finufft package by Alex Barnett and others (see the finufft Acknowledgements).

Many parts of this package are an adaptation of Astropy LombScargle, in particular the Press & Rybicki (1989) method.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

nifty_ls-1.0.0rc5.tar.gz (174.2 kB view details)

Uploaded Source

Built Distributions

nifty_ls-1.0.0rc5-cp312-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (181.6 kB view details)

Uploaded CPython 3.12+ manylinux: glibc 2.17+ x86-64

nifty_ls-1.0.0rc5-cp312-abi3-macosx_11_0_arm64.whl (79.9 kB view details)

Uploaded CPython 3.12+ macOS 11.0+ ARM64

nifty_ls-1.0.0rc5-cp312-abi3-macosx_10_15_x86_64.whl (359.1 kB view details)

Uploaded CPython 3.12+ macOS 10.15+ x86-64

nifty_ls-1.0.0rc5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (182.8 kB view details)

Uploaded CPython 3.11 manylinux: glibc 2.17+ x86-64

nifty_ls-1.0.0rc5-cp311-cp311-macosx_11_0_arm64.whl (80.7 kB view details)

Uploaded CPython 3.11 macOS 11.0+ ARM64

nifty_ls-1.0.0rc5-cp311-cp311-macosx_10_15_x86_64.whl (359.8 kB view details)

Uploaded CPython 3.11 macOS 10.15+ x86-64

nifty_ls-1.0.0rc5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (183.0 kB view details)

Uploaded CPython 3.10 manylinux: glibc 2.17+ x86-64

nifty_ls-1.0.0rc5-cp310-cp310-macosx_11_0_arm64.whl (80.9 kB view details)

Uploaded CPython 3.10 macOS 11.0+ ARM64

nifty_ls-1.0.0rc5-cp310-cp310-macosx_10_15_x86_64.whl (360.1 kB view details)

Uploaded CPython 3.10 macOS 10.15+ x86-64

nifty_ls-1.0.0rc5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (183.2 kB view details)

Uploaded CPython 3.9 manylinux: glibc 2.17+ x86-64

nifty_ls-1.0.0rc5-cp39-cp39-macosx_11_0_arm64.whl (81.0 kB view details)

Uploaded CPython 3.9 macOS 11.0+ ARM64

nifty_ls-1.0.0rc5-cp39-cp39-macosx_10_15_x86_64.whl (360.3 kB view details)

Uploaded CPython 3.9 macOS 10.15+ x86-64

nifty_ls-1.0.0rc5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (183.1 kB view details)

Uploaded CPython 3.8 manylinux: glibc 2.17+ x86-64

nifty_ls-1.0.0rc5-cp38-cp38-macosx_11_0_arm64.whl (80.9 kB view details)

Uploaded CPython 3.8 macOS 11.0+ ARM64

nifty_ls-1.0.0rc5-cp38-cp38-macosx_10_15_x86_64.whl (360.1 kB view details)

Uploaded CPython 3.8 macOS 10.15+ x86-64

File details

Details for the file nifty_ls-1.0.0rc5.tar.gz.

File metadata

  • Download URL: nifty_ls-1.0.0rc5.tar.gz
  • Upload date:
  • Size: 174.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/5.0.0 CPython/3.12.3

File hashes

Hashes for nifty_ls-1.0.0rc5.tar.gz
Algorithm Hash digest
SHA256 57f7921c19afc16f10d2fd8ae83f3a88420420cf88b5493aa0b13c9e0656762c
MD5 25e032b7f183a761ea8efeea732802b7
BLAKE2b-256 2b7f98f268571a8f2e0db9ff1b340c28b68e20d1293314d5970a2da118ed2c41

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp312-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp312-abi3-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 c1fd2a954a9f40df7635ea13e0040e36f93c4393a2d55700a36e8ab3c4f7a2f8
MD5 ec404eee595612425deb739a2fec8e41
BLAKE2b-256 a33f1bdb5fdf65f7a0eace64898e8c1e67647ab585a1a493f3bdbc54f6653e80

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp312-abi3-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp312-abi3-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 f18760fc939b93825a5c972df0e8d5d1a0d23bbda77424e518ade6d5c0b5648e
MD5 b731147a3c1c9870054091d72a59b5c9
BLAKE2b-256 7c27c3da9ad97b7332993c6de32bef2945583d362d6986b3239931ae841d2541

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp312-abi3-macosx_10_15_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp312-abi3-macosx_10_15_x86_64.whl
Algorithm Hash digest
SHA256 f1ca354122999990f8475fdd4aa893a26b52ade4065192ebe1e4d26cf211baf7
MD5 e4c618f33bffbbb06686ecb60045f8d4
BLAKE2b-256 bf141fd4bd8c707f152cdc83467f1b7f7e0a90ae33930dcbb0248caf61e289e2

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 90bfc50ed70c1d148d91fc67d28b8d88632ca39f1b2b75257f74f74a14da57a7
MD5 625875737bc4009cd92d1fd6f8f59bc4
BLAKE2b-256 ba31d9fe75d204fb2b651d3e5b843f88c92857204a7f5aa6e7b2b3efa3a6a99f

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp311-cp311-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp311-cp311-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 483e681ca85b33df8677d4c3e0cc9ac2eb71174950d782a0674086221e5ea726
MD5 8ad3ca4d2e4f1537c80ccd3b2046bc6b
BLAKE2b-256 ce550ecbfff04ea42d597e6d5978012dc5607d5cc6c074431abd8cedb5e981e6

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp311-cp311-macosx_10_15_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp311-cp311-macosx_10_15_x86_64.whl
Algorithm Hash digest
SHA256 7db9a618cb43b91d46e7957fadf5cf25019383d5ae626ed5ba025345fa1fc174
MD5 f9b94569fe1d178dee3303b17bcfc901
BLAKE2b-256 c97ef989e879683e769e565638364340d6d9760086cb390c820aac861c3ae051

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 b1c411dfe0c1c2cb4e928798b81bb5b5039ac6fe0eea7f4b2b42a09f03d579d8
MD5 1c6d96fa0d8ac1a6711ae6c9ff0b08fc
BLAKE2b-256 d291aaa1e8d2127a91d7d91c378e4ce669c9ed020239703b75bcc36eb42f9537

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp310-cp310-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp310-cp310-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 9dc53f69e989ab2376e9bc0a1b7cfce06ea920efa5ea20eb34f91a887f2e1acf
MD5 14d0c2e5b84bd3dea243adff69cd93a8
BLAKE2b-256 d094d59a2344dee23501327f047860ead1416cc041e2007a14cbb739f3a3c8e1

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp310-cp310-macosx_10_15_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp310-cp310-macosx_10_15_x86_64.whl
Algorithm Hash digest
SHA256 03531b4ac8939645595645bbb53302dc4ee0f76da5d14bb830376cb9fe6a3717
MD5 b90889b5ae89565ee45cab650b045c3a
BLAKE2b-256 3ed85ef67fabdc2a061c310f2d1366e10d90da0889ad778f71ed27100bb0087d

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 e875cad5cf6e74d0cd52971aefd0ff719a3773a2d47253fe13eb34e304b245f5
MD5 a7b17a538da0b89f0879df12e8d4be67
BLAKE2b-256 86b4fcfa2e638ae1a0be1842d3b6bc4945ee13dafed990f8da278878b792a790

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp39-cp39-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp39-cp39-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 c484aee8e672ad745d6110e22c23dcc7a5a4ef15539ee0ec539db126dd958de2
MD5 5eaf3e5220447f43e92bb808f3a2dc09
BLAKE2b-256 7bebc1a4c3809e18e95c4fd6198bb9b731429991d1f2f6020fd31f2133d5e541

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp39-cp39-macosx_10_15_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp39-cp39-macosx_10_15_x86_64.whl
Algorithm Hash digest
SHA256 1e7915be51a4de3e454ea79a3b12b0c648c3d208fd656b231e71698a25ef63ce
MD5 33528ebf6c510c235487acb71217aafb
BLAKE2b-256 a886abf14bccc29db2dcdfeb52dc17e721834801ba6522a46681676214ab81bb

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 f8393e616f679a74fc6f4e0d2b691dc865828214954475d1acc10e22a01cdb81
MD5 ba7ae8c1a3dd59443a1098077cbfb8f2
BLAKE2b-256 e3a9738eb281f56532ff5f5ade4381aa00a64b0b3f7109993a42001d4ef8a75a

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp38-cp38-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp38-cp38-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 168f8655cdd3e9deb2af83a4ba0d92a91603bb841fa825ca9bdfac4a46e2393f
MD5 11455e625088c7d1fd51dee2cd7b4d96
BLAKE2b-256 9162aea48aee5f6c5eebff7cfdf0abbb2f230e314781d4fea5fa1c8ee02b51b1

See more details on using hashes here.

File details

Details for the file nifty_ls-1.0.0rc5-cp38-cp38-macosx_10_15_x86_64.whl.

File metadata

File hashes

Hashes for nifty_ls-1.0.0rc5-cp38-cp38-macosx_10_15_x86_64.whl
Algorithm Hash digest
SHA256 22c79b45cd5ae3f860de9ba477fe71299bef6ce69f20eddfd3e2b6ca4ca10f93
MD5 f8c9a595e662de4e1dbc9715d22b4d92
BLAKE2b-256 869a1b309d07cb389d6a1fb1c0f102394d37a6d59b548af805899e1b616be46a

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page