Skip to main content

Python bindings for the annembed Rust crate

Project description

A data embedding tool and related data analysis or clustering

The crate provides mainly in the form of a library (See documentation of the binary embed for a small executable embedding data in csv files):

  1. Some variations on data embedding tools from t-Sne (2008) to Umap(2018).

    Our implementation is a mix of the various embedding algorithms mentioned in References.

  • The graph is initialized by the Hnsw nearest neighbour algorithm as implemented in: hnsw_rs.
    This provides for free, sub-sampling in the data to embed by considering only less densely occupied layers (the upper layers). This corresponds generally to a subsampling of 2%-4%, but can give a guarantee as the distance beetween points leaved out the sampling and its nearest sampled neighbour are known. The hnsw structure thus enables also an iterative/hierarchical initialization of the embedding by taking into account an increasing number of layers.

  • The preliminary graph built for the embedding uses an exponential function of distances to neighbour nodes (as in Umap),but keeps a probability normalization constraint with respect to neighbours (as in T-sne). It is possible to modulate the initial edge weight by :
    - Considering a power of the distance function to neighbours (See documentation in module EmbedderParams).
    - Increase or decrease the impact of the local density of points around each node. There is no symetrization of the graph. (except when initializing the embedding with diffusion maps in this case it is done as in t-sne or LargeVis). We use the diffusion maps algorithm (Lafon-Keller-Coifman).

  • We also use a cross entropy optimization of this initial layout but take into account the initial local density estimate of each point when computing the cauchy weight of an embedded edge. The corresponding "perplexity" distribution is estimated on the fly. (See documentation in module EmbedderParams).

  1. A tentative estimation of embedding faithfulness.

    We quantify the stability of neighbourhoods through the embedding as this helps selecting among varying results between runs for a given data set. The default dimension of 2 is used for visualization purposes, but for other purposes, increasing the dimension shows a better conservation of neighborhoods. It is known that the dimension of the target space is critical in the embedding process. (See Whitney embedding theorem).

    This can be verified in examples (especially the Higgs data) and detailed in the documentation of function Embedder::get_quality_estimate_from_edge_length and is illustrated in examples directory.

  2. Some by-products :

    • an implementation of range approximation and approximated SVD for dense and/or row compressed matrices as described in the svdapprox module and the paper of Halko-Tropp (Cf. Tsvd).

    • An estimation of the data intrinsic dimension as described in:
      Levina E. and Bickel P.J NIPS 2004. See paper.

    • An estimation of the hubness of the graph of the Hnsw structure as described in: Radovanovic M., Nanopoulos A. and Ivanovic M. See paper

    • a Diffusion Maps implementation.

    • A link to the Topological Data Analysis Julia package Ripserer.jl (See the directory Julia in the crate).
      The distance matrix between points in a neighbourhood or from a reduced projected graph can be dumped to further processsing (see docs in module fromhnsw::toripserer). It is thus possible to produce persistence diagrams/barcodes of cloud points with the aid of the julia functions provided in the Julia directory of this crate (providing also visualization of the embedded data from the related csv files results).

Building

Blas choice

The crate provides 3 features to choose between openblas-static, intel-mkl-static or openblas-system as defined in the ndarray-linalg crate.

compile with :

  • cargo build --release --features="openblas-static" to link statically with rust downloaded openblas

  • cargo build --release --features="intel-mkl-static" to link with mkl intel's library (intel mkl will be automatically downloaded, see README.md of crate ndarray-linalg)

  • cargo build --release --features="openblas-system" to link with system installed openblas library. (In this case you must have an openblas library compiled with INTERFACE64=0, corresponding to 32bit fortran integers).

You can also add the feature you want in default features.

simd

On Intel cpu the you can add the simdeez_f feature to default features, or use the command cargo build --release --features="openblas-system,simdeez_f". On non intel cpu it is possible to use the stdsimd feature or "cargo build --release --features="openblas-system,stdsimd". Note that stdsimd requires the nightly compiler.

Julia

