Eric Frederickson

E

hms

HH:MM:SS durations calculator

Version 0.1.4.1



“Rain might wash you away
Yet time already does”

-- Protesilaos Stavrou, “Tempests

Introduction

hms is a command-line tool for doing math with time durations in the standard HH:MM:SS format (hours, minutes, seconds). It supports convenient syntax for writing these forms, so instead of typing out the full 00:00:05 to specify a duration of 5 seconds, you can simply write 5, or 0:5, or 00:05, etc.


# 20 seconds + 40 seconds = 1 minute
hms "20 + 40"
# 00:01:00

# 1 minute - 2 seconds = 58 seconds
hms "1:00 - 2"
# 00:00:58

# 2 minutes and 20 seconds + 4 minutes and 40 seconds
# = 7 minutes
hms "2:20 + 4:40"
# 00:07:00

# 1 hour + 2 minutes + 3 seconds
hms "1:0:0 + 0:2:0 + 0:0:3"
# 01:02:03

# equivalently:
hms "1:0:0 + 2:0 + 3"
# 01:02:03

# equivalently:
hms "01:00:00 + 00:02:00 + 00:00:03"
# 01:02:03

Features

⏲ Operations

As of version 0.1.4.1, hms currently supports the following operations:

+ -- addition, left-associative

- -- subtraction, left-associative

* -- multiplication, left-associative

Multiplication makes the most sense when thinking of one of the operands as a scalar instead of a duration, as it then becomes a shorthand for summing a duration with itself a given number of times:


# 1 hour * 2 = (1 hour + 1 hour) = 2 hours
hms "1:00:00 * 2"
# 02:00:00

# 1:30 * 3 = (1:30 + 1:30 + 1:30) = 4:30
hms "1:30 * 3"
# 00:04:30

Under the hood, an expression like "1:30 * 3" actually becomes 00:01:30 * 00:00:03, so multiplication is always between durations and durations, not durations and pure scalars. The logic behind this is that a duration can be seen simply as a number of seconds, so 1:30 * 3 is really 90 * 3 (because 1:30 is equivalent to 0:90), so we then do integer multiplication in the standard way, i.e. 90 * 3 = 270, and then 270, which we’re viewing as a number of seconds, is written out to the user as 4:30 (by having the seconds overflow to minutes when it’s >= 60).

This system means that you can do calculations like this:


# 1 minute * 1 minute = 1 minute * 60
# = 60 minutes = 1 hour
hms "1:00 * 1:00"
# 01:00:00

which don’t make much sense semantically, but may be useful in some situation that I can’t think of.

⏲ Nested expressions

hms supports nested expressions with parentheses:


hms "0:10 - (0:05 + 0:01)"
# 00:00:04

# without parentheses, the result differs:
hms "0:10 - 0:05 + 0:01"
# 00:00:06

⏲ Descriptive failures

If hms can’t parse an expression you give it, it’ll tell you why:


hms "10,20,30"
# Parse Error: (line 1, column 3):
# unexpected ','
# expecting " ", "\n", "\t", "+", "-", "(" or end of input

hms "00:10 plus 00:20"
# Parse Error: (line 1, column 7):
# unexpected 'p'
# expecting " ", "\n", "\t", "+", "-", "(" or end of input

If hms can parse, but can’t evaluate, the expression you give it, it’ll tell you why the evaluation failed:


hms "+ 01:00"
# Evaluation Error:
# Infix operator '+' cannot appear at the beginning of an expression.

hms "01:00 +"
# Evaluation Error:
# Infix operator '+' missing second operand.

hms "01:00 02:00"
# Evaluation Error:
# Operator missing between durations '00:01:00' and '00:02:00'.

hms "01:00 + (02:00 + - 03:00)"
# Evaluation Error:
# While evaluating parenthesized expression '(00:02:00 + - 00:03:00)':
# Infix operator '+' cannot be applied to infix operator '-'.

Use cases

I wrote hms because I wanted add up the lengths of some audio files, but didn’t have an easy way to. Programs like mediainfo can give you the length of an audio file in the format that hms understands, but I couldn’t find a simple tool to do arithmetic on these values. A typical small script doesn’t suffice because, for instance, you need to handle overflows from seconds to minutes and minutes to hours.

Using hms, if you have a sequence of durations in a file like so:

~/lengths.txt

05:10 03:24 02:55 04:29

then you can use hms to easily add up the durations:


cat ~/lengths.txt | sed 's/ /+/g' | xargs hms
# 00:15:58

Installation

The source code for hms is hosted on my Github page at https://github.com/emfred-dot-com/hms. The project is written in the Haskell programming language and uses the cabal build system, making it possible to compile and install hms with a single command.

The following shell commands run through an installation:


git clone https://github.com/emfred-dot-com/hms
cd hms
cabal install
# ^ if you get a dependency-resolution error here,
# use ghcup to install ghc version 9.10.1, and then
# retry the command.

This will install the executable to ~/.cabal/bin/hms, so make sure to add ~/.cabal/bin to your shell’s PATH environment variable if you haven’t already.

(Note for developers: the hms source repository also has a test suite, which you can run using the cabal test command.)





Banner image: “Crossbeams”. As the creator of this photo I declare it licensed under CC BY-NC-SA 4.0.