Eric Frederickson

E

pfcalc

prefix calculator



Introduction

pfcalc is a simple command-line prefix calculator written in Haskell. It provides a concise syntax for integer arithmetic, and is intended for situations where some quick math over a list of numbers is needed.


# Calculate a sum
pfcalc + 121 242 363
# 726

# Calculate a remainder
pfcalc mod 894 16
# 14

# Find the number of seconds in a year
pfcalc x 365 24 60 60
# 31536000

# Find the gcd of some piped-in arguments
echo "91 455 56" | xargs pfcalc gcd
# 7

# Add up numbers from a file
seq 1 100 > numbers_1_through_100.txt
pfcalc + $(cat numbers_1_through_100.txt)
# 5050

Implementation

pfcalc is written in the wonderful Haskell programming language, and is hence written in a functional style, which enabled the code to be appropriately concise. Find the source code here.

Operations

pfcalc supports the following operations:


+ (sum)
- (subtraction)
x (product)
/ (integer division)
% (modulus)
mod (alias for "%")
gcd (greatest common divisor)

(Note that the symbol x is used for product instead of *, because shells read * as a wildcard.)

These operations are implemented in src/Ops.hs as functions of type Op,


type Op = NonEmpty Integer -> Integer

which describes a function from a non-empty list of integers to a single integer.

Implementing operations like + and x is easily accomplished by folding the input list over the corresponding Haskell built-in operation.


opSum :: Op =
  foldl' (+) 0

opProduct :: Op =
  foldl' (*) 1

(Note that 0 and 1 are used in the fold calls because they’re the identity elements for addition and multiplication respectively.)

Implementing gcd is easily done via folding the list over the Euclidean Algorithm:


opGCD :: Op =
  foldl' euc 0
  where
    euc a b =
 -- euclidean algorithm
      if b == 0 then a
      else euc b (a `mod` b)

Implementing subtraction-chains presents an issue: Haskell’s lazy evaluation means that the math will only happen after the fold has created the entire expression; evaluation will then happen outside-in, so subtractions will cancel into additions. We can get around this problem by hand-rigging the fold, using the $! operator to force strict-evaluation on the arguments to the recursive call.


opSubtract :: Op =
  \nums ->
    case nums of
      a :| [] -> a
      a :| b : cs ->
        opSubtract $! ((a - b) :| cs)
        -- need strict eval with "$!" so that subtractions don't cancel into
        -- additions

Main

The rest of the program is just the main function, which handles reading in arguments from the command line, checking for invalid input, and passing the arguments along to the proper op function if everything checks out.


main :: IO () =
  do args <- getArgs
     case args of
       [] -> do putStrLn "Error: no arguments given"
                putStrLn opsDescription
       op : opArgs ->
         let numericArgs :: [Integer] =
               mapMaybe (readMaybe) (opArgs)
         in case numericArgs of
           [] -> opError op "cannot be applied to no arguments"
           arg1 : argRest ->
             let args = arg1 :| argRest
                 opDo opFn = print (opFn args)
             in case op of
               "+" -> opDo opSum
               "-" -> opDo opSubtract
			   -- ...
               "gcd" -> opDo opGCD
               _ -> do opError op "unknown"
                       putStrLn opsDescription

(Note that the pattern-match on numericArgs enabled us to branch into a case where we are certain that the list has at least one element, which enabled the use of the :| constructor to create the NonEmpty list needed by the op functions).

pfcalc handles non-numeric input tokens by simply discarding them, making it a useful tool for calculating over blocks of text which contain numbers.


echo "I wish I knew the gcd of 120, 330, and 540" \
| sed s/\,/\ /g \
| xargs pfcalc gcd
# 30

Installation

A Makefile is included in the pfcalc source repository, making compilation easy (as long as you have the ghc Haskell compiler).


git clone https://github.com/emfred-dot-com/pfcalc
cd pfcalc
make
sudo make install