With all the wonderful graphical interfaces available, it’s easy to forget that a terminal is still a powerful tool. And depending on your job or what you’re doing, it can be the only interface available. Or you may just be longer in the tooth like me and have been using terminal emulators or actual real-life terminals for decades, and feel comfortable and at home working within a terminal.

Note

I will use the term “terminal” from here on out, even though for almost everyone we’re really using terminal emulators.

But even if you’re used to using terminals, you may not have heard of some of the newer tools available, which can seriously make you much more productive. So whether you’re relatively new to terminals or a seasoned pro, here are tools for the terminal that have made me much more productive. Some of these are replacements for tools you may be familiar with, and others are new(ish) and make one wonder how we got along without them.

While terminal emulators themselves are important “terminal” applications in and of themselves, I’m not going to cover them here. Perhaps in a future post I will discuss terminal emulators, but for now, I’m going to focus on tools that run within the terminals. Similarly I’m not going to cover shells, since I feel that choice of shell is someone like a choice of IDE - it can be a very religious discussion. Nor will I cover editors, since that could be a blog in itself. Finally, one terminal tool that deserves mention, though I won’t do more than mention it here, is tmux. I think it is indispensable, even with man terminal emulators providing similar features.

So with that out of the way, I’ve separated the tools into two categories:

  • CLIs: tools that (for the most part) “just” output text to the terminal
  • TUIs: tools that “take over” and render a user interface within the terminal

As you’ll see some of the tools in the CLI set blur the lines between these two categories. I’ll calls those out when I describe them.

Info

Interestingly, a lot of these tools are written in Rust! Terminal applications are one of the sweet spots for Rust.

CLIs

eza

The eza tool is a replacement for ls. Like many of the tools in this list, eza is aware of the modern terminal ecosystem. It supports themes for the terminal output, is aware of Git, showing Git file status in the output, and can be configured to show file sizes in friendlier formats, among other things. For example,

$ eza --git
drwxr-xr-x    - ray 26 Jan 10:07 -I build
.rw-r--r--  274 ray 25 Jan 21:12 -M CMakeLists.txt
.rw------- 451k ray 25 Jan 21:34 -N core.3670938
.rw-r--r--   12 ray 25 Jan 11:08 -- in
.rw-r--r-- 2.0k ray 20 Jan 11:15 -- README.md
drwxr-xr-x    - ray 26 Jan 10:18 NM src

The next-to-last column displays the Git status of the file/directory. Note that the build directory has an I indicating it is in the .gitignore file. To have eza respect the .gitignore file, you can use the --git-ignore option.

Another “cool” and modern if somewhat frivolous feature is the ability to show icons in the output listing.

$ eza --icons=always -1
 build
 CMakeLists.txt
 codecrafters.yml
 core.3670938
 in
󰂺 README.md
 src

Many of the options to eza match ls options, such as -l for a long listing, or -a to show hidden files. But not all the options align. For example, to sort a listing by timestamp, you would use the -s option with an argument of ‘new’ or ‘old’, not -tr or -t respectively, which is arguably more intuitive, but it is a difference.

As with a lot of the “replacement” commands in this list, I alias eza to ls:

alias ls='eza --git'

with the --git option to show Git status.

fd

fd is a replacement for find. It’s faster and arguably more powerful. If you’re used to find and its command-line options, you may have to adjust a bit, but it’s fundamentally the same - searching for files with various characteristics. Common use cases are much easier, and it is aware of things like .gitignore files, excluding files in .gitignore by default. You can also define a config file ~/.fdignore for fd to specify excludes that you want to always apply, such as

$ echo "/mnt/foo\n*.bak" > .fdignore
$ cat ~/.fdignore
/mnt/foo
*.bak

Unlike find, to find files you just specify the pattern you’re looking for,

fd foo

which will look for files or directories with foo in the name, from the current directory. By default it is case-insensitive, unlike find. You can look for files with a given extension using the -e option. To find all Markdown files in a given directory, you would use

fd -e md

Some things may take a little adjustment when you’re used to find. For example, expanding on the previous example, if you want to find all Markdown files in a specific directory, you would use

$ fd . -e md SomeDirectory
# or $ fd -e md . SomeDirectory

In this case the . is not specifying the current directory but rather the pattern to match, within the directory SomeDirectory.

This just scratches the surface of what fd can do. Look at the GitHub page for details and a demo.

rg