Julia scripts provide graphic functions.
Julia can be downloaded from julia. Packages mentionned by a using clause in julia sources must then be installed see Pkg. Then in a Julia REPL, include("annembed.jl") give access to functions Annembed.localPersistency and Annembed.projectedPersistency. Possibly you will need to run export LD_PRELOAD=/lib/x86_64-linux-gnu/libstdc++.so.6 or equivalent to force Julia to use your C++ library, due to subtle interaction between GPU drivers and GLFW.jl see.

Results

Timings are given for a 24-core (32 threads) i9 laptop with 64Gb memory.

Embedder examples

Sources of examples are in corresponding directory.

  1. MNIST digits database Cf mnist-digits

    It consists in 70000 images of handwritten digits of 784 pixels

    • initialized by an approximated svd.

    It tooks 11s (system time) to run (cpu time 340s), of which 3s system time (100s cpu) were spent in the ann construction.

    mnist

    • hierarchical initialization

    mnist

    It took 11s to run (334s of cpu time) of which 3s were spent in the ann construction.

    • The estimated intrinsic dimension of the data is 18.5 with standard deviation depending on points: 7.2 taking into account sorted neighbours around each point between the 9-th and 20-th first ranks.
  2. MNIST fashion database Cf mnist-fashion

    It consists in 70000 images of clothes.

    • initialized by an approximated svd.

    mnist

    system time : 14s, cpu time 428s

    • hierarchical initialization (This is useful for large data embedding where we initialize the embedding with first layers above the deeper populated ones of the Hnsw structure to speed up the process).

    mnist

    system time : 15s, cpu time 466s

    • The estimated intrinsic dimension of the data is 21.9 with standard deviation depending on points is 12.2, taking into account sorted neighbours around each point between the 9-th and 20-th first ranks.
  3. Higgs boson Cf Higgs-data

    It consists in 11 millions float vectors of dimension 28. First we run on the first 21 columns, keeping out the last 7 variables constructed by the physicists to help the discrimination in machine learning tasks and then on the 28 variables.

    In both cases we use hierarchical initialization. We run 200 batches in the first pass by using layers from layer 1 (included) to the upper layer. The first batches runs thus on about 460000 nodes. Then 40 batches are done on the 11 millions points.

    Run times are in both cases around 2 hours (45' for the Hnsw construction and 75' for the entropy iterations)

    • Images for the 21 and 28 variables full data

      Quality estimation requires data subsampling due to the size of data (see examples and results). Moreover a basic exploration of the data can be found in a Notebook at Higgs.jl to assess the quality of the embedding via random projections.

      21 variables image:

    higgs-21

    28 variables image:

    higgs-28

    • Quality estimation with a subsampling factor of 0.15

      Running with subsampling factor 0.15 on the data with 28 variables (1.65 million data points) we get in 16':

      nb neighbourhoods without a match : 869192, mean number of neighbours conserved when match : 5.449e0

      So about half of the neighbourhoods are partially respected in the embedding.

      quantiles on ratio : distance in embedded space of neighbours of origin space / distance of neighbours in embedded space

      half of the original neighborhoods are embedded within a factor 1.30 of the radius of neighbourhood in embedded graph

      quantiles at 0.25 : 7.87e-2, 0.5 : 1.30e0, 0.75 : 4.75e0, 0.85 : 7.94e0, 0.95 : 1.62e1

      the mean ratio is : 4.36

    28 variables, subsampling 0.15:

    higgs-28-subs0.15

    density of points obtained by transforming distance to first neighbour (See visu.jl):

    higgs-28-density

    • Diffusion Maps initialization in the non hierarchical case.

      In the direct case the initial diffusion maps with approximated svd runs in 1650 sec and produces the following initialization image:

    higgs_dmap

Genomics applications can be found in the following paper:

Approximate nearest neighbor graph provides fast and efficient embedding with applications for large-scale biological data.
Jianshu Zhao, Jean Pierre Both, Konstantinos T Konstantinidis.
NAR Genomics and Bioinformatics, Volume 6, Issue 4, December 2024.

