Metadata-Version: 2.4
Name: atpbar
Version: 2.0.7
Summary: Progress bars for threading and multiprocessing tasks
Project-URL: Homepage, https://github.com/alphatwirl/atpbar
Author-email: Tai Sakuma <tai.sakuma@gmail.com>
License-Expression: MIT
License-File: LICENSE
Classifier: Development Status :: 5 - Production/Stable
Classifier: Intended Audience :: Developers
Classifier: Intended Audience :: End Users/Desktop
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Programming Language :: Python :: 3.14
Requires-Python: >=3.10
Requires-Dist: typing-extensions>=4.12; python_version < '3.10'
Provides-Extra: doc
Requires-Dist: mkdocs-material>=9.5; extra == 'doc'
Requires-Dist: pymdown-extensions>=10.8; extra == 'doc'
Provides-Extra: jupyter
Requires-Dist: ipywidgets>=8.1; extra == 'jupyter'
Provides-Extra: tests
Requires-Dist: hypothesis>=6.102.6; extra == 'tests'
Requires-Dist: pytest-console-scripts>=1.4; extra == 'tests'
Requires-Dist: pytest-cov>=5.0; extra == 'tests'
Requires-Dist: pytest>=8.2; extra == 'tests'
Description-Content-Type: text/markdown

[![PyPI - Version](https://img.shields.io/pypi/v/atpbar.svg)](https://pypi.org/project/atpbar)
[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/atpbar.svg)](https://pypi.org/project/atpbar)
[![Anaconda-Server Badge](https://anaconda.org/conda-forge/atpbar/badges/version.svg)](https://anaconda.org/conda-forge/atpbar)
[![DOI](https://zenodo.org/badge/doi/10.5281/zenodo.2567283.svg)](https://doi.org/10.5281/zenodo.2567283)

[![Test Status](https://github.com/alphatwirl/atpbar/actions/workflows/unit-test.yml/badge.svg)](https://github.com/alphatwirl/atpbar/actions/workflows/unit-test.yml)
[![Test Status](https://github.com/alphatwirl/atpbar/actions/workflows/type-check.yml/badge.svg)](https://github.com/alphatwirl/atpbar/actions/workflows/type-check.yml)
[![codecov](https://codecov.io/gh/alphatwirl/atpbar/branch/master/graph/badge.svg)](https://codecov.io/gh/alphatwirl/atpbar)

# atpbar

_Progress bars_ for threading and multiprocessing tasks on the terminal and
Jupyter Notebook.

![Demo](demo.svg)

_atpbar_ can display multiple progress bars simultaneously growing to show the
progress of each iteration of loops in
[threading](https://docs.python.org/3/library/threading.html) or
[multiprocessing](https://docs.python.org/3/library/multiprocessing.html)
tasks. _atpbar_ can display progress bars on the terminal and [Jupyter
Notebook](https://jupyter.org/).

## Quick Start

### Installation

If `atpbar` is not installed, you can install it with the `pip` command on the
terminal.

```bash
pip install -U atpbar
```

### How to use

#### Start Python

You can try the examples in this tutorial in the Python interactive shell.

```bash
$ python
Python 3.13.9 (...)
...
...
>>>
```

#### Import packages

Import `atpbar` and other objects that we will use in the examples.

```python
from random import randint
from time import sleep
from atpbar import atpbar
```

#### One loop

The `atpbar` can wrap an iterable to show a progress bar for the iterations.

```python
n = randint(1000, 10000)
for _ in atpbar(range(n)):
    sleep(0.001)
```

This example randomly selects the number of iterations and, in each iteration,
sleeps for a short time.

The progress bar will be shown as the loop progresses.

```plaintext
  51.25% ::::::::::::::::::::                     |     4132 /     8062 |:  range(0, 8062)
```

Note: `atpbar` won't show a progress bar if the length of the iterable cannot be
obtained by `len()`.

#### Nested loops

The `atpbar` can show progress bars for nested loops.

```python
for i in atpbar(range(4), name='Outer'):
    n = randint(1000, 10000)
    for _ in atpbar(range(n), name=f'Inner {i}'):
        sleep(0.001)
```

This example iterates over an outer loop four times. In each iteration, it
iterates over an inner loop. The progress bars for both the outer and inner
loops are shown.

```plaintext
 100.00% :::::::::::::::::::::::::::::::::::::::: |     3287 /     3287 |:  Inner 0
 100.00% :::::::::::::::::::::::::::::::::::::::: |     5850 /     5850 |:  Inner 1
  50.00% ::::::::::::::::::::                     |        2 /        4 |:  Outer
  34.42% :::::::::::::                            |     1559 /     4529 |:  Inner 2
```

In the snapshot of the progress bars above, the outer loop is in its 3rd
iteration. The inner loop has been completed twice and is running the third.
The progress bars for the completed tasks move up. The progress bars for the
active tasks are growing at the bottom.

#### Threading

As the last example, we show how to use `atpbar` with threading. We will use
the
[`ThreadPoolExecutor`](https://docs.python.org/3/library/concurrent.futures.html#threadpoolexecutor)
from the
[`concurrent.futures`](https://docs.python.org/3/library/concurrent.futures.html)
module.

Import `ThreadPoolExecutor` and also `flushing` from `atpbar`.

```python
from concurrent.futures import ThreadPoolExecutor
from atpbar import flushing
```

Define a function that will be executed by the threads.

```python
def func(n, name):
    for _ in atpbar(range(n), name=name):
        sleep(0.001)
```

We will submit ten jobs each runs the `func` function to five threads.

```python
n_workers = 5
n_jobs = 10

with flushing(), ThreadPoolExecutor(max_workers=n_workers) as executor:
    for i in range(n_jobs):
        n = randint(1000, 10000)
        f = executor.submit(func, n, name=f'Job {i}')

```

The context manager `flushing()` exits after the progress bars have finished
updating.

The progress bars will be simultaneously updated for concurrent jobs.

```plaintext
 100.00% :::::::::::::::::::::::::::::::::::::::: |     2326 /     2326 |:  Job 0
 100.00% :::::::::::::::::::::::::::::::::::::::: |     2971 /     2971 |:  Job 1
 100.00% :::::::::::::::::::::::::::::::::::::::: |     1386 /     1386 |:  Job 6
 100.00% :::::::::::::::::::::::::::::::::::::::: |     5316 /     5316 |:  Job 3
 100.00% :::::::::::::::::::::::::::::::::::::::: |     7786 /     7786 |:  Job 4
 100.00% :::::::::::::::::::::::::::::::::::::::: |     5500 /     5500 |:  Job 5
  91.33% ::::::::::::::::::::::::::::::::::::     |     8188 /     8965 |:  Job 2
  39.85% :::::::::::::::                          |     3842 /     9642 |:  Job 7
  34.89% :::::::::::::                            |     2882 /     8260 |:  Job 8
  29.11% :::::::::::                              |      414 /     1422 |:  Job 9
```

### For more information

This is the end of the quick start tutorial. For more information, see
[the Users Guide](https://alphatwirl.github.io/atpbar/guide/index.html).
