Random zsh tips

My friend fREW and I both use zsh, and we've picked up a number of tricks over the years. We decided to collaborate on a post sharing some assorted tips that can really improve the zsh experience!

Noclobber

This is a small thing, but it's saved me from many a headache. I (Rob) will often be working on some data, and I'll squirrel away a long-running or expensive computation to a temporary file, like so:

$ long-running-or-expensive-command > /tmp/data

All's well and good, until I search through history too far, or until I mistype a search that yields the above command line, and I overzealously hit enter - whoops, there goes that data! This is especially frustrating if the data you just clobbered changes over time, and there was a case in the old data that isn't present in the new! Or maybe you saved the data, got on a plane, and wanted to work with it.

Well, if you set noclobber in your zshrc, zsh will prevent you from making this mistake - it will refuse to run commands that would destroy a file via shell redirection (something like curl -Lo my-data.txt http://example.com/data.txt will still clobber my-data.txt). You can override this via the >| operator - personally, I prefer to explicitly rm the destination file and then re-run the command.

On a side note - wouldn't it be cool if the shell were smart enough to only clobber the destination file in the case of success, or if the shell somehow knew that long-running-or-expensive-command could result in different results? Or maybe it would be neat to have a facility to keep a history of temporary files as you work with them!

Vi Mode

Like noclobber, another little tweak that changed how I use zsh: bindkey -v to enable Vi mode! I was skeptical about this until I tried it, but if you're a Vim user, I recommend you at least try it out and see how you like it - I feel like it's made me so much more efficient at editing command lines over the years!

Widgets & Keybindings

zsh provides a feature called "widgets", which are really just functions that have a special relationship with the line editor. They're used to implement all of the various keybindings - for example, the default keymap assigns (most) keys to the self-insert widget, which inserts the character corresponding to the key you just typed into the line editor.

You can view the current keybindings via bindkey -L (if you're using vi mode, you might want to check out bindkey -L -M vicmd as well to see what widgets are bound in command mode) - this can reveal some interesting key combinations you didn't know about before! There are plenty of widgets that aren't bound to anything by default - you can find a bunch of interesting widgets in man zshzle and man zshcontrib.

It's also pretty easy to write your own - for example, I will sometimes accidentally type !4 instead of !$, which expands to the fourth entry in my history. This is never what I want, so I wrote a little widget to detect this situation and fix it to expand !$ instead.

Tips by fREW

Here are some random, but cool, features that I (fREW) really enjoy:

push-line

I bound the q key in the vicmd mode to push-line like this:

bindkey -M vicmd "q" push-line

What this does is set aside the current commandline, give you a fresh commandline to modify, and then pop the current line back. I typically use it in situations like this:

$ vi some/dir/some/file<ESC>q
$ mkdir -p some/dir/some/
$ vi some/dir/some/file

AUTO_PUSHD

In both zsh and bash you can use pushd some/dir to change directory into some/dir, and then if you use popd you'll change directory back to where you came from. If you set the AUTO_PUSHD option, cd will automatically pushd for you. I'll then use popd (or my alias for it, the symettrical mk) to get back to where I started.

global aliases

In zsh you can have aliases that are allowed anywhere in a commandline. I use G and L constantly below, and always forget that I've defined V even though I do that action manually a lot:

alias -g G="| grep"
alias -g L="| $PAGER"
alias -g V="| vim -"

Example:

$ long-out-is-long G needle # finds lines matching needle
$ curl https://blog.afoolishmanifest.com/ L # runs with pager

You can also pass arguments! This works fine:

$ ls G -P '^f'

History Events

This tip has been shared a lot over the years, but honestly it's one of my favorites, and I feel like it saves me so much time and typing! Let's say you're on the command line, and you issue the following commands:

$ mkdir a-really-long-directory-name
$ cd a-really-long-directory-name

It's no fun having to type that long directory name twice, is it? Fortunately, you can use history expansion (see "History Expansion" in man zshexpn for more details) to save yourself some time here:

We hope you found something useful that you're itching to try out here! If you're interested in checking out our zsh configs, here they are:

Many thanks to fREW for contributing to this post!

Published on 2020-10-24