Usage

    // allocation of a Hnsw structure to store data
    let ef_c = 50;
    let max_nb_connection = 70;
    let nbimages = images_as_v.len();
    let nb_layer = 16.min((nbimages as f32).ln().trunc() as usize);
    let hnsw = Hnsw::<f32, DistL2>::new(max_nb_connection, nbimages, nb_layer, ef_c, DistL2{});
    let data_with_id : Vec<(&Vec<f32>, usize)>= images_as_v.iter().zip(0..images_as_v.len()).collect();
    // data insertion in the hnsw structure
    hnsw.parallel_insert(&data_with_id);
    // choice of embedding parameters 
    let mut embed_params = EmbedderParams::new();
    embed_params.nb_grad_batch = 15;
    embed_params.scale_rho = 1.;
    embed_params.beta = 1.;
    embed_params.grad_step = 1.;
    embed_params.nb_sampling_by_edge = 10;
    embed_params.dmap_init = true;
    // conversion of the hnsw to a graph structure
    let knbn = 8;
    let kgraph = kgraph_from_hnsw_all(&hnsw, knbn).unwrap();
    // allocation of the embedder and embedding
    embedder = Embedder::new(&kgraph, embed_params);
    let embed_res = embedder.embed();

Randomized SVD

The randomized SVD is based on the paper of Halko-Tropp. The implementation covers dense matrices or matrices in compressed row storage as provided in the sprs crate.

Two algorithms for range approximation used in approximated SVD are:

  • subspace_iteration_csr , corresponds to algo 4.4 in Tropp paper. It uses QR stabilization.

  • adaptative_range_finder_matrep correponds to algo 4.2 in Tropp paper. The algorithm is less precise than subspace_iteration_csr but can work on larger matrices for example on sparse matrices with a million rows.

References

  • Visualizing data using t_sne. Van der Maaten and Hinton 2008.

  • Visualizing Large Scale High Dimensional Data Tang Liu WWW2016 2016 LargeVis

  • Phate Visualizing Structure and Transitions for Biological Data Exploration K.R Moon 2017.

  • Umap: Uniform Manifold Approximation and Projection for Dimension Reduction. L.MacInnes, J.Healy and J.Melville 2018

  • Stochastic Cluster Embedding. Zhirong Yang, Yuwei Chen, Denis Sedov, Samuel Kaski, Jukka Corander.
    Statistics and Computing 2023. SCE

Contributions

The logo was designed by Jianshu Zhao.

License

Licensed under either of

  1. Apache License, Version 2.0, LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0

  2. MIT license LICENSE-MIT or http://opensource.org/licenses/MIT

at your option.

Project details


Download files

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

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

annembed_rs-0.1.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (3.4 MB view details)

Uploaded CPython 3.13manylinux: glibc 2.17+ x86-64

annembed_rs-0.1.0-cp313-cp313-macosx_11_0_arm64.whl (1.6 MB view details)

Uploaded CPython 3.13macOS 11.0+ ARM64

annembed_rs-0.1.0-cp313-cp313-macosx_10_12_x86_64.whl (1.7 MB view details)

Uploaded CPython 3.13macOS 10.12+ x86-64

annembed_rs-0.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (3.4 MB view details)

Uploaded CPython 3.12manylinux: glibc 2.17+ x86-64

annembed_rs-0.1.0-cp312-cp312-macosx_11_0_arm64.whl (1.6 MB view details)

Uploaded CPython 3.12macOS 11.0+ ARM64

annembed_rs-0.1.0-cp312-cp312-macosx_10_12_x86_64.whl (1.7 MB view details)

Uploaded CPython 3.12macOS 10.12+ x86-64

annembed_rs-0.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (3.4 MB view details)

Uploaded CPython 3.11manylinux: glibc 2.17+ x86-64

annembed_rs-0.1.0-cp311-cp311-macosx_11_0_arm64.whl (1.6 MB view details)

Uploaded CPython 3.11macOS 11.0+ ARM64

annembed_rs-0.1.0-cp311-cp311-macosx_10_12_x86_64.whl (1.7 MB view details)

