Biff is a command line datetime Swiss army knife

12 hours ago (github.com)

Respect for programming this. I did some date/time calculations a few years ago using Perl and it was full of corner cases and trouble. Did I enjoy it? I enjoyed seeing it work. Hopefully with the right answers! This tool looks great.

I'm the author of Biff. I just wanted to share a really cool example of something that Biff can do that I _think_ is kinda hard to do otherwise. (And also, I want to make an assertion about it and I hope this will lead to me being wrong and learning something new.)

The use case is: "I want to see a list of all files in a repository, sorted in ascending order of when it was most recently changed according to source control. I also want to highlight the time with color, make it be in local time and format it in my own bespoke way using strftime." Here's the full command (run from the root of https://github.com/BurntSushi/ripgrep):

    $ git ls-files |
        biff tag exec git log -n1 --format='%aI' |
        biff time in system |
        biff time sort |
        biff time fmt -f '%a %Y-%m-%d %H:%M:%S' |
        biff untag -f '{tag}|t{data}'
    ...
    Thu 2025-10-30 13:30:14 crates/ignore/Cargo.toml
    Sat 2025-11-29 14:11:38 crates/core/flags/lowargs.rs
    Wed 2025-12-17 11:38:12 tests/misc.rs
    Wed 2025-12-17 11:38:12 tests/util.rs
    Thu 2026-02-12 20:39:46 crates/ignore/src/default_types.rs
    Fri 2026-02-20 16:06:29 crates/core/flags/config.rs
    Fri 2026-02-27 11:25:19 GUIDE.md
    Fri 2026-02-27 11:25:19 crates/core/flags/defs.rs
    Mon 2026-05-25 23:56:53 CONTRIBUTING.md
    Tue 2026-05-26 08:32:43 AI_POLICY.md

Or even ask for a specific time window:

    $ git ls-files |
        biff tag exec git log -n1 --format='%aI' |
        biff time in system |
        biff time cmp ge 2026-01-01 |
        biff time cmp lt 2026-04-01 |
        biff time sort |
        biff time fmt -f '%a %Y-%m-%d %H:%M:%S' |
        biff untag -f '{tag}|t{data}'
    Thu 2026-02-12 20:39:46 crates/ignore/src/default_types.rs
    Fri 2026-02-20 16:06:29 crates/core/flags/config.rs
    Fri 2026-02-27 11:25:19 GUIDE.md
    Fri 2026-02-27 11:25:19 crates/core/flags/defs.rs