The next new and powerful replacement tool is rg or ripgrep. This is a fast and powerful grep replacement. Somewhat similar to the improvements that fd made in how to invoke the command, you don’t need to specify options for common things. By default, rg searches recursively for all files for the current directory, case-insensitively. It will also respect .gitignore and hidden files and directories, ignoring them by default.

To search for a pattern therefore is as simple as

$ rg main
<snip>
 
arch/mips/include/asm/octeon/cvmx-bootmem.h
52: * which is used to maintain the free memory list.  Since the bootloader is
 
crypto/nhpoly1305.c
9: * "NHPoly1305" is the main component of Adiantum hashing.
99:             if (state->nh_remaining == 0) {
103:                    state->nh_remaining = NH_MESSAGE_BYTES - bytes;
110:                    pos = NH_MESSAGE_BYTES - state->nh_remaining;
111:                    bytes = min(srclen, state->nh_remaining);
116:                    state->nh_remaining -= bytes;
118:            if (state->nh_remaining == 0)
150:    state->nh_remaining = 0;
210:    if (state->nh_remaining)
 
arch/mips/kernel/rtlx-mt.c
3: * License.  See the file "COPYING" in the main directory of this archive
<snip>

rg supports different character sets and can search in many compressed file types. There is also a configuration file ~/.ripgreprc to customize its behavior.

bat

bat is a replacement for cat that adds syntax-aware display of files, line numbers, themes, paging and more. You can of course disable these options as desired, or set defaults. For example, setting a theme can be done using the BAT_THEME environment variable. You can use a predefined theme, or create your own. View available themes with bat --list-themes which will show a snippet of code using each theme. You can also create your own.

Personally, I use the GitHub theme and I specify less as my pager. I also alias bat to cat:

alias cat='bat'
export BAT_PAGER='ls -F'
export BAT_THEME=GitHub

An example of bat output (not with the GitHub theme):

$ cat .zshrc
───────┬────────────────────────────────────────────────────────────────────────
 File: .zshrc
───────┼────────────────────────────────────────────────────────────────────────
   1 # Add deno completions to search path
   2 if [[ ":$FPATH:" != *":$HOME/.zsh/completions:"* ]]; then export FPATH=
 "$HOME/.zsh/completions:$FPATH"; fi
   3
   4 # If you come from bash you might have to change your $PATH.
   5 # export PATH=$HOME/bin:/usr/local/bin:$PATH
   6 if command -v brew &>/dev/null; then
   7   eval "$(brew shellenv)"
   8 elif [ -e /home/linuxbrew/.linuxbrew/bin/brew ]; then
   9   eval "$(/home/linuxbrew/.linuxbrew/bin/brew shellenv)"
  10 else
  11   echo "can't find brew!"
  12 fi
  13
  14 ZSH_WEB_SEARCH_ENGINES=(
  15   rdoc "https://docs.rs/"
  16   crates "https://crates.io/search/?q="
  17 )
  18 #
  19 # Path to your oh-my-zsh installation.
  20 export ZSH="$HOME/.oh-my-zsh"

fzf

One of the coolest command line tools is fzf, a fuzzy finder with a UI. This is one of those tools that blurs the lines between a CLI and a TUI. It provides a UI picker for files, directories, and command output. It can display a “preview” of the currently selected item in the picker. It integrates nicely with your shell, and is often embedded in other tools (e.g. nvim).

Note

There is a related project fzf-git that provides shell integration allowing use of fzf with git.

fzf works well integrated with other tools. For example, you can use it with fd and your shell to have fuzzy command completion. One thing you can do is combine it with ls (or eza) to fuzzy find files or directories, with a preview:

ls-and-fzf

Or you can combine it with ssh and have it list hosts from your /etc/hosts file, and showing a preview with dig:

ssh-and-fzf

Here is my zsh configuration for fzf enabling the above functionality:

source <(fzf --zsh)
 
# -- Use fd instead of fzf --
 
export FZF_DEFAULT_COMMAND="fd --hidden --strip-cwd-prefix --exclude .git --exclude .obsidian"
export FZF_CTRL_T_COMMAND="$FZF_DEFAULT_COMMAND"
export FZF_ALT_C_COMMAND="fd --type=d --hidden --strip-cwd-prefix --exclude .git"
 
# Use fd (https://github.com/sharkdp/fd) for listing path candidates.
# - The first argument to the function ($1) is the base path to start traversal
# - See the source code (completion.{bash,zsh}) for the details.
_fzf_compgen_path() {
  fd --hidden --exclude .git . "$1"
}
 
# Use fd to generate the list for directory completion
_fzf_compgen_dir() {
  fd --type=d --hidden --exclude .git . "$1"
}
 
if [ -e ~/.fzf-git/fzf-git.sh ]; then
  source ~/.fzf-git/fzf-git.sh
fi
 
show_file_or_dir_preview="if [ -d {} ]; then eza --tree --color=always {} | head -200; else bat -n --color=always --line-range :500 {}; fi"
 
export FZF_CTRL_T_OPTS="--preview '$show_file_or_dir_preview'"
export FZF_ALT_C_OPTS="--preview 'eza --tree --color=always {} | head -200'"
 
# Advanced customization of fzf options via _fzf_comprun function
# - The first argument to the function is the name of the command.
# - You should make sure to pass the rest of the arguments to fzf.
_fzf_comprun() {
  local command=$1
  shift
 
  case "$command" in
    export|unset) fzf --preview "eval 'echo \${}'"          "$@" ;;
    ssh)          fzf --preview 'dig {}'                    "$@" ;;
    *)            fzf --preview "$show_file_or_dir_preview" "$@" ;;
  esac
}