Uploaded CPython 3.11macOS 10.12+ x86-64

annembed_rs-0.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (3.4 MB view details)

Uploaded CPython 3.10manylinux: glibc 2.17+ x86-64

annembed_rs-0.1.0-cp310-cp310-macosx_11_0_arm64.whl (1.6 MB view details)

Uploaded CPython 3.10macOS 11.0+ ARM64

annembed_rs-0.1.0-cp310-cp310-macosx_10_12_x86_64.whl (1.7 MB view details)

Uploaded CPython 3.10macOS 10.12+ x86-64

annembed_rs-0.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (3.4 MB view details)

Uploaded CPython 3.9manylinux: glibc 2.17+ x86-64

annembed_rs-0.1.0-cp39-cp39-macosx_11_0_arm64.whl (1.6 MB view details)

Uploaded CPython 3.9macOS 11.0+ ARM64

annembed_rs-0.1.0-cp39-cp39-macosx_10_12_x86_64.whl (1.7 MB view details)

Uploaded CPython 3.9macOS 10.12+ x86-64

annembed_rs-0.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl (3.4 MB view details)

Uploaded CPython 3.8manylinux: glibc 2.17+ x86-64

annembed_rs-0.1.0-cp38-cp38-macosx_11_0_arm64.whl (1.6 MB view details)

Uploaded CPython 3.8macOS 11.0+ ARM64

annembed_rs-0.1.0-cp38-cp38-macosx_10_12_x86_64.whl (1.7 MB view details)

Uploaded CPython 3.8macOS 10.12+ x86-64

File details