If you run this on a big repository, it will take quite a lot of time because `git log -n1` takes a long time. I think this is the fastest way to get the most recent commit time on a single file? (That's the assertion that I hope someone can correct me on!) In any case, `biff tag exec` is using parallelism under the hood to make this even faster.

  • This is pretty neat. My proficiency with the command line is woefully underdeveloped and seeing examples like this help me see the possibilities.

  • Quick Note: You can put the pipe operator where your backslash is and you won’t have to escape the newline character. Works in bash, zsh and ksh (what I’ve tested).

No, Biff informs the system whether you want to be notified when mail arrives during the current terminal session.

  • I.e.,

        NAME
           biff -- be notified if mail arrives and who it is from
        
        […]
        
        HISTORY
           The biff command appeared in 4.0BSD.  It was named  after  the  dog  of
           Heidi Stettner. He died in August 1993, at 15.
    

    * https://man.freebsd.org/cgi/man.cgi?query=biff

        Eric Cooper, a student contemporary to Foderero and 
        Stettner, reports that the dog would bark at the mail 
        carrier,[4][5] making it a natural choice for the name 
        of a mail notification system. Stettner herself 
        contradicts this.[3][6]
    

    * https://en.wikipedia.org/wiki/Biff_(Unix)

    • From the excellent "A Quarter Century of UNIX" (by the late Peter H. Salus):

      Heidi would bring her dog with her to class and to her office. He was a very friendly dog, and a lot of the students enjoyed throwing a ball for him down the corridor to fetch. He even had his picture on the bulletin board with the graduate students: the legend read that he was working on his Ph.Dog. John decided to name the program after the dog: Biff. According to Heidi, John and Bill Joy then spent a lot of time trying to compose an explanation for biff - they came up with "Be notified if mail arrived." Biff, who died in August 1993, at 15, once got a B in a compiler class. According to Heidi, the story of Biff barking at the mailman is a scurrilous canard.

      One of my favourite bits of trivia from that excellent book, but hardly anyone I bump into these days knows anything about that kind of multi-user Unix experience/environment these days. I barely caught any of it myself.

  • Yeah the name collision is unfortunate, but probably fine. The name Biff was just too good to pass up.

    The name comes from the fact that Biff is a character in Back to the Future, and it rhymes with Jiff[1]. Jiff is the datetime library that Biff uses.

    "Make like a tree and get out of here!" https://www.youtube.com/shorts/9Jabplo2pZU

    [1]: https://github.com/BurntSushi/jiff

    • > Yeah the name collision is unfortunate, but probably fine.

      collisions, lol

        % apt-cache search biff
        biff - a mail notification tool
        gnubiff - mail notification program for GNOME (and others)
        wmbiff - Dockable app that displays information about mailboxes
        xlbiff - mail notification pop-up with configurable message scans
      

      (along with 9 more matches without biff in command name)

    • Griff is still available for future projects or Buford if you create a throwback project.

  • All short names, that is, pronounceable strings of 4 or maybe even 5 letters are already taken. Some of them taken many times over.

    I think fewer people now care about mail notifications in a terminal session than about wrangling datetimes on the command line.

  • Yes I'm sure root is anxious to read all the mail in their local mailbox

    • Sending mail to root@<whatever> really did use to be a pretty reliable way of getting somebody useful's attention - the early-to-mid 90s equivalent of making a "Can someone from Google please unlock my account?" post on HN.

      1 reply →

been hand-rolling date -d plus a wrapper script for newsletter scheduling for years. Stuff like "next second Tuesday after a holiday" and "convert these timestamps to a reader's local time before send". biff time seq monthly -w 2-tue would have replaced about 40 lines of bash for me.

I remember when biff was what we ran in a CSH to be informed of new email. I don’t remember if this was a local UCB tool or if it was part of BSD.

The thing Biff gets right that gnu `date` and most stdlib datetime APIs get wrong: it treats "civil time" and "absolute instants" as different types. You cannot answer "what's 30 days from 2024-03-08 in America/New_York" without picking a side — DST means that's either 29d23h or 30d0h of elapsed time, and most APIs silently pick one without telling you.

Jiff (the underlying Rust crate) gets this from Temporal in TC39, which is the first time JS standards have led anything datetime-shaped. Hopefully the rest of the ecosystem catches up — Python's `zoneinfo` only landed in 3.9 and `datetime.timezone` still has sharp edges.

% biff

2026 M05 28, Thu 17:27:46

Ahh, the month of M05

  • This is a fair critique actually. And this shouldn't be the default. It is for now because I haven't gotten around to making Biff read your POSIX locale settings and converting them to a Unicode locale. If you build with `cargo build --release --features locale` (or get Biff from a release binary), then you can do:

        $ BIFF_LOCALE=en-US biff
        Thu, May 28, 2026, 6:38:09 AM EDT
    

    If that doesn't work, then you can enable logging to see an error message:

        $ BIFF_LOCALE=watwat BIFF_LOG=warn biff
        2026-05-28T06:39:08.876336708-04:00[America/New_York]|WARN|src/main.rs:76: reading `BIFF_LOCALE` failed, using unknown locale `und`: failed to parse `BIFF_LOCALE` environment variable: The given language subtag is invalid
        2026 M05 28, Thu 06:39:08
    

    What you're seeing is what ICU4X does when the user's locale is unknown or undetermined. The `M` prefix occurs to indicate that the number is the month, and is unrelated to the name. For example:

        $ BIFF_LOCALE=watwat biff time fmt -f '%c' '1 month'
        2026 M06 28, Sun 06:39:50