There is so much more you can do! I’m just touching the surface. Check the documentation. There are sites with various examples, but the fzf docs have a lot of community-provided examples.

atuin

Another tool that blurs the line between CLI and TUI is atuin, a command history fuzzy finder + database, that can also sync across your systems. All your commands are saved to a database, and you can search through that history to find what you’re looking for. How many times have you typed something like

history | grep kubectl

or some such hoping to find something you did some time in the past, but hopefully not so long ago that it’s aged out of your history file? And even if you did find what you’re looking for, you then have to copy/paste what you found. With autin it’s all from your terminal prompt, with fuzzy matching. Following the search for kubectl example, and with a shell key mapping to launch atuin with my up arrow key, I can do $ kube<up-arrow> and see:

autin-and-kube

Note that “kube” is shown at the bottom, and if I keep typing, the list will narrow down to what I’m looking for. I can then select the command I want, either executing it with <enter> or giving myself the opportunity to edit it with <tab>.

There’s even a new “introspect” feature providing info on stats of the selected command. I mean, it’s a database after all.

As I mentioned, you can also have this database sync across machines, but that’s optional. It does require a free account on the atuin server.

Finally, after first installing atuin you can import your current command history into the autin database with atuin import.

To set up atuin you just add this to your shell configuration, e.g. .zshrc:

# atuin configuration
if [ -e $HOME/.atuin/bin/env ]; then
  . "$HOME/.atuin/bin/env"
fi
eval "$(atuin init zsh)"

zoxide

A nifty tool that helps you navigate your filesystem is zoxide, an alternative to cd. It uses a database to keep track of the directories you change to, and then allows you to jump to those directories with a fuzzy search.

The command is z but I alias cd to z. You can then “cd” to some directory from anywhere in your filesystem by providing just part of the directory name.

Zoxide depends on fzf for the fuzzy search, so you need to have fzf installed. You can see fuzzy matches of your target directory, and then select the one you want e.g. $ z rust<sp><tab>:

zoxide

Shell integration is provided. For my zsh configuration I have:

eval "$(zoxide init zsh)"
alias cd=z

dust

The dust tool can replace du. It provides a faster and graphical view of disk usage.

dust

curl replacements

Curl has been around forever, and is a great tool for making HTTP requests. But it’s not necessarily the most user-friendly tool. There are various tools that try and address that. Two I want to call out are curlie and httpie.

httpie

The httpie tool is a user-friendly replacement for curl. It doesn’t provide all the options that curl does, but it’s easier to use and is sort of taking the 80/20 approach, providing what people use most and doing it with a simpler interface than curl. The command itself is called http.

The best way to describe it is to show some examples.

  • a simple GET request - note the pretty-printed JSON by default
$ http httpbin.org/stream/1
HTTP/1.1 200 OK
Access-Control-Allow-Credentials: true
Access-Control-Allow-Origin: *
Connection: keep-alive
Content-Type: application/json
Date: Fri, 07 Feb 2025 18:56:56 GMT
Server: gunicorn/19.9.0
Transfer-Encoding: chunked
 
