# tensorizer
Module, Model, and Tensor Serialization/Deserialization
## TLDR
Extremely fast model loads from HTTP/HTTPS, Redis, and S3 endpoints.
GPT-J (`20GB`) loads at wire-speed (`~5GB/s`) on a 40GbE network, and
is only bottlenecked by the Linux kernel TCP stack.
## Rationale
CoreWeave and our customers use KNative to deploy models as serverless
functions. How long a model takes to load is a major factor in the latency
of KNative scale-up. `tensorizer` is a tool to serialize models and their
associated tensors into a single file that can be loaded quickly and
efficiently off an HTTP/HTTPS or S3 endpoint.
By not embedding the model in the container image, we can reduce the
container image size and the time it takes to load the model. This is
especially important for models that are large in size, such as
[EleutherAI/gpt-neox-20B](https://huggingface.co/EleutherAI/gpt-neox-20B)
that weighs in at `~40GB`.
This decoupling of the model from the container image also allows us to
update the model without having to rebuild the container image. This allows
us to quickly iterate on the model and deploy new versions without having
to wait for the container image to build or for the container image cache
to be populated.
`tensorizer` has S3 support, so we can store the serialized model in S3
object storage, and perform streaming loads from S3. This allows us to
stream the model directly from S3 into the container without having to
download the model to the container's local filesystem. This also
pertains to HTTP/HTTPS endpoints, as S3 is just an HTTP/HTTPS endpoint.
`tensorizer` also has support for loading models from a local filesystem,
so you can use it to serialize models locally and load them locally. This
is extremely fast, as the same principles that make it fast for HTTP/HTTPS
and S3 endpoints also apply to local filesystems.
`tensorizer` has preliminary support for Redis, but it is not recommended
for model deployment due to the lack of distributed caching. It is intended
for sharing state between inference pods, or for loading data on a per-request
basis from a Redis cache.
## Speed
`tensorizer`'s deserialization speed is primarily network-bound.
The following graph presents data collected from the scripts and Kubernetes
manifests in [examples/benchmark_buffer_size](examples/benchmark_buffer_size)
comparing the various deserialization modes available in `tensorizer` release
2.5.0—along with the raw network speed, and the speed of `torch.load()`.
![A letter-value plot comparing 7 deserialization modes and their respective deserialization speeds with a granularity of 0.125 GiB/sec. For local files, "torch.load()" has a median speed between 1.875 and 2.000 GiB/sec; "tensorizer file" has a median of 2.250; "tensorizer file, plaid_mode" has a median of about 4.625; "tensorizer file, lazy_load" has a median between 1.750 and 1.875. The raw network speed is also listed on the chart with a median between 1.250 and 1.375. For HTTP streaming, "tensorizer http" has a median between 0.875 and 1.000; "tensorizer http, plaid_mode" has a median between 1.000 and 1.125; and "tensorizer http, lazy_load" has a median between 0.875 and 1.000.](https://github.com/coreweave/tensorizer/assets/24918963/28786a79-0bfe-4f09-b7c9-f45766f6259c)
## Installation
### From PyPI
`tensorizer` can be installed from PyPI with `pip`:
```bash
python -m pip install tensorizer
```
### From Source
You can also install `tensorizer` from source using `pip`.
To clone the repository and install `tensorizer` in
[editable mode](https://pip.pypa.io/en/stable/topics/local-project-installs/#editable-installs),
run:
```bash
git clone https://github.com/coreweave/tensorizer
cd tensorizer
python -m pip install -e .
```
Or, run the following for `pip` to install `tensorizer`
[directly from GitHub](https://pip.pypa.io/en/stable/topics/vcs-support/#git):
```bash
python -m pip install git+https://github.com/coreweave/tensorizer
```
## Basic Usage
Serialization is done with the `TensorSerializer` class. It takes a
`path_uri` argument that can be a local filesystem path, an HTTP/HTTPS
endpoint, or an S3 endpoint.
`write_module` is the main method of the `TensorSerializer` class. It
takes a `torch.nn.Module` and serializes the tensors to the `path_uri`
endpoint.
The below example serializes the `EleutherAI/gpt-j-6B` model to an S3
endpoint. It assumes that you have already configured your S3
credentials in `~/.s3cfg`.
**NOTE:** Loading and serializing `gpt-j-6B` will take a lot of CPU RAM,
up to `~20GB`. Additionally, when loading `gpt-j-6B` into a GPU, you
will need about `~16GB` of VRAM. If you don't have that much RAM or VRAM,
you can use the smaller `gpt-neo-125M` model instead.
**NOTE2:** The below examples require the `transformers` and `accelerate`
libraries. You can install them with `pip`:
```bash
python -m pip install transformers accelerate
```
[serialize.py](examples/serialize.py)
```python
import torch
from tensorizer import TensorSerializer
from transformers import AutoModelForCausalLM
model_ref = "EleutherAI/gpt-j-6B"
# For less intensive requirements, swap above with the line below:
# model_ref = "EleutherAI/gpt-neo-125M"
model_name = model_ref.split("/")[-1]
# Change this to your S3 bucket.
s3_bucket = "bucket"
s3_uri = f"s3://{s3_bucket}/{model_name}.tensors"
model = AutoModelForCausalLM.from_pretrained(
model_ref,
revision="float16",
torch_dtype=torch.float16,
low_cpu_mem_usage=True,
)
serializer = TensorSerializer(s3_uri)
serializer.write_module(model)
serializer.close()
```
Conversely, deserialization is done with the `TensorDeserializer` class.
It takes a `path_uri` argument that can be a local filesystem path, an
HTTP/HTTPS endpoint, or an S3 endpoint.
`load_into_module` is the main method of the `TensorDeserializer` class.
It takes a `torch.nn.Module` and loads the tensors from the `path_uri`
endpoint into the `torch.nn.Module`.
The below example loads the `EleutherAI/gpt-j-6B` model from an S3
endpoint.
[deserialize-simple.py](examples/deserialize-simple.py)
```python
import time
import torch
from tensorizer import TensorDeserializer
from tensorizer.utils import no_init_or_tensor, convert_bytes, get_mem_usage
from transformers import AutoModelForCausalLM, AutoTokenizer, AutoConfig
model_ref = "EleutherAI/gpt-j-6B"
# To run this at home, swap this with the line below for a smaller example:
# model_ref = "EleutherAI/gpt-neo-125M"
model_name = model_ref.split("/")[-1]
# Change this to your S3 bucket.
s3_bucket = "bucket"
s3_uri = f"s3://{s3_bucket}/{model_name}.tensors"
config = AutoConfig.from_pretrained(model_ref)
device = torch.device("cuda" if torch.cuda.is_available() else "cpu")
# This ensures that the pretrained model weights are not initialized,
# and non-persistent buffers (generated at runtime) are on the correct device.
with torch.device(device), no_init_or_tensor():
model = AutoModelForCausalLM.from_config(config)
print(f"Deserializing to {device}:")
before_mem = get_mem_usage()
# Lazy load the tensors from S3 into the model.
start = time.perf_counter()
deserializer = TensorDeserializer(s3_uri, device=device)
deserializer.load_into_module(model)
end = time.perf_counter()
after_mem = get_mem_usage()
# Brag about how fast we are.
total_bytes_str = convert_bytes(deserializer.total_tensor_bytes)
duration = end - start
per_second = convert_bytes(deserializer.total_tensor_bytes / duration)
deserializer.close()
print(f"Deserialized {total_bytes_str} in {end - start:0.2f}s, {per_second}/s")
print(f"Memory usage before: {before_mem}")
print(f"Memory usage after: {after_mem}")
# Tokenize and generate
model.eval()
tokenizer = AutoTokenizer.from_pretrained(model_ref)
eos = tokenizer.eos_token_id
input_ids = tokenizer.encode(
"¡Hola! Encantado de conocerte. hoy voy a", return_tensors="pt"
).to(device)
with torch.no_grad():
output = model.generate(
input_ids, max_new_tokens=50, do_sample=True, pad_token_id=eos
)
print(f"Output: {tokenizer.decode(output[0], skip_special_tokens=True)}")
```
It should produce output similar to the following, with GPT-J-6B:
```
Deserialized model in 6.25 seconds
Test Output: ¡Hola! Encantado de conocerte. hoy voy a comentar por primera
vez una teoría de trineo, que quizá te parezca
algo desconocido, ya que en este mundo han
llegado a dominar tantos
```
More practical examples for the usage of `tensorizer` can be found in
[examples/hf_serialization.py](examples/hf_serialization.py),
where `df_main()` serializes models from
[HuggingFace Diffusers](https://github.com/huggingface/diffusers)
and `hf_main()` serializes
[HuggingFace Transformers](https://github.com/huggingface/transformers) models.
## Tensor Weight Encryption
`tensorizer` supports fast tensor weight encryption and decryption during
serialization and deserialization, respectively.
Be aware that metadata (tensor names, dtypes, shapes, etc.) are not encrypted,
only the weights themselves.
> [!NOTE]
>
> Refer to [docs/encryption.md](/docs/encryption.md) for details, instructions,
> and warnings on using `tensorizer` encryption correctly and safely.
To use `tensorizer` encryption, a recent version of `libsodium` must be
installed. Install `libsodium` with `apt-get install libsodium23`
on Ubuntu or Debian, or follow
[the instructions in `libsodium`'s documentation](https://doc.libsodium.org/installation)
for other platforms.
### Quick Encryption Example
The following outline demonstrates how to encrypt and decrypt a tensorized model
with a randomly-generated encryption key:
```py
from tensorizer import (
EncryptionParams, DecryptionParams, TensorDeserializer, TensorSerializer
)
# Serialize and encrypt a model:
encryption_params = EncryptionParams.random()
serializer = TensorSerializer("model.tensors", encryption=encryption_params)
serializer.write_module(...) # or write_state_dict(), etc.
serializer.close()
# Save the randomly-generated encryption key somewhere
with open("tensor.key", "wb") as key_file:
key_file.write(encryption_params.key)
# Then decrypt it again:
# Load the randomly-generated key from where it was saved
with open("tensor.key", "rb") as key_file:
key: bytes = key_file.read()
decryption_params = DecryptionParams.from_key(key)
deserializer = TensorDeserializer("model.tensors", encryption=decryption_params)
deserializer.load_into_module(...)
deserializer.close()
```
For more detail, refer to [docs/encryption.md](/docs/encryption.md).
A complete example is also available as
[examples/encryption.py](examples/encryption.py).
The `EncryptionParams` and `DecryptionParams` class docstrings additionally
contain some usage information for quick reference from an IDE.
An example command line tool to add or remove encryption from existing
serialized models is also available as
[examples/encryption.py](examples/encrypt_existing.py).
## Benchmarks
You can run your own benchmarks on CoreWeave or your own Kubernetes cluster
by using the `benchmark.yaml` file in the `examples/benchmark_buffer_size`
directory. Please see the [README](examples/benchmark_buffer_size/README.md).
## Available Pre-Tensorized Models on the CoreWeave Cloud
The following models are available on the CoreWeave Cloud for free, and can be
used with the `TensorDeserializer` class. The S3 support defaults to the
`accel-object.ord1.coreweave.com` endpoint, and the bucket to use as `tensorized`.
We name the keys in the S3 bucket after the HuggingFace model identifier, and
append the `/fp16` suffix for the half-precision version.
For example, the S3 URI for the `EleutherAI/gpt-j-6B` model is:
`s3://tensorized/EleutherAI/gpt-j-6B/fp16/model.tensors`
The below table shows the available models and their S3 URIs.
### Large Language Models
| Model | Precision | S3 URI |
|-----------------------------------------------------------------------------------------|-----------|---------------------------------------------------------------------|
| [EleutherAI/gpt-neo-125M](https://huggingface.co/EleutherAI/gpt-neo-125M) | `fp32` | `s3://tensorized/EleutherAI/gpt-neo-125M/model.tensors` |
| [EleutherAI/gpt-neo-125M](https://huggingface.co/EleutherAI/gpt-neo-125M) | `fp16` | `s3://tensorized/EleutherAI/gpt-neo-125M/fp16/model.tensors` |
| [EleutherAI/gpt-neo-1.3B](https://huggingface.co/EleutherAI/gpt-neo-1.3B) | `fp32` | `s3://tensorized/EleutherAI/gpt-neo-1.3B/model.tensors` |
| [EleutherAI/gpt-neo-1.3B](https://huggingface.co/EleutherAI/gpt-neo-1.3B) | `fp16` | `s3://tensorized/EleutherAI/gpt-neo-1.3B/fp16/model.tensors` |
| [EleutherAI/gpt-neo-2.7B](https://huggingface.co/EleutherAI/gpt-neo-2.7B) | `fp32` | `s3://tensorized/EleutherAI/gpt-neo-2.7B/model.tensors` |
| [EleutherAI/gpt-neo-2.7B](https://huggingface.co/EleutherAI/gpt-neo-2.7B) | `fp16` | `s3://tensorized/EleutherAI/gpt-neo-2.7B/fp16/model.tensors` |
| [EleutherAI/gpt-j-6B](https://huggingface.co/EleutherAI/gpt-j-6B) | `fp32` | `s3://tensorized/EleutherAI/gpt-j-6B/model.tensors` |
| [EleutherAI/gpt-j-6B](https://huggingface.co/EleutherAI/gpt-j-6B) | `fp16` | `s3://tensorized/EleutherAI/gpt-j-6B/fp16/model.tensors` |
| [EleutherAI/gpt-neox-20b](https://huggingface.co/EleutherAI/gpt-neox-20b) | `fp32` | `s3://tensorized/EleutherAI/gpt-neox-20b/model.tensors` |
| [EleutherAI/gpt-neox-20b](https://huggingface.co/EleutherAI/gpt-neox-20b) | `fp16` | `s3://tensorized/EleutherAI/gpt-neox-20b/fp16/model.tensors` |
| [EleutherAI/pythia-70m](https://huggingface.co/EleutherAI/pythia-70m) | `fp32` | `s3://tensorized/EleutherAI/pythia-70m/model.tensors` |
| [EleutherAI/pythia-70m](https://huggingface.co/EleutherAI/pythia-70m) | `fp16` | `s3://tensorized/EleutherAI/pythia-70m/fp16/model.tensors` |
| [EleutherAI/pythia-1.4b](https://huggingface.co/EleutherAI/pythia-1.4b) | `fp32` | `s3://tensorized/EleutherAI/pythia-1.4b/model.tensors` |
| [EleutherAI/pythia-1.4b](https://huggingface.co/EleutherAI/pythia-1.4b) | `fp16` | `s3://tensorized/EleutherAI/pythia-1.4b/fp16/model.tensors` |
| [EleutherAI/pythia-2.8b](https://huggingface.co/EleutherAI/pythia-2.8b) | `fp32` | `s3://tensorized/EleutherAI/pythia-2.8b/model.tensors` |
| [EleutherAI/pythia-2.8b](https://huggingface.co/EleutherAI/pythia-2.8b) | `fp16` | `s3://tensorized/EleutherAI/pythia-2.8b/fp16/model.tensors` |
| [EleutherAI/pythia-6.9b](https://huggingface.co/EleutherAI/pythia-6.9b) | `fp32` | `s3://tensorized/EleutherAI/pythia-6.9b/model.tensors` |
| [EleutherAI/pythia-6.9b](https://huggingface.co/EleutherAI/pythia-6.9b) | `fp16` | `s3://tensorized/EleutherAI/pythia-6.9b/fp16/model.tensors` |
| [EleutherAI/pythia-12b](https://huggingface.co/EleutherAI/pythia-12b) | `fp32` | `s3://tensorized/EleutherAI/pythia-12b/model.tensors` |
| [EleutherAI/pythia-12b](https://huggingface.co/EleutherAI/pythia-12b) | `fp16` | `s3://tensorized/EleutherAI/pythia-12b/fp16/model.tensors` |
| [EleutherAI/pythia-70m-deduped](https://huggingface.co/EleutherAI/pythia-70m-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-70m-deduped/model.tensors` |
| [EleutherAI/pythia-70m-deduped](https://huggingface.co/EleutherAI/pythia-70m-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-70m-deduped/fp16/model.tensors` |
| [EleutherAI/pythia-1.4b-deduped](https://huggingface.co/EleutherAI/pythia-1.4b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-1.4b-deduped/model.tensors` |
| [EleutherAI/pythia-1.4b-deduped](https://huggingface.co/EleutherAI/pythia-1.4b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-1.4b-deduped/fp16/model.tensors` |
| [EleutherAI/pythia-2.8b-deduped](https://huggingface.co/EleutherAI/pythia-2.8b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-2.8b-deduped/model.tensors` |
| [EleutherAI/pythia-2.8b-deduped](https://huggingface.co/EleutherAI/pythia-2.8b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-2.8b-deduped/fp16/model.tensors` |
| [EleutherAI/pythia-6.9b-deduped](https://huggingface.co/EleutherAI/pythia-6.9b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-6.9b-deduped/model.tensors` |
| [EleutherAI/pythia-6.9b-deduped](https://huggingface.co/EleutherAI/pythia-6.9b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-6.9b-deduped/fp16/model.tensors` |
| [EleutherAI/pythia-12b-deduped](https://huggingface.co/EleutherAI/pythia-12b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-12b-deduped/model.tensors` |
| [EleutherAI/pythia-12b-deduped](https://huggingface.co/EleutherAI/pythia-12b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-12b-deduped/fp16/model.tensors` |
| [KoboldAI/fairseq-dense-125M](https://huggingface.co/KoboldAI/fairseq-dense-125M) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-125M/model.tensors` |
| [KoboldAI/fairseq-dense-125M](https://huggingface.co/KoboldAI/fairseq-dense-125M) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-125M/fp16/model.tensors` |
| [KoboldAI/fairseq-dense-355M](https://huggingface.co/KoboldAI/fairseq-dense-355M) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-355M/model.tensors` |
| [KoboldAI/fairseq-dense-355M](https://huggingface.co/KoboldAI/fairseq-dense-355M) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-355M/fp16/model.tensors` |
| [KoboldAI/fairseq-dense-2.7B](https://huggingface.co/KoboldAI/fairseq-dense-2.7B) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-2.7B/model.tensors` |
| [KoboldAI/fairseq-dense-2.7B](https://huggingface.co/KoboldAI/fairseq-dense-2.7B) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-2.7B/fp16/model.tensors` |
| [KoboldAI/fairseq-dense-6.7B](https://huggingface.co/KoboldAI/fairseq-dense-6.7B) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-6.7B/model.tensors` |
| [KoboldAI/fairseq-dense-6.7B](https://huggingface.co/KoboldAI/fairseq-dense-6.7B) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-6.7B/fp16/model.tensors` |
| [KoboldAI/fairseq-dense-13B](https://huggingface.co/KoboldAI/fairseq-dense-13B) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-13B/model.tensors` |
| [KoboldAI/fairseq-dense-13B](https://huggingface.co/KoboldAI/fairseq-dense-13B) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-13B/fp16/model.tensors` |
| [Salesforce/codegen-350M-mono](https://huggingface.co/Salesforce/codegen-350M-mono) | `fp32` | `s3://tensorized/Salesforce/codegen-350M-mono/model.tensors` |
| [Salesforce/codegen-350M-mono](https://huggingface.co/Salesforce/codegen-350M-mono) | `fp16` | `s3://tensorized/Salesforce/codegen-350M-mono/fp16/model.tensors` |
| [Salesforce/codegen-350M-multi](https://huggingface.co/Salesforce/codegen-350M-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-350M-multi/model.tensors` |
| [Salesforce/codegen-350M-multi](https://huggingface.co/Salesforce/codegen-350M-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-350M-multi/fp16/model.tensors` |
| [Salesforce/codegen-2B-multi](https://huggingface.co/Salesforce/codegen-2B-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-2B-multi/model.tensors` |
| [Salesforce/codegen-2B-multi](https://huggingface.co/Salesforce/codegen-2B-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-2B-multi/fp16/model.tensors` |
| [Salesforce/codegen-6B-mono](https://huggingface.co/Salesforce/codegen-6B-mono) | `fp32` | `s3://tensorized/Salesforce/codegen-6B-mono/model.tensors` |
| [Salesforce/codegen-6B-mono](https://huggingface.co/Salesforce/codegen-6B-mono) | `fp16` | `s3://tensorized/Salesforce/codegen-6B-mono/fp16/model.tensors` |
| [Salesforce/codegen-6B-multi](https://huggingface.co/Salesforce/codegen-6B-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-6B-multi/model.tensors` |
| [Salesforce/codegen-6B-multi](https://huggingface.co/Salesforce/codegen-6B-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-6B-multi/fp16/model.tensors` |
| [Salesforce/codegen-16B-mono](https://huggingface.co/Salesforce/codegen-16B-mono) | `fp32` | `s3://tensorized/Salesforce/codegen-16B-mono/model.tensors` |
| [Salesforce/codegen-16B-mono](https://huggingface.co/Salesforce/codegen-16B-mono) | `fp16` | `s3://tensorized/Salesforce/codegen-16B-mono/fp16/model.tensors` |
| [Salesforce/codegen-16B-multi](https://huggingface.co/Salesforce/codegen-16B-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-16B-multi/model.tensors` |
| [Salesforce/codegen-16B-multi](https://huggingface.co/Salesforce/codegen-16B-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-16B-multi/fp16/model.tensors` |
### Generative Diffusion Models
| Model | Component | Precision | S3 URI |
|-------------------------------------------------------------------------------------------------------------|--------------|-----------|----------------------------------------------------------------------------------------|
| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `VAE` | `fp32` | `s3://tensorized/runwayml/stable-diffusion-v1-5/vae.tensors` |
| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `UNet` | `fp32` | `s3://tensorized/runwayml/stable-diffusion-v1-5/unet.tensors` |
| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `TextEnc` | `fp32` | `s3://tensorized/runwayml/stable-diffusion-v1-5/text_encoder.tensors` |
| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `VAE` | `fp16` | `s3://tensorized/runwayml/stable-diffusion-v1-5/fp16/vae.tensors` |
| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `UNet` | `fp16` | `s3://tensorized/runwayml/stable-diffusion-v1-5/fp16/unet.tensors` |
| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `TextEnc` | `fp16` | `s3://tensorized/runwayml/stable-diffusion-v1-5/fp16/text_encoder.tensors` |
| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `VAE` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-2-1/vae.tensors` |
| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `UNet` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-2-1/unet.tensors` |
| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `TextEnc` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-2-1/text_encoder.tensors` |
| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `VAE` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-2-1/fp16/vae.tensors` |
| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `UNet` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-2-1/fp16/unet.tensors` |
| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `TextEnc` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-2-1/fp16/text_encoder.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `VAE` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/vae.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `UNet` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/unet.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/text_encoder.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc2` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/text_encoder_2.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `VAE` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/vae.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `UNet` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/unet.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/text_encoder.tensors` |
| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc2` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/text_encoder_2.tensors` |
## S3 Usage Notes
`tensorizer` uses the `boto3` library to interact with S3. The easiest way
to use `tensorizer` with S3 is to configure your S3 credentials in
`~/.s3cfg`.
If you don't want to use `~/.s3cfg`, or wish to use a `.s3cfg` config file
saved at a nonstandard location (e.g. under `/var/run`), you can also specify
your S3 credentials using the `tensorizer.stream_io.open_stream()` function,
and then pass that into the `TensorSerializer` or `TensorDeserializer`
constructor.
The `stream_io.open_stream()` function takes a `path_uri` argument, which can
be an `s3://` URI, and accepts the following keyword arguments:
* `s3_access_key_id`: S3 access key ID
* `s3_secret_access_key`: S3 secret access key
* `s3_endpoint`: S3 endpoint
*Or,*
* `s3_config_path`: Alternative filesystem path to a `.s3cfg` config file
For example:
```python
TensorSerializer(
open_stream(s3_uri,
"wb",
s3_access_key_id=ACCESS_KEY,
s3_secret_access_key=SECRET_KEY,
s3_endpoint="object.ord1.coreweave.com"))
```
and...
```python
TensorDeserializer(
open_stream(s3_uri,
"rb",
s3_access_key_id=ACCESS_KEY,
s3_secret_access_key=SECRET_KEY,
s3_endpoint="object.ord1.coreweave.com"))
```
**NOTE:** For faster object downloads in the CoreWeave Cloud, you can use
the `accel-object.ord1.coreweave.com` endpoint. This endpoint is optimized
for object downloads, and will be faster than the `object.ord1.coreweave.com`
endpoint once the object is cached.
**NOTE2:** The cache above does not get invalidated when the object is updated
in S3. If you update an object in S3, you will need to wait for the cache to
expire before you can download the updated object. This takes 24 hours since
the last download.
For this reason, it is recommended to use a unique S3 key for each version
of a model if you use the `accel-object.ord1.coreweave.com` endpoint.
## Additional Features
`tensorizer` has a few additional features that make it more useful than
just a serialization/deserialization tool.
### Concurrent Reads
The `TensorDeserializer` class has a `num_readers` argument that controls
how many threads are allowed to read concurrently from the source file.
This can greatly improve performance, since in many cases the network or the
file is the bottleneck. A few caveats to running with `num_readers > 1`:
* The specified file must be able to be reopened, so that the
`TensorDeserializer` can open more streams against the source.
* Local files, paths, and HTTP(S) and S3 URIs / open streams
are all able to be reopened
* Special files like pipes and sockets, or synthetic file-like objects such as
`BytesIO` are not currently able to be reopened
* For HTTP(S) and S3 streams and URIs, the host must support the `Range` header.
Each reader will read a stream from a different Range offset in the source.
The default is `num_readers=1`, which has no special requirements.
### `state_dict` Support
The `TensorDeserializer` object can be used as-is as a `state_dict` for
`torch.nn.Module.load_state_dict`. This is useful for loading the tensors
into a `torch.nn.Module` that is already initialized, or for inspection.
Keep in mind that `load_state_dict` is not a fast operation, and will
likely be much slower than `load_into_module`.
The `state_dict` can also be used to initialize a HuggingFace Transformers
AutoModel. But HuggingFace Transformers performs three or more copies of
the data, so memory use will explode.
### `bfloat16` Support
Tensorizer supports models using the `bfloat16` data type. However, tensorizer
uses numpy to save the tensors as binary and numpy doesn't support `bfloat16`.
This means that special conversions need to be applied.
To be saved, the torch tensor is cast to `int16` before being converted to
numpy, which doesn't change any of the underlying data. When serialized, the
original `bfloat16` datatype string is also saved so that it will be cast back
to `bfloat16` during the deserialization process.
The `complex32` datatype is supported in a similar way, by casting to `int32`.
The quantized datatypes (`qint8`, `qint32`, etc.) are not currently supported
by tensorizer as they would require supplemental quantization parameters to be
deserialized correctly.
**NOTE:** The exact choice of intermediate types as `int16` and `int32` is
considered an implementation detail, and is subject to change,
so they should not be relied upon.
**NOTE2:** This does not interfere with storing actual `int` datatypes
used in tensors in tensorized files.
### Numpy Support
Tensorizer can be used with `numpy` directly to read and write
`numpy.ndarray`s.
The serializer's `write_tensor` function handles supplying both
`torch.Tensor`s and `numpy.ndarray`s.
The deserializer has a separate function `read_numpy_arrays` that will return
the data as `numpy.ndarray`s.
As explained above in [bfloat16 support](#bfloat16-support), tensorizer uses
special conversions to write "opaque" datatypes, those not supported by numpy.
Therefore, special considerations need to be taken when loading such data as
`numpy.ndarray`s.
By default, the `TensorDeserializer.read_numpy_arrays` function sets its
`allow_raw_data` parameter to `False`. This means that if a file contains
opaque datatypes, a `ValueError` will be raised during deserialization.
If you want to return the raw data regardless, set `allow_raw_data` to `True`.
Otherwise, the file may be read with `TensorDeserializer.read_tensors`
instead, which yields `torch.Tensor` objects of the correct datatype.
A fifth and sixth variable are also returned by the `read_numpy_arrays`
generator. The fifth is a `bool` that indicates whether the returned array
has an opaque datatype and requires special handling (only legal when
`allow_raw_data=True`). The sixth is a string describing the true, non-numpy
datatype that the raw data should be interpreted as in such cases.
For all other datatypes that require no special handling, these are returned as
`False` and `None`, respectively.
The exact numpy datatypes used by the returned opaque `numpy.ndarray` objects
is not guaranteed, and should not be relied upon.
### Plaid mode
Older versions of Tensorizer had an argument called `plaid_mode` that reused
buffers when copying to CUDA devices. This now happens automatically.
`plaid_mode` and `plaid_mode_buffers` are left as arguments for backwards
compatibility but are deprecated and have no effect.
## Running Tests
`tensorizer` uses `unittest` for testing.
The tests have their own set of dependencies, which can be installed with
`pip install -r tests/requirements.txt`.
Some tests require a GPU, and will be skipped if no GPU is available.
To run the tests, run the following in the root of the repository:
```bash
python -m pip install -e .
python -m pip install -r tests/requirements.txt
python -m unittest discover tests/ --verbose
```
## Serialization in a subprocess
You may want to do Serialization in a separate process so that your main process can continue executing and not get bogged down by GIL contention.
See [docs/subprocess-serialization.md](docs/subprocess-serialization.md) for more details.
Raw data
{
"_id": null,
"home_page": null,
"name": "tensorizer",
"maintainer": null,
"docs_url": null,
"requires_python": ">=3.8",
"maintainer_email": null,
"keywords": "tensorizer, machine learning, serialization, tensor, pytorch",
"author": "CoreWeave",
"author_email": null,
"download_url": "https://files.pythonhosted.org/packages/0a/ed/2062ab9492612309c51d78ccdc0cb316a2ecc8c90e08fd4d1754540ee0df/tensorizer-2.9.1.tar.gz",
"platform": null,
"description": "# tensorizer\nModule, Model, and Tensor Serialization/Deserialization\n\n## TLDR\nExtremely fast model loads from HTTP/HTTPS, Redis, and S3 endpoints.\nGPT-J (`20GB`) loads at wire-speed (`~5GB/s`) on a 40GbE network, and\nis only bottlenecked by the Linux kernel TCP stack.\n\n## Rationale\nCoreWeave and our customers use KNative to deploy models as serverless\nfunctions. How long a model takes to load is a major factor in the latency\nof KNative scale-up. `tensorizer` is a tool to serialize models and their\nassociated tensors into a single file that can be loaded quickly and\nefficiently off an HTTP/HTTPS or S3 endpoint.\n\nBy not embedding the model in the container image, we can reduce the\ncontainer image size and the time it takes to load the model. This is\nespecially important for models that are large in size, such as\n[EleutherAI/gpt-neox-20B](https://huggingface.co/EleutherAI/gpt-neox-20B)\nthat weighs in at `~40GB`.\n\nThis decoupling of the model from the container image also allows us to\nupdate the model without having to rebuild the container image. This allows\nus to quickly iterate on the model and deploy new versions without having\nto wait for the container image to build or for the container image cache\nto be populated.\n\n`tensorizer` has S3 support, so we can store the serialized model in S3\nobject storage, and perform streaming loads from S3. This allows us to\nstream the model directly from S3 into the container without having to\ndownload the model to the container's local filesystem. This also\npertains to HTTP/HTTPS endpoints, as S3 is just an HTTP/HTTPS endpoint.\n\n`tensorizer` also has support for loading models from a local filesystem,\nso you can use it to serialize models locally and load them locally. This\nis extremely fast, as the same principles that make it fast for HTTP/HTTPS\nand S3 endpoints also apply to local filesystems.\n\n`tensorizer` has preliminary support for Redis, but it is not recommended\nfor model deployment due to the lack of distributed caching. It is intended\nfor sharing state between inference pods, or for loading data on a per-request\nbasis from a Redis cache.\n\n## Speed\n\n`tensorizer`'s deserialization speed is primarily network-bound.\n\nThe following graph presents data collected from the scripts and Kubernetes\nmanifests in [examples/benchmark_buffer_size](examples/benchmark_buffer_size)\ncomparing the various deserialization modes available in `tensorizer` release\n2.5.0\u2014along with the raw network speed, and the speed of `torch.load()`.\n\n![A letter-value plot comparing 7 deserialization modes and their respective deserialization speeds with a granularity of 0.125 GiB/sec. For local files, \"torch.load()\" has a median speed between 1.875 and 2.000 GiB/sec; \"tensorizer file\" has a median of 2.250; \"tensorizer file, plaid_mode\" has a median of about 4.625; \"tensorizer file, lazy_load\" has a median between 1.750 and 1.875. The raw network speed is also listed on the chart with a median between 1.250 and 1.375. For HTTP streaming, \"tensorizer http\" has a median between 0.875 and 1.000; \"tensorizer http, plaid_mode\" has a median between 1.000 and 1.125; and \"tensorizer http, lazy_load\" has a median between 0.875 and 1.000.](https://github.com/coreweave/tensorizer/assets/24918963/28786a79-0bfe-4f09-b7c9-f45766f6259c)\n\n## Installation\n\n### From PyPI\n`tensorizer` can be installed from PyPI with `pip`:\n```bash\npython -m pip install tensorizer\n```\n\n### From Source\nYou can also install `tensorizer` from source using `pip`.\n\nTo clone the repository and install `tensorizer` in\n[editable mode](https://pip.pypa.io/en/stable/topics/local-project-installs/#editable-installs),\nrun:\n```bash\ngit clone https://github.com/coreweave/tensorizer\ncd tensorizer\npython -m pip install -e .\n```\n\nOr, run the following for `pip` to install `tensorizer`\n[directly from GitHub](https://pip.pypa.io/en/stable/topics/vcs-support/#git):\n```bash\npython -m pip install git+https://github.com/coreweave/tensorizer\n```\n\n## Basic Usage\nSerialization is done with the `TensorSerializer` class. It takes a\n`path_uri` argument that can be a local filesystem path, an HTTP/HTTPS\nendpoint, or an S3 endpoint.\n\n`write_module` is the main method of the `TensorSerializer` class. It\ntakes a `torch.nn.Module` and serializes the tensors to the `path_uri`\nendpoint.\n\nThe below example serializes the `EleutherAI/gpt-j-6B` model to an S3\nendpoint. It assumes that you have already configured your S3\ncredentials in `~/.s3cfg`.\n\n**NOTE:** Loading and serializing `gpt-j-6B` will take a lot of CPU RAM,\nup to `~20GB`. Additionally, when loading `gpt-j-6B` into a GPU, you\nwill need about `~16GB` of VRAM. If you don't have that much RAM or VRAM,\nyou can use the smaller `gpt-neo-125M` model instead.\n\n**NOTE2:** The below examples require the `transformers` and `accelerate`\nlibraries. You can install them with `pip`:\n```bash\npython -m pip install transformers accelerate\n```\n\n[serialize.py](examples/serialize.py)\n```python\nimport torch\nfrom tensorizer import TensorSerializer\nfrom transformers import AutoModelForCausalLM\n\nmodel_ref = \"EleutherAI/gpt-j-6B\"\n# For less intensive requirements, swap above with the line below:\n# model_ref = \"EleutherAI/gpt-neo-125M\"\nmodel_name = model_ref.split(\"/\")[-1]\n# Change this to your S3 bucket.\ns3_bucket = \"bucket\"\ns3_uri = f\"s3://{s3_bucket}/{model_name}.tensors\"\n\nmodel = AutoModelForCausalLM.from_pretrained(\n model_ref,\n revision=\"float16\",\n torch_dtype=torch.float16,\n low_cpu_mem_usage=True,\n)\n\nserializer = TensorSerializer(s3_uri)\nserializer.write_module(model)\nserializer.close()\n```\n\nConversely, deserialization is done with the `TensorDeserializer` class.\nIt takes a `path_uri` argument that can be a local filesystem path, an\nHTTP/HTTPS endpoint, or an S3 endpoint.\n\n`load_into_module` is the main method of the `TensorDeserializer` class.\nIt takes a `torch.nn.Module` and loads the tensors from the `path_uri`\nendpoint into the `torch.nn.Module`.\n\nThe below example loads the `EleutherAI/gpt-j-6B` model from an S3\nendpoint.\n\n[deserialize-simple.py](examples/deserialize-simple.py)\n```python\nimport time\nimport torch\nfrom tensorizer import TensorDeserializer\nfrom tensorizer.utils import no_init_or_tensor, convert_bytes, get_mem_usage\n\nfrom transformers import AutoModelForCausalLM, AutoTokenizer, AutoConfig\n\nmodel_ref = \"EleutherAI/gpt-j-6B\"\n# To run this at home, swap this with the line below for a smaller example:\n# model_ref = \"EleutherAI/gpt-neo-125M\"\nmodel_name = model_ref.split(\"/\")[-1]\n# Change this to your S3 bucket.\ns3_bucket = \"bucket\"\ns3_uri = f\"s3://{s3_bucket}/{model_name}.tensors\"\n\nconfig = AutoConfig.from_pretrained(model_ref)\n\ndevice = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n# This ensures that the pretrained model weights are not initialized,\n# and non-persistent buffers (generated at runtime) are on the correct device.\nwith torch.device(device), no_init_or_tensor():\n model = AutoModelForCausalLM.from_config(config)\n\nprint(f\"Deserializing to {device}:\")\nbefore_mem = get_mem_usage()\n\n# Lazy load the tensors from S3 into the model.\nstart = time.perf_counter()\ndeserializer = TensorDeserializer(s3_uri, device=device)\ndeserializer.load_into_module(model)\nend = time.perf_counter()\n\nafter_mem = get_mem_usage()\n\n# Brag about how fast we are.\ntotal_bytes_str = convert_bytes(deserializer.total_tensor_bytes)\nduration = end - start\nper_second = convert_bytes(deserializer.total_tensor_bytes / duration)\ndeserializer.close()\nprint(f\"Deserialized {total_bytes_str} in {end - start:0.2f}s, {per_second}/s\")\nprint(f\"Memory usage before: {before_mem}\")\nprint(f\"Memory usage after: {after_mem}\")\n\n# Tokenize and generate\nmodel.eval()\ntokenizer = AutoTokenizer.from_pretrained(model_ref)\neos = tokenizer.eos_token_id\ninput_ids = tokenizer.encode(\n \"\u00a1Hola! Encantado de conocerte. hoy voy a\", return_tensors=\"pt\"\n).to(device)\n\nwith torch.no_grad():\n output = model.generate(\n input_ids, max_new_tokens=50, do_sample=True, pad_token_id=eos\n )\n\nprint(f\"Output: {tokenizer.decode(output[0], skip_special_tokens=True)}\")\n```\n\nIt should produce output similar to the following, with GPT-J-6B:\n```\nDeserialized model in 6.25 seconds\nTest Output: \u00a1Hola! Encantado de conocerte. hoy voy a comentar por primera\nvez una teor\u00eda de trineo, que quiz\u00e1 te parezca\nalgo desconocido, ya que en este mundo han\nllegado a dominar tantos\n```\n\nMore practical examples for the usage of `tensorizer` can be found in\n[examples/hf_serialization.py](examples/hf_serialization.py),\nwhere `df_main()` serializes models from\n[HuggingFace Diffusers](https://github.com/huggingface/diffusers)\nand `hf_main()` serializes\n[HuggingFace Transformers](https://github.com/huggingface/transformers) models.\n\n## Tensor Weight Encryption\n\n`tensorizer` supports fast tensor weight encryption and decryption during\nserialization and deserialization, respectively.\n\nBe aware that metadata (tensor names, dtypes, shapes, etc.) are not encrypted,\nonly the weights themselves.\n\n> [!NOTE]\n> \n> Refer to [docs/encryption.md](/docs/encryption.md) for details, instructions,\n> and warnings on using `tensorizer` encryption correctly and safely.\n\nTo use `tensorizer` encryption, a recent version of `libsodium` must be\ninstalled. Install `libsodium` with `apt-get install libsodium23`\non Ubuntu or Debian, or follow\n[the instructions in `libsodium`'s documentation](https://doc.libsodium.org/installation)\nfor other platforms.\n\n### Quick Encryption Example\n\nThe following outline demonstrates how to encrypt and decrypt a tensorized model\nwith a randomly-generated encryption key:\n\n```py\nfrom tensorizer import (\n EncryptionParams, DecryptionParams, TensorDeserializer, TensorSerializer\n)\n\n# Serialize and encrypt a model:\n\nencryption_params = EncryptionParams.random()\n\nserializer = TensorSerializer(\"model.tensors\", encryption=encryption_params)\nserializer.write_module(...) # or write_state_dict(), etc.\nserializer.close()\n\n# Save the randomly-generated encryption key somewhere\nwith open(\"tensor.key\", \"wb\") as key_file:\n key_file.write(encryption_params.key)\n\n\n# Then decrypt it again:\n\n# Load the randomly-generated key from where it was saved\nwith open(\"tensor.key\", \"rb\") as key_file:\n key: bytes = key_file.read()\n \ndecryption_params = DecryptionParams.from_key(key)\n\ndeserializer = TensorDeserializer(\"model.tensors\", encryption=decryption_params)\ndeserializer.load_into_module(...)\ndeserializer.close()\n```\n\nFor more detail, refer to [docs/encryption.md](/docs/encryption.md).\nA complete example is also available as\n[examples/encryption.py](examples/encryption.py).\nThe `EncryptionParams` and `DecryptionParams` class docstrings additionally\ncontain some usage information for quick reference from an IDE.\n\nAn example command line tool to add or remove encryption from existing\nserialized models is also available as\n[examples/encryption.py](examples/encrypt_existing.py).\n\n## Benchmarks\n\nYou can run your own benchmarks on CoreWeave or your own Kubernetes cluster\nby using the `benchmark.yaml` file in the `examples/benchmark_buffer_size`\ndirectory. Please see the [README](examples/benchmark_buffer_size/README.md).\n\n## Available Pre-Tensorized Models on the CoreWeave Cloud\n\nThe following models are available on the CoreWeave Cloud for free, and can be\nused with the `TensorDeserializer` class. The S3 support defaults to the\n`accel-object.ord1.coreweave.com` endpoint, and the bucket to use as `tensorized`.\n\nWe name the keys in the S3 bucket after the HuggingFace model identifier, and\nappend the `/fp16` suffix for the half-precision version.\n\nFor example, the S3 URI for the `EleutherAI/gpt-j-6B` model is:\n`s3://tensorized/EleutherAI/gpt-j-6B/fp16/model.tensors`\n\nThe below table shows the available models and their S3 URIs.\n\n### Large Language Models\n\n| Model | Precision | S3 URI |\n|-----------------------------------------------------------------------------------------|-----------|---------------------------------------------------------------------|\n| [EleutherAI/gpt-neo-125M](https://huggingface.co/EleutherAI/gpt-neo-125M) | `fp32` | `s3://tensorized/EleutherAI/gpt-neo-125M/model.tensors` |\n| [EleutherAI/gpt-neo-125M](https://huggingface.co/EleutherAI/gpt-neo-125M) | `fp16` | `s3://tensorized/EleutherAI/gpt-neo-125M/fp16/model.tensors` |\n| [EleutherAI/gpt-neo-1.3B](https://huggingface.co/EleutherAI/gpt-neo-1.3B) | `fp32` | `s3://tensorized/EleutherAI/gpt-neo-1.3B/model.tensors` |\n| [EleutherAI/gpt-neo-1.3B](https://huggingface.co/EleutherAI/gpt-neo-1.3B) | `fp16` | `s3://tensorized/EleutherAI/gpt-neo-1.3B/fp16/model.tensors` |\n| [EleutherAI/gpt-neo-2.7B](https://huggingface.co/EleutherAI/gpt-neo-2.7B) | `fp32` | `s3://tensorized/EleutherAI/gpt-neo-2.7B/model.tensors` |\n| [EleutherAI/gpt-neo-2.7B](https://huggingface.co/EleutherAI/gpt-neo-2.7B) | `fp16` | `s3://tensorized/EleutherAI/gpt-neo-2.7B/fp16/model.tensors` |\n| [EleutherAI/gpt-j-6B](https://huggingface.co/EleutherAI/gpt-j-6B) | `fp32` | `s3://tensorized/EleutherAI/gpt-j-6B/model.tensors` |\n| [EleutherAI/gpt-j-6B](https://huggingface.co/EleutherAI/gpt-j-6B) | `fp16` | `s3://tensorized/EleutherAI/gpt-j-6B/fp16/model.tensors` |\n| [EleutherAI/gpt-neox-20b](https://huggingface.co/EleutherAI/gpt-neox-20b) | `fp32` | `s3://tensorized/EleutherAI/gpt-neox-20b/model.tensors` |\n| [EleutherAI/gpt-neox-20b](https://huggingface.co/EleutherAI/gpt-neox-20b) | `fp16` | `s3://tensorized/EleutherAI/gpt-neox-20b/fp16/model.tensors` |\n| [EleutherAI/pythia-70m](https://huggingface.co/EleutherAI/pythia-70m) | `fp32` | `s3://tensorized/EleutherAI/pythia-70m/model.tensors` |\n| [EleutherAI/pythia-70m](https://huggingface.co/EleutherAI/pythia-70m) | `fp16` | `s3://tensorized/EleutherAI/pythia-70m/fp16/model.tensors` |\n| [EleutherAI/pythia-1.4b](https://huggingface.co/EleutherAI/pythia-1.4b) | `fp32` | `s3://tensorized/EleutherAI/pythia-1.4b/model.tensors` |\n| [EleutherAI/pythia-1.4b](https://huggingface.co/EleutherAI/pythia-1.4b) | `fp16` | `s3://tensorized/EleutherAI/pythia-1.4b/fp16/model.tensors` |\n| [EleutherAI/pythia-2.8b](https://huggingface.co/EleutherAI/pythia-2.8b) | `fp32` | `s3://tensorized/EleutherAI/pythia-2.8b/model.tensors` |\n| [EleutherAI/pythia-2.8b](https://huggingface.co/EleutherAI/pythia-2.8b) | `fp16` | `s3://tensorized/EleutherAI/pythia-2.8b/fp16/model.tensors` |\n| [EleutherAI/pythia-6.9b](https://huggingface.co/EleutherAI/pythia-6.9b) | `fp32` | `s3://tensorized/EleutherAI/pythia-6.9b/model.tensors` |\n| [EleutherAI/pythia-6.9b](https://huggingface.co/EleutherAI/pythia-6.9b) | `fp16` | `s3://tensorized/EleutherAI/pythia-6.9b/fp16/model.tensors` |\n| [EleutherAI/pythia-12b](https://huggingface.co/EleutherAI/pythia-12b) | `fp32` | `s3://tensorized/EleutherAI/pythia-12b/model.tensors` |\n| [EleutherAI/pythia-12b](https://huggingface.co/EleutherAI/pythia-12b) | `fp16` | `s3://tensorized/EleutherAI/pythia-12b/fp16/model.tensors` |\n| [EleutherAI/pythia-70m-deduped](https://huggingface.co/EleutherAI/pythia-70m-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-70m-deduped/model.tensors` |\n| [EleutherAI/pythia-70m-deduped](https://huggingface.co/EleutherAI/pythia-70m-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-70m-deduped/fp16/model.tensors` |\n| [EleutherAI/pythia-1.4b-deduped](https://huggingface.co/EleutherAI/pythia-1.4b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-1.4b-deduped/model.tensors` |\n| [EleutherAI/pythia-1.4b-deduped](https://huggingface.co/EleutherAI/pythia-1.4b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-1.4b-deduped/fp16/model.tensors` |\n| [EleutherAI/pythia-2.8b-deduped](https://huggingface.co/EleutherAI/pythia-2.8b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-2.8b-deduped/model.tensors` |\n| [EleutherAI/pythia-2.8b-deduped](https://huggingface.co/EleutherAI/pythia-2.8b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-2.8b-deduped/fp16/model.tensors` |\n| [EleutherAI/pythia-6.9b-deduped](https://huggingface.co/EleutherAI/pythia-6.9b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-6.9b-deduped/model.tensors` |\n| [EleutherAI/pythia-6.9b-deduped](https://huggingface.co/EleutherAI/pythia-6.9b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-6.9b-deduped/fp16/model.tensors` |\n| [EleutherAI/pythia-12b-deduped](https://huggingface.co/EleutherAI/pythia-12b-deduped) | `fp32` | `s3://tensorized/EleutherAI/pythia-12b-deduped/model.tensors` |\n| [EleutherAI/pythia-12b-deduped](https://huggingface.co/EleutherAI/pythia-12b-deduped) | `fp16` | `s3://tensorized/EleutherAI/pythia-12b-deduped/fp16/model.tensors` |\n| [KoboldAI/fairseq-dense-125M](https://huggingface.co/KoboldAI/fairseq-dense-125M) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-125M/model.tensors` |\n| [KoboldAI/fairseq-dense-125M](https://huggingface.co/KoboldAI/fairseq-dense-125M) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-125M/fp16/model.tensors` |\n| [KoboldAI/fairseq-dense-355M](https://huggingface.co/KoboldAI/fairseq-dense-355M) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-355M/model.tensors` |\n| [KoboldAI/fairseq-dense-355M](https://huggingface.co/KoboldAI/fairseq-dense-355M) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-355M/fp16/model.tensors` |\n| [KoboldAI/fairseq-dense-2.7B](https://huggingface.co/KoboldAI/fairseq-dense-2.7B) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-2.7B/model.tensors` |\n| [KoboldAI/fairseq-dense-2.7B](https://huggingface.co/KoboldAI/fairseq-dense-2.7B) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-2.7B/fp16/model.tensors` |\n| [KoboldAI/fairseq-dense-6.7B](https://huggingface.co/KoboldAI/fairseq-dense-6.7B) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-6.7B/model.tensors` |\n| [KoboldAI/fairseq-dense-6.7B](https://huggingface.co/KoboldAI/fairseq-dense-6.7B) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-6.7B/fp16/model.tensors` |\n| [KoboldAI/fairseq-dense-13B](https://huggingface.co/KoboldAI/fairseq-dense-13B) | `fp32` | `s3://tensorized/KoboldAI/fairseq-dense-13B/model.tensors` |\n| [KoboldAI/fairseq-dense-13B](https://huggingface.co/KoboldAI/fairseq-dense-13B) | `fp16` | `s3://tensorized/KoboldAI/fairseq-dense-13B/fp16/model.tensors` |\n| [Salesforce/codegen-350M-mono](https://huggingface.co/Salesforce/codegen-350M-mono) | `fp32` | `s3://tensorized/Salesforce/codegen-350M-mono/model.tensors` |\n| [Salesforce/codegen-350M-mono](https://huggingface.co/Salesforce/codegen-350M-mono) | `fp16` | `s3://tensorized/Salesforce/codegen-350M-mono/fp16/model.tensors` |\n| [Salesforce/codegen-350M-multi](https://huggingface.co/Salesforce/codegen-350M-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-350M-multi/model.tensors` |\n| [Salesforce/codegen-350M-multi](https://huggingface.co/Salesforce/codegen-350M-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-350M-multi/fp16/model.tensors` |\n| [Salesforce/codegen-2B-multi](https://huggingface.co/Salesforce/codegen-2B-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-2B-multi/model.tensors` |\n| [Salesforce/codegen-2B-multi](https://huggingface.co/Salesforce/codegen-2B-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-2B-multi/fp16/model.tensors` |\n| [Salesforce/codegen-6B-mono](https://huggingface.co/Salesforce/codegen-6B-mono) | `fp32` | `s3://tensorized/Salesforce/codegen-6B-mono/model.tensors` |\n| [Salesforce/codegen-6B-mono](https://huggingface.co/Salesforce/codegen-6B-mono) | `fp16` | `s3://tensorized/Salesforce/codegen-6B-mono/fp16/model.tensors` |\n| [Salesforce/codegen-6B-multi](https://huggingface.co/Salesforce/codegen-6B-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-6B-multi/model.tensors` |\n| [Salesforce/codegen-6B-multi](https://huggingface.co/Salesforce/codegen-6B-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-6B-multi/fp16/model.tensors` |\n| [Salesforce/codegen-16B-mono](https://huggingface.co/Salesforce/codegen-16B-mono) | `fp32` | `s3://tensorized/Salesforce/codegen-16B-mono/model.tensors` |\n| [Salesforce/codegen-16B-mono](https://huggingface.co/Salesforce/codegen-16B-mono) | `fp16` | `s3://tensorized/Salesforce/codegen-16B-mono/fp16/model.tensors` |\n| [Salesforce/codegen-16B-multi](https://huggingface.co/Salesforce/codegen-16B-multi) | `fp32` | `s3://tensorized/Salesforce/codegen-16B-multi/model.tensors` |\n| [Salesforce/codegen-16B-multi](https://huggingface.co/Salesforce/codegen-16B-multi) | `fp16` | `s3://tensorized/Salesforce/codegen-16B-multi/fp16/model.tensors` |\n\n### Generative Diffusion Models\n\n| Model | Component | Precision | S3 URI |\n|-------------------------------------------------------------------------------------------------------------|--------------|-----------|----------------------------------------------------------------------------------------|\n| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `VAE` | `fp32` | `s3://tensorized/runwayml/stable-diffusion-v1-5/vae.tensors` |\n| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `UNet` | `fp32` | `s3://tensorized/runwayml/stable-diffusion-v1-5/unet.tensors` |\n| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `TextEnc` | `fp32` | `s3://tensorized/runwayml/stable-diffusion-v1-5/text_encoder.tensors` |\n| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `VAE` | `fp16` | `s3://tensorized/runwayml/stable-diffusion-v1-5/fp16/vae.tensors` |\n| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `UNet` | `fp16` | `s3://tensorized/runwayml/stable-diffusion-v1-5/fp16/unet.tensors` |\n| [RunwayML/stable-diffusion-v1-5](https://huggingface.co/runwayml/stable-diffusion-v1-5) | `TextEnc` | `fp16` | `s3://tensorized/runwayml/stable-diffusion-v1-5/fp16/text_encoder.tensors` |\n| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `VAE` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-2-1/vae.tensors` |\n| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `UNet` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-2-1/unet.tensors` |\n| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `TextEnc` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-2-1/text_encoder.tensors` |\n| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `VAE` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-2-1/fp16/vae.tensors` |\n| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `UNet` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-2-1/fp16/unet.tensors` |\n| [StabilityAI/stable-diffusion-2-1](https://huggingface.co/stabilityai/stable-diffusion-2-1) | `TextEnc` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-2-1/fp16/text_encoder.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `VAE` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/vae.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `UNet` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/unet.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/text_encoder.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc2` | `fp32` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/text_encoder_2.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `VAE` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/vae.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `UNet` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/unet.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/text_encoder.tensors` |\n| [StabilityAI/stable-diffusion-xl-base-1.0](https://huggingface.co/stabilityai/stable-diffusion-xl-base-1.0) | `TextEnc2` | `fp16` | `s3://tensorized/stabilityai/stable-diffusion-xl-base-1.0/fp16/text_encoder_2.tensors` |\n\n\n## S3 Usage Notes\n`tensorizer` uses the `boto3` library to interact with S3. The easiest way\nto use `tensorizer` with S3 is to configure your S3 credentials in\n`~/.s3cfg`.\n\nIf you don't want to use `~/.s3cfg`, or wish to use a `.s3cfg` config file\nsaved at a nonstandard location (e.g. under `/var/run`), you can also specify\nyour S3 credentials using the `tensorizer.stream_io.open_stream()` function,\nand then pass that into the `TensorSerializer` or `TensorDeserializer`\nconstructor.\n\nThe `stream_io.open_stream()` function takes a `path_uri` argument, which can\nbe an `s3://` URI, and accepts the following keyword arguments:\n* `s3_access_key_id`: S3 access key ID\n* `s3_secret_access_key`: S3 secret access key\n* `s3_endpoint`: S3 endpoint\n\n*Or,*\n\n* `s3_config_path`: Alternative filesystem path to a `.s3cfg` config file\n\nFor example:\n```python\nTensorSerializer(\n open_stream(s3_uri,\n \"wb\",\n s3_access_key_id=ACCESS_KEY,\n s3_secret_access_key=SECRET_KEY,\n s3_endpoint=\"object.ord1.coreweave.com\"))\n```\n\nand...\n\n```python\nTensorDeserializer(\n open_stream(s3_uri,\n \"rb\",\n s3_access_key_id=ACCESS_KEY,\n s3_secret_access_key=SECRET_KEY,\n s3_endpoint=\"object.ord1.coreweave.com\"))\n```\n\n**NOTE:** For faster object downloads in the CoreWeave Cloud, you can use\nthe `accel-object.ord1.coreweave.com` endpoint. This endpoint is optimized\nfor object downloads, and will be faster than the `object.ord1.coreweave.com`\nendpoint once the object is cached.\n\n**NOTE2:** The cache above does not get invalidated when the object is updated\nin S3. If you update an object in S3, you will need to wait for the cache to\nexpire before you can download the updated object. This takes 24 hours since\nthe last download.\n\nFor this reason, it is recommended to use a unique S3 key for each version\nof a model if you use the `accel-object.ord1.coreweave.com` endpoint.\n\n## Additional Features\n`tensorizer` has a few additional features that make it more useful than\njust a serialization/deserialization tool.\n\n### Concurrent Reads\n\nThe `TensorDeserializer` class has a `num_readers` argument that controls\nhow many threads are allowed to read concurrently from the source file.\nThis can greatly improve performance, since in many cases the network or the\nfile is the bottleneck. A few caveats to running with `num_readers > 1`:\n\n* The specified file must be able to be reopened, so that the\n `TensorDeserializer` can open more streams against the source.\n * Local files, paths, and HTTP(S) and S3 URIs / open streams\n are all able to be reopened\n * Special files like pipes and sockets, or synthetic file-like objects such as\n `BytesIO` are not currently able to be reopened\n* For HTTP(S) and S3 streams and URIs, the host must support the `Range` header.\n Each reader will read a stream from a different Range offset in the source.\n\nThe default is `num_readers=1`, which has no special requirements.\n\n### `state_dict` Support\n\nThe `TensorDeserializer` object can be used as-is as a `state_dict` for\n`torch.nn.Module.load_state_dict`. This is useful for loading the tensors\ninto a `torch.nn.Module` that is already initialized, or for inspection.\n\nKeep in mind that `load_state_dict` is not a fast operation, and will\nlikely be much slower than `load_into_module`.\n\nThe `state_dict` can also be used to initialize a HuggingFace Transformers\nAutoModel. But HuggingFace Transformers performs three or more copies of\nthe data, so memory use will explode.\n\n### `bfloat16` Support\n\nTensorizer supports models using the `bfloat16` data type. However, tensorizer\nuses numpy to save the tensors as binary and numpy doesn't support `bfloat16`.\nThis means that special conversions need to be applied.\n\nTo be saved, the torch tensor is cast to `int16` before being converted to\nnumpy, which doesn't change any of the underlying data. When serialized, the\noriginal `bfloat16` datatype string is also saved so that it will be cast back\nto `bfloat16` during the deserialization process.\n\nThe `complex32` datatype is supported in a similar way, by casting to `int32`.\nThe quantized datatypes (`qint8`, `qint32`, etc.) are not currently supported\nby tensorizer as they would require supplemental quantization parameters to be\ndeserialized correctly.\n\n**NOTE:** The exact choice of intermediate types as `int16` and `int32` is\nconsidered an implementation detail, and is subject to change,\nso they should not be relied upon.\n\n**NOTE2:** This does not interfere with storing actual `int` datatypes\nused in tensors in tensorized files.\n\n### Numpy Support\n\nTensorizer can be used with `numpy` directly to read and write\n`numpy.ndarray`s.\n\nThe serializer's `write_tensor` function handles supplying both\n`torch.Tensor`s and `numpy.ndarray`s.\n\nThe deserializer has a separate function `read_numpy_arrays` that will return\nthe data as `numpy.ndarray`s.\n\nAs explained above in [bfloat16 support](#bfloat16-support), tensorizer uses\nspecial conversions to write \"opaque\" datatypes, those not supported by numpy.\nTherefore, special considerations need to be taken when loading such data as\n`numpy.ndarray`s.\n\nBy default, the `TensorDeserializer.read_numpy_arrays` function sets its\n`allow_raw_data` parameter to `False`. This means that if a file contains\nopaque datatypes, a `ValueError` will be raised during deserialization.\n\nIf you want to return the raw data regardless, set `allow_raw_data` to `True`.\nOtherwise, the file may be read with `TensorDeserializer.read_tensors`\ninstead, which yields `torch.Tensor` objects of the correct datatype.\n\nA fifth and sixth variable are also returned by the `read_numpy_arrays`\ngenerator. The fifth is a `bool` that indicates whether the returned array\nhas an opaque datatype and requires special handling (only legal when\n`allow_raw_data=True`). The sixth is a string describing the true, non-numpy\ndatatype that the raw data should be interpreted as in such cases.\nFor all other datatypes that require no special handling, these are returned as\n`False` and `None`, respectively.\nThe exact numpy datatypes used by the returned opaque `numpy.ndarray` objects\nis not guaranteed, and should not be relied upon.\n\n### Plaid mode\nOlder versions of Tensorizer had an argument called `plaid_mode` that reused\nbuffers when copying to CUDA devices. This now happens automatically.\n`plaid_mode` and `plaid_mode_buffers` are left as arguments for backwards\ncompatibility but are deprecated and have no effect.\n\n## Running Tests\n`tensorizer` uses `unittest` for testing.\nThe tests have their own set of dependencies, which can be installed with\n`pip install -r tests/requirements.txt`.\n\nSome tests require a GPU, and will be skipped if no GPU is available.\nTo run the tests, run the following in the root of the repository:\n\n```bash\npython -m pip install -e .\npython -m pip install -r tests/requirements.txt\npython -m unittest discover tests/ --verbose\n```\n\n## Serialization in a subprocess\nYou may want to do Serialization in a separate process so that your main process can continue executing and not get bogged down by GIL contention.\nSee [docs/subprocess-serialization.md](docs/subprocess-serialization.md) for more details.\n",
"bugtrack_url": null,
"license": "MIT License",
"summary": "A tool for fast PyTorch module, model, and tensor serialization + deserialization.",
"version": "2.9.1",
"project_urls": {
"Changelog": "https://github.com/coreweave/tensorizer/blob/main/CHANGELOG.md",
"Homepage": "https://github.com/coreweave/tensorizer",
"Repository": "https://github.com/coreweave/tensorizer"
},
"split_keywords": [
"tensorizer",
" machine learning",
" serialization",
" tensor",
" pytorch"
],
"urls": [
{
"comment_text": "",
"digests": {
"blake2b_256": "780977c9ac288c1b15b78fa449ff1b77b4d947d24fabe866fcc368a033e990b0",
"md5": "df6a66fa680229a7c8f0ff8c111ba9fb",
"sha256": "eacfed20674ca139f88b24229fd21eb7a1806607fdc3204c46c50b3ef87a600b"
},
"downloads": -1,
"filename": "tensorizer-2.9.1-py3-none-any.whl",
"has_sig": false,
"md5_digest": "df6a66fa680229a7c8f0ff8c111ba9fb",
"packagetype": "bdist_wheel",
"python_version": "py3",
"requires_python": ">=3.8",
"size": 96028,
"upload_time": "2024-11-28T01:00:16",
"upload_time_iso_8601": "2024-11-28T01:00:16.160610Z",
"url": "https://files.pythonhosted.org/packages/78/09/77c9ac288c1b15b78fa449ff1b77b4d947d24fabe866fcc368a033e990b0/tensorizer-2.9.1-py3-none-any.whl",
"yanked": false,
"yanked_reason": null
},
{
"comment_text": "",
"digests": {
"blake2b_256": "0aed2062ab9492612309c51d78ccdc0cb316a2ecc8c90e08fd4d1754540ee0df",
"md5": "d82205c2c307b8255a809e07aa744492",
"sha256": "f79ac50d0a3bc328e3ee8cdf838ba04ddb5b9fa1d72be69f6ae618533d349e8e"
},
"downloads": -1,
"filename": "tensorizer-2.9.1.tar.gz",
"has_sig": false,
"md5_digest": "d82205c2c307b8255a809e07aa744492",
"packagetype": "sdist",
"python_version": "source",
"requires_python": ">=3.8",
"size": 133695,
"upload_time": "2024-11-28T01:00:17",
"upload_time_iso_8601": "2024-11-28T01:00:17.166243Z",
"url": "https://files.pythonhosted.org/packages/0a/ed/2062ab9492612309c51d78ccdc0cb316a2ecc8c90e08fd4d1754540ee0df/tensorizer-2.9.1.tar.gz",
"yanked": false,
"yanked_reason": null
}
],
"upload_time": "2024-11-28 01:00:17",
"github": true,
"gitlab": false,
"bitbucket": false,
"codeberg": false,
"github_user": "coreweave",
"github_project": "tensorizer",
"travis_ci": false,
"coveralls": false,
"github_actions": true,
"requirements": [],
"lcname": "tensorizer"
}