The lib allows for threaded pipelines as well as so-called co-routine pipelines. The main use-case is limiting peak memory usage when doing complex operations on large-ish data-sets.
>>> pipeline = paipa.Pipeline( ... (DownloadImage, 4), ... (StoreDatabase, 1), ... ) >>> pipeline.run()
This example will create 5 threads, 4 for downloading images, one for storing stuff to the database. The ouputs of all DownloadImage steps will be forwared to the one StoreDatabase thread via a Queue.
Thread startup and tear-down is handled by the library and doesn’t concern the programmer at all. All (known) failure cases lead to either a re-spawning of the failed thread or a controlled shutdown of the system. In no case the system should block and do nothing, if it does then it’s definitely a bug and needs to be reported.
Pipeline ingestion can be done via a separate thread or by consuming an iterable. In the case of using an iterable, an ingestion thread is created which consumes the iterable in a controled manner.
The whole pipeline can also be run in the background like this:
>>> pl = pipeline.run_forever(background=True) >>> # do some other stuff >>> pl.stop() # this waits for the system to complete and do a shutdown
>>> def remove_odd(iterable): ... for entry in iterable: ... if entry % 2 == 0: ... yield entry >>> steps = [remove_odd]# Arbitrary many steps supported >>> gen = combine_pipeline(range(100), steps) >>> print(sum(gen)) 2450
This will create nested generators which do the relevant processing. The individual steps only need to support the iterator protocol and don’t necessarily need to be generators. Memory usage may make using generators appealing though.