Petastorm is a library enabling the use of Parquet storage from Tensorflow, Pytorch, and other Python-based ML training frameworks. From the perspective of deep-learning training and evaluation procedures, using datasets stored in Parquet bridges a common, big-data storage format with deep-learning frameworks, making it simpler to combine them.
Petastorm defines tensorflow or tensorflow-gpu as an “extra” package. That way you can choose to depend on either a CPU or a GPU version of tensorflow, or not to install it at all.
Note that although petastorm depends on opencv, it is defined as an extra package to allow opencv to be installed by debian package (or source) as opposed to pip. It is recommended to install it when testing out petastorm however (and is therefore shown in the examples).
Install with tensorflow CPU dependency:
pip install petastorm[opencv, tf]
Install with tensorflow GPU dependency:
pip install petastorm[opencv, tf_gpu]
A dataset created using petastorm is stored in Parquet format. On top of a Parquet schema, petastorm also stores higher-level schema information that makes multidimensional arrays into a native part of a petastorm dataset.
Petastorm also supports extensible data codecs. These enable a user to use one of the standard data compressions (jpeg, png) or implement her own.
Generating a dataset is done using pyspark. Pyspark natively supports Parquet format, making it easy to run on a single machine or on a Spark compute cluster. Here is a minimalistic example writing out a table with some random data.
HelloWorldSchema = Unischema('HelloWorldSchema', [
UnischemaField('id', np.int32, (), ScalarCodec(IntegerType()), False),
UnischemaField('image1', np.uint8, (128, 256, 3), CompressedImageCodec('png'), False),
UnischemaField('other_data', np.uint8, (None, 128, 30, None), NdarrayCodec(), False),
])
def row_generator(x):
"""Returns a single entry in the generated dataset. Return a bunch of random values as an example."""
return {'id': x,
'image1': np.random.randint(0, 255, dtype=np.uint8, size=(128, 256, 3)),
'other_data': np.random.randint(0, 255, dtype=np.uint8, size=(4, 128, 30, 3))}
def generate_hello_world_dataset(output_url='file:///tmp/hello_world_dataset'):
rows_count = 10
rowgroup_size_mb = 256
spark = SparkSession.builder.config('spark.driver.memory', '2g').master('local[2]').getOrCreate()
sc = spark.sparkContext
# Wrap dataset materialization portion. Will take care of setting up spark environment variables as
# well as save petastorm specific metadata
with materialize_dataset(spark, output_url, HelloWorldSchema, rowgroup_size_mb):
rows_rdd = sc.parallelize(range(rows_count))\
.map(row_generator)\
.map(lambda x: dict_to_spark_row(HelloWorldSchema, x))
spark.createDataFrame(rows_rdd, HelloWorldSchema.as_spark_schema()) \
.coalesce(10) \
.write \
.mode('overwrite') \
.parquet(output_url)
HelloWorldSchema
is an instance of aUnischema
object.Unischema
is capable of rendering types of its fields into different framework specific formats, such as: Spark’sStructType
, Tensorflow’stf.DType
and numpy’snumpy.dtype
.- To define a dataset field, you need to specify a type, shape, a codec instance and whether the field is nullable for each field of the Unischema.
- We use pyspark for writing output Parquet files. In this example, we launch pyspark on a local box (
.master('local[2]')
). Of course for a larger scale dataset generation we would need a real compute cluster. - We wrap spark dataset generation code with
materialize_dataset
context manager. The context manager is responsible to configure row group size at the beginning and write out petastorm specific metadata at the end. - The row generating code is expected to return a Python dictionary indexed by a field name. We use
row_generator
function for that. dict_to_spark_row
converts the dictionary into apyspark.Row
object while ensuring schemaHelloWorldSchema
compliance (shape, type and is-nullable condition are tested).- Once we have a
pyspark.DataFrame
we write it out to a parquet storage. The parquet schema is automatically derives fromHelloWorldSchema
.
Reading a dataset is simple using petastorm.Reader
class:
with Reader('hdfs://myhadoop/some_dataset') as reader:
for row in reader:
print(row)
Reader
supports taking either a hdfs://...
or file://...
protocols.
Once a Reader is instantiated, you can use it as an iterator.
To hookup the reader into a tensorflow graph, you can use tf_tensors
function:
with Reader('file:///some/localpath/a_dataset') as reader:
row_tensors = tf_tensors(reader)
with tf.Session() as session:
for _ in range(3):
print(session.run(row_tensors))
petastorm.Reader
is the main entry point for a user code that accesses the data from an ML framework such as Tensorflow or Pytorch. The reader has multiple features such as:
- Selective column readout
- Multiple parallelism strategies: thread, process, single-threaded (for debug)
- N-grams readout support
- Row filtering (row predicates)
- Shuffling
- Partitioning for multi-GPU training
- Local caching
As illustrated in examples/mnist/main.py, reading a petastorm dataset from pytorch can be done via the adapter class petastorm.pytorch.DataLoader
,
which allows custom pytorch collating function and transforms to be supplied.
Be sure you have torch and torchvision installed:
pip install torchvision
The minimalist example below assumes the definition of a Net
class and train
and test
functions, shown in the mnist example:
import torch
from petastorm.pytorch import DataLoader
torch.manual_seed(1)
device = torch.device('cpu')
model = Net().to(device)
optimizer = torch.optim.SGD(model.parameters(), lr=0.01, momentum=0.5)
def _transform_row(mnist_row):
transform = transforms.Compose([
transforms.ToTensor(),
transforms.Normalize((0.1307,), (0.3081,))
])
return (transform(mnist_row.image), mnist_row.digit)
with DataLoader(Reader('file:///localpath/mnist/train', num_epochs=10),
batch_size=64, transform=_transform_row) as train_loader:
train(model, device, train_loader, 10, optimizer, 1)
with DataLoader(Reader('file:///localpath/mnist/test', num_epochs=10),
batch_size=1000, transform=_transform_row) as test_loader:
test(model, device, test_loader)
See the Troubleshooting page and please submit a ticket if you can't find an answer.
See the Development page for instructions on how to develop Petastorm and run tests.