Details for the file annembed_rs-0.1.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp313-cp313-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 fff2258bb6f5fcf9e6be53352dfe8df39af853a84cc8cd01c223123792116446
MD5 8587e1dd3217804a65f4ee6825ffb8a2
BLAKE2b-256 3932ee23055cb000f89ba9af9466b3f7bd2276a725094a4d8accb3c310ebf0b7

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp313-cp313-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp313-cp313-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 1f4684f0555e234dbe41102bb7d59ebff5f3c4f838d1fce115b5ea8dc79f030e
MD5 ade42a9e8b754c60206f5fe4d498cbf0
BLAKE2b-256 dbc4c1725551f50347acd59efe517bd49d076d577c0509674b249876ef09dde9

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp313-cp313-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp313-cp313-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 849b86b7e8746e1f8830ce7d778019af00398f9090f78ce9cfb086ae037da87a
MD5 17b838d74c54de91311dd226d227cc56
BLAKE2b-256 e162f42dfe854610baff56b75b7ea3e13cc911566d83c681014eba10888a9c40

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 5514a49064ffdada64a8ef240a40af53e9ced7bed88af51340d2f6a2e5c0d1ec
MD5 dc293a1bc410d48d446a2d95bfde18f5
BLAKE2b-256 794ef25177c07e7fdacfc7287296b2d375e396ac4178abf5cd07c0e362fcfb77

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp312-cp312-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp312-cp312-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 b2f91663f05cb1a11e267af9006c4e300e306e4bbbba47bad0cbb62b40364f6c
MD5 679793ce763dbd8e7f57053f35191370
BLAKE2b-256 9252f869ea8cd818e42f00d924dfd19be464cb864e5721fe5b74f0b011806ac3

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp312-cp312-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp312-cp312-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 a3f1395762408e2017f9fba7085aa9a00af6bded8610925d320d80d9949acc0e
MD5 b2c8bb482099d236d9ec4ca97a6dbca5
BLAKE2b-256 6bde528036ce9820fe915ce986a1effe7afe7d35fdb3e3a96294f377e0a07dab

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 db9493f98f7c1930553dad82460a644935ca96ce7a8bb8eea3aff025b79c2308
MD5 ab786d11d44b63834a487e234b8aea38
BLAKE2b-256 97445e464873e09907a0af533fa9c4c030ed41552011050af97aeec1b0eab17f

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp311-cp311-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp311-cp311-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 f652be9a5868c0bcf63b2c346b4614b95753374ae486b8d82cacc20ae8bb0e28
MD5 8479184aefab46411618a95a34450add
BLAKE2b-256 03b794376b827c415f540113dc37b5c158af850269fde49c24f2af88950e059d

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp311-cp311-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp311-cp311-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 a29d75cf61a717d5ba34462e615db03b47db6362d79b461c1ebe3ba8b683a1b5
MD5 7e879f76604a7b7bcf805e96544c0a73
BLAKE2b-256 f5967b0496b35f4b33b29073043c0f482c29efa22349dbc47e25258c5ef4dd08

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 b01165d4be38df4ca05cb921c357efe4ecdf94c75ea582ec9ed7e98cae14c316
MD5 8ababc47dc922aed743d2016a9fa3da9
BLAKE2b-256 5a331568db01b93cd661c3aa0f7673855b9dccbb1e40a08acb3b63e9170b6dbe

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp310-cp310-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp310-cp310-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 33f778d103dda411deb0f376e0d9ad2dce498938a7a63b086fdd620055378c9b
MD5 5b8cd58f6a6650ae87274a58d6864134
BLAKE2b-256 1eb74e2345c8fe8a84a6ff4a5b31e6eb9485f035214012cff4e46b07f1fa51b0

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp310-cp310-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp310-cp310-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 33cafcbd534a5aecc715c3de7be929caeadc53ef8787ab81fdad2d582b338d0a
MD5 6b94f0d360f48c04b2a6f0d59bb8addb
BLAKE2b-256 b694f7c8f5992be96e57cf2bed1ab5ac83713568cfc085b4bc9c8a6e54c470ca

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 8315c69d15072b0a303f0cf442236c02be266de590ff6fab0cc606e4c6f4e710
MD5 1125adab48b00e52fdf3a7379c9ce534
BLAKE2b-256 18aaa24f6811d20cc385ef65a85202223676f313953fea910b00dad79770e50d

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp39-cp39-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp39-cp39-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 eb12bc11be8c6d2a6b4058a966482d6340d3f913bcbf2b57a1f663c047245080
MD5 e5ffbf5f0b863b91db4bab306bdbfb58
BLAKE2b-256 4252fa9277a8c7311ce236a8fdbdc3a35bf501d12e41c8cbbb045a4cabccd361

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp39-cp39-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp39-cp39-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 caf3ac4be2b0072d7b17ecbb0cebbe9711b94cd642a591d5d6fca40c15f8f483
MD5 a87a83a1151dc5f3326d8a7f9df58e24
BLAKE2b-256 e16f637a0b9145ede8989c17e7022034894f0d79f6b18e496170dc1f4dd6ac30

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl
Algorithm Hash digest
SHA256 c9589a755a92cfcb4c09b10296c51658da6c631e5d05c577988a48d09865ff43
MD5 a54ba2c699ee7cb5a0ff6b1de4c05937
BLAKE2b-256 ba307e1171b819d9606c8a7ca16f7c627681ba18836e70786825d0eb58aded88

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp38-cp38-macosx_11_0_arm64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp38-cp38-macosx_11_0_arm64.whl
Algorithm Hash digest
SHA256 71d025557a151761c9c58f39efba176476c94724fc2c4e9916e1acff77583597
MD5 9f854ba4d35b783d7607c7eb09ec5b0f
BLAKE2b-256 5e2cb028793da00138b1e65f423ecc855fa429c19c1f13ad383f5db42af67ca2

See more details on using hashes here.

File details

Details for the file annembed_rs-0.1.0-cp38-cp38-macosx_10_12_x86_64.whl.

File metadata

File hashes

Hashes for annembed_rs-0.1.0-cp38-cp38-macosx_10_12_x86_64.whl
Algorithm Hash digest
SHA256 942b86310ba6587d65ad458781ba5bc7015e51fc0340869dcf63de572d23980c
MD5 468403930d105840e87d58d5348dac36
BLAKE2b-256 0374a4f90f01bb43adf3f11a52a8a35953704d53a5b9f2f460765f2c41f65444

See more details on using hashes here.

Supported by

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