process I/O progress viewer

progress is a small Linux application which reports the progress of file read/write operations initiated by coreutils programs like cp, mv, and dd, but also arbitrary processes that have entries in /proc.

progress overview

Quick Facts
Version Controlgit on github
Author(Xfennec)
LanguageC
Binary Size56K
Version Reviewed0.14

Installation

progress is installed via source distribution and make, like so:

git clone https://github.com/Xfennec/progress
cd progress/
make
make install

There’s one dependency, on the library ncurses- you may already have the development headers installed, but if you don’t it may be called ncurses-devel or libncurses-dev.

Unfortunately since progress relies on the /proc pseudo-filesystem, this utility only works on Linux.

Usage

Running progress with no arguments prints out any currently-running coreutils processes that are transferring data:

$ progress

Whoops, I don’t have any running. Let’s set up a script that is constantly reading and writing files, so we can illustrate progress on some relatively common operations. The following bash script will create a few files between 1 and 10 gigabytes, copy them to /tmp, then remove the files in /tmp and remove the original files.

#!/bin/bash -e

ORIGIN_FILENAMES=(File_1000000 File_2000000 File_10000000)
TMP_DIR=/tmp/progress_test

# Create a few large files using dd
for fn in ${ORIGIN_FILENAMES[*]}
do
  # Spawn each dd process with a different size
  size=$(echo "$fn" | awk -F'_' '{print $NF}')
  dd if=/dev/urandom of="$fn" bs=1024 count="$size" &
done
# Wait on all to complete before continuing
wait < <(jobs -p)

# Copy these large files into /tmp simultaneously
mkdir -p "$TMP_DIR" 
for fn in ${ORIGIN_FILENAMES[*]}
do
  # Spawn each copy process
  cp "$fn" "$TMP_DIR" &
done
# Wait on all processes before continuing
wait < <(jobs -p)

# Verify they were copied, remove them from /tmp
for fn in ${ORIGIN_FILENAMES[*]}
do
  test -f "$TMP_DIR/$fn" && rm "$TMP_DIR/$fn"
done
rmdir "$TMP_DIR"

# Remove the original files
for fn in ${ORIGIN_FILENAMES[*]}
do
  rm "$fn" &
done
wait < <(jobs -p)

Now that we have some operations to report, let’s try progress:

$ progress
[20293] dd /home/delucks/dev/cli.fan/File_1000000
        100.0% (440.1 MiB / 440.1 MiB)

[20297] dd /home/delucks/dev/cli.fan/File_2000000
        100.0% (433.2 MiB / 433.2 MiB)

[20301] dd /home/delucks/dev/cli.fan/File_10000000
        100.0% (413.7 MiB / 413.7 MiB)

There we go, three parallel dd operations as the files were being created. Upon seeing this I immediately noticed they’re each showing 100% progress on the file write, which seems inaccurate given their differing filesizes. That’s because the origin file for each of these dd operations was a character special file, /dev/urandom, and not an actual file. The special file does not have a constant length, so progress can only report the amount of data that’s already been transferred. You’ll find some other situations don’t have full filesizes reported by progress; the most common one I’ve encountered is downloading a file from the network.

You can see what this should look like a bit below, when we copy these files to a new location.

progress by default just reports the percentage and size of each file being copied, but it can also estimate a rate of data transfer with the -w/--wait flag. This will wait a set period of time to measure how much data was transferred in that interval. You can change the interval with the -W/--wait-delay argument, whose single parameter is the count of seconds in the interval. Here I’m trying the default wait time of 1, then increasing that to 5 seconds.

$ progress -w
[20293] dd /home/delucks/dev/cli.fan/File_1000000
        100.0% (829.1 MiB / 829.1 MiB) 6.9 MiB/s

[20297] dd /home/delucks/dev/cli.fan/File_2000000
        100.0% (823.4 MiB / 823.4 MiB) 6.9 MiB/s

[20301] dd /home/delucks/dev/cli.fan/File_10000000
        100.0% (816.7 MiB / 816.7 MiB) 5.9 MiB/s

$ progress -w -W 5
[20293] dd /home/delucks/dev/cli.fan/File_1000000
        100.0% (889.8 MiB / 889.8 MiB) 6.8 MiB/s

[20297] dd /home/delucks/dev/cli.fan/File_2000000
        100.0% (884.4 MiB / 884.4 MiB) 7.0 MiB/s

[20301] dd /home/delucks/dev/cli.fan/File_10000000
        100.0% (877.8 MiB / 877.8 MiB) 6.8 MiB/s

You can see all the detail that was missing from the dd commands in the following cp commands:

$ progress -w
[22791] cp /home/delucks/dev/cli.fan/File_2000000
        86.0% (1.6 GiB / 1.9 GiB) 18.1 MiB/s remaining 0:00:15

[22792] cp /home/delucks/dev/cli.fan/File_10000000
        17.1% (1.6 GiB / 9.5 GiB) 20.5 MiB/s remaining 0:06:34

$ progress -w -W 5
[22791] cp /home/delucks/dev/cli.fan/File_2000000
        98.3% (1.9 GiB / 1.9 GiB)

[22792] cp /home/delucks/dev/cli.fan/File_10000000
        25.7% (2.4 GiB / 9.5 GiB) 120.1 MiB/s remaining 0:01:00

Not only does progress calulculate rate, it also calculates ETA when the full size of the origin file is known.

By default, progress only reports the disk write operations being done by coreutils programs- cp, mv, dd, tar, cat, etc. The full list is available in the source code here. Thankfully, options are provided to extend the list of programs monitored by progress.

To illustrate this, let’s set up a download using curl that we want to monitor. I’ll be using the same large files which were created in the script above.

In one terminal window,

$ python3 -m http.server
Serving HTTP on 0.0.0.0 port 8000 (http://0.0.0.0:8000/) ...

and in another,

$ cd /tmp/progress_test
$ curl --output somefile http://localhost:8080/File_1000000

While this curl command is running, we can observe the file write operation happening by watching the curl process name with -c:

$ progress -c curl
[ 2418] curl /tmp/progress_test/somefile
        100.0% (557.0 MiB / 557.0 MiB)

The -c/--command flag tells progress to only watch for that command name when scanning /proc for commands that are copying data. This essentially replaces the default list of commands referenced above. If you’d instead like to include other commands while still scanning for all the default commands, pass -a/--additional-command, which can be given multiple times.

If you want to bypass this “scanning through /proc” behavior, you can directly pass process IDs with the -p/--pid argument.

progress provides some built-in watch functionality with the -m/--monitor flag, which is similar in behavior to watch progress, except that it only loops for the lifetime of the processes initially found in the first search. Closely related is the -M/--monitor-continuously flag, which loops endlessly picking up new processes on each iteration. Here’s a recording of using the -M flag with the curl examples above:

progress monitoring example

That’s all the functionality progress has! It’s tightly focused and useful at what it does. Unfortunately, it seems the open-source project is getting less active- some PRs haven’t been responded to in nearly a year.

tl;dr

progress fills a niche. Being able to see the progress of operations that typically occur with no user-facing signs of progress is a major step forward for operators that aren’t already used to digging through /proc. The ability to monitor arbitrary programs including web browsers makes progress a useful tool to keep around.

ℹ️ Interested in seeing more posts like this? Subscribe to the cli.fan RSS feed linked here and in the top navigation bar.