{
    "args": {},
    "headers": {
        "Accept": "*/*",
        "Accept-Encoding": "gzip, deflate",
        "Host": "httpbin.org",
        "User-Agent": "HTTPie/3.2.4",
        "X-Amzn-Trace-Id": "Root=1-67a65778-119ef1ea5fcfc82a60ef63ef"
    },
    "id": 0,
    "origin": "193.19.109.180",
    "url": "http://httpbin.org/stream/1"
}
  • passing a custom header - just use a name:value pair (-v to show the request headers, which by default are omitted)
$ http -v httpbin.org/status/200 X-Custom:1234
GET /status/200 HTTP/1.1
Accept: */*
Accept-Encoding: gzip, deflate
Connection: keep-alive
Host: httpbin.org
User-Agent: HTTPie/3.2.4
X-Custom: 1234
 
<snip>
  • posting data - add name=value pairs; hierarchical data is supported as well
$ http -v POST httpbin.org/status/201 field1=data field2=moredata field3="even more data"
POST /status/201 HTTP/1.1
Accept: application/json, */*;q=0.5
Accept-Encoding: gzip, deflate
Connection: keep-alive
Content-Length: 68
Content-Type: application/json
Host: httpbin.org
User-Agent: HTTPie/3.2.4
 
{
    "field1": "data",
    "field2": "moredata",
    "field3": "even more data"
}
 
<snip>
  • passing a query string - use name==value pairs, and no need to pass (and escape) & characters
$ http -v httpbin.org/status/200 q==foo bar==baz
GET /status/200?q=foo&bar=baz HTTP/1.1
Accept: */*
Accept-Encoding: gzip, deflate
Connection: keep-alive
Host: httpbin.org
User-Agent: HTTPie/3.2.4
 
<snip>

Obviously, there’s a lot more you can do with httpie. Check out the examples documentation.

curlie

curlie is supposed to be somewhere between curl and httpie, providing all of curl’s features but easier to use, with more sensible defaults. You can alias curlie to curl and not notice much difference, except hopefully increased performance. Like httpie you can pass headers and data via the command line.

tldr

Think of tldr as the TL;DR for a man page. It has a database of commands that has documentation for common usage of the particular command. For example, here is the TL;DR for tldr itself:

$ tldr tldr
 
tldr
 
Display simple help pages for command-line tools from the tldr-pages project.
Note: the `--language` and `--list` options are not required by the client specification, but most clients implement them.
More information: <https://github.com/tldr-pages/tldr/blob/main/CLIENT-SPECIFICATION.md#command-line-interface>.
 
- Print the tldr page for a specific command (hint: this is how you got here!):
    tldr command
 
- Print the tldr page for a specific subcommand:
    tldr command subcommand
 
- Print the tldr page for a command in the given [L]anguage (if available, otherwise fall back to English):
    tldr --language language_code command
 
- Print the tldr page for a command from a specific [p]latform:
    tldr --platform android|common|freebsd|linux|osx|netbsd|openbsd|sunos|windows command
 
- [u]pdate the local cache of tldr pages:
    tldr --update
 
- [l]ist all pages for the current platform and `common`:
    tldr --list
 
- [l]ist all available subcommand pages for a command:
    tldr --list | grep command | column
 
- Print the tldr page for a random command:
    tldr --list | shuf -n1 | xargs tldr

The last example in its own TL;DR is fun, giving the TL;DR for a random command.

TUIs

Now we’re getting into some full-screen terminal applications. You can really see the power of the terminal with these tools.

top replacements

The top command is a top (pun intended) command for monitoring system resources that has been around forever. It does it’s job, but there are more modern tools with theming and graphs that can provide more information in a more readable way. Here are a few. I’m currently using bpytop but I’ve tried them all.

  • bpytop

bpytop

  • btm

btm

  • gtop

gtop

  • htop

htop

lazygit

lazygit is a TUI for git. It’s pretty full-featured. If you love the terminal and don’t want to use a GUI, then lazygit is really the way to go.

lazygit

lazygit-keybindings

You can also use lazygit from within editors like neovim:

lazygit-in-neovim

dive

The last but not least tool for this blog is called dive. dive is a tool for exploring a Docker image. Using a graphical interface, you can view stats about an image, view each layer, navigate the filesystem tree, see what was added each layer, and more. If you’re trying to understand and optimize your image, this is a great tool. And while it can be used interactively as a TUI, it can also be used headless and integrated into your CI/CD pipeline.

dive

Conclusion

I hope you liked this tour of what I consider some of the best terminal tools. If you have additional ones you like, or ones you think are superior to the ones I list here, for the given purpose, please let me know. I’m always looking for ways to improve my workflow.