Go program stores directory paths
Programming Snapshot – cdbm
When you change directories at the command line, you often find yourself jumping back and forth between known paths. With a utility written in Go, Mike Schilli records the jumps and shows the way back.
While younger coworkers tend to edit their programs with clever IDEs, I still find it most natural to jump to local Git repositories with a quick cd
at the command line and fire up Vim at files with the source code residing there. Typing in the directory path each time is a pain in the ass, and there are usually only half a dozen paths back and forth – so the command line should be able to remember that.
The C shell invented the pushd
and popd
commands many years ago, but wouldn't it be much more convenient to automatically record the directories you visit, store them in a database, and even offer search queries for previously visited directories based on criteria such as frequency or the timestamp of the last visit?
In this issue, a Go program by the name of cdbm
collects the paths accessed by the user during a shell session; the command line user just uses cd,
and some magic glue in the shell's configuration then taps into the $PS1
prompt generator. If the directory changes, cdbm
gets called and stores the new path in an SQLite database on the disk, which later allows search queries whose results can be accessed directly by the user for navigation help. Bash users can modify their .bashrc
file to enable this. On typing a newly introduced command c
, users will see a selection list with the last directories visited (Figure 1). After selecting one of them with the cursor keys and pressing Enter, the shell directly jumps there (Figure 2).
If the list of hits grows beyond the preconfigured set limit of five entries, the nifty terminal UI (as shown in Figure 1) displays a small down arrow, indicating that the user can move the cursor further down to reveal previously hidden entries. So how does this work?
Hitchhiking a Prompt Ride
Once the Bash shell has executed a command, it generates the line prompt so that the user knows that it is his turn again. Instead of boring $
or #
characters, experienced shell users often define individual prompts in the $PS1
variable; they can display the username, the hostname, and the current directory. For example, the following statement
export PS1='\h.\u:\W$ '
defines a prompt with the hostname (\h
), a separating dot, the username (\u
), a separating colon, the current directory, a dollar sign, and a space. On my machine in the git
directory, this comes up as:
mybox.mschilli:git$
Now the $PS1
prompt variable does not just support the placeholders used above, which it replaces with current values, but also commands to be executed, whose output it interpolates into the prompt string:
export PS1='$(cdbm -add)\h.\u:\W\$ '
This definition tells Bash to call the cdbm
program with the -add
option after every shell command executed. cdbm
is the Go program in Listing 1 [1] that determines the current directory in -add
mode and stores the path with the current timestamp in a table of an automatically created SQLite single file database. If the path already exists, cdbm
only refreshes the timestamp for the entry. While the user keeps changing directories with cd
, paths with timestamps accumulate in the database (Figure 3).
Listing 1
cdbm.go
Of course cdbm -add
does not output anything, but returns without comment after the work is done, so that the $PS1
prompt defined above remains the same, even if the Bash shell secretly called the directory butler while composing the prompt.
Here We Go
To compile Listing 1, the following command sequence generates a new Go module in the same directory where the build process happens later on:
go mod init cdbm go build
Listing 1 references a number of useful Go packages on GitHub, which the call to go build
automatically retrieves as source code, because of the previous module definition, and compiles as libraries before compiling Listing 1. The resulting cdbm
binary contains everything, including a driver for creating and querying SQLite databases.
Once you have copied the binary to a location where the shell can find it in the search $PATH
, you have to change two things in the .bashrc
bash profile in order to benefit from the new utility. First, add the $PS1
definition from above and second, define a Bash function c
that calls cdbm
in selection mode and later outputs the path the user selected:
export PS1='$(cdbm -add)\h.\u:\W\$ ' function c() { dir=$(cdbm 3>&1 1>&2 2>&3); cd $dir; }
Now, if you type c
after .bashrc
has run (either automatically when opening a new shell or manually via source .bashrc
) in the shell, the newly defined bash function c
above will call the cdbm
program. The latter writes the selection list to stdout
, the user then interacts with the cursor keys, selects a directory with Enter, and cdbm
writes the result to stderr
.
Now the function only has to pass the contents of stderr
(the selected path) to the shell's cd
function, which then changes to the specified directory. This is easier said than done, because cd
is not a program, but a built-in shell function. A program could change its own working directory, but not that of the parent process, the shell itself. To complicate matters, unlike other Unix commands, cd
insists on being provided with an actual argument, the directory, which it cannot read from stdin
by way of a pipe.
This explains the wild trick the Bash function resorts to above. After calling cdbm
it swaps its stdout
and stderr
channels. To do this, it first uses 3>&1
to define a file descriptor named 3
and points it to the same channel as the file descriptor 1
(i.e., stdout)
. The following redirection 1>&2
assigns a new value to the 1
descriptor and points it to a descriptor 2
(i.e., stderr)
. The third – that is 2>&3
– assigns the value of the temporarily used file descriptor 3
(i.e., the cached stdout
) to stderr
. In other words, the terminal UI's output of cdbm
no longer ends up in stdout
, but instead in stderr
, and the result of the selected directory is sent to stdout
. We in engineering call this a "switcheroo."
The dir=$(...)
construct then grabs stdout
and assigns it to the $dir
variable. The following cd
statement for the directory change, separated by a semicolon, receives the value from the variable and jumps to the specified directory. This whole rigmarole was necessary, because the easy way of capturing stdout
does not work, as the terminal UI insists on writing to it, and redirecting it would leave the user without any visual output with which to interact.
Nitty Gritty
The cdbm.go
program in Listing 1 only has to do two things. First, if the -add
option is present, it stores the current working directory in the SQLite database. If the option is not set, it displays the terminal UI with the SQLite entries, lets the user select one, and outputs the chosen path to stderr
.
To do so, it defines the -add
option with the help of the standard flag
package in line 15. If cdbm
is called with -add
, the pointer value dereferenced with *addMode
has a true value after parsing the command-line arguments with flag.Parse()
, and line 33 branches to the function dirInsert()
starting in line 85. In display mode, the else
branch starting in line 34 uses dirList()
to fetch all the paths stored in the SQLite database and sorts them in descending order of the dates on which they were added.
The terminal UI for selecting a directory gets drawn by the promptui
package, which offers Select()
and Run()
functions to configure the list and then switch to user interaction mode. The result, the path selected by the user as a string, is finally output to stderr
by lines 44 and 45, whereupon the program terminates.
Buy this article as PDF
(incl. VAT)
Buy Linux Magazine
Subscribe to our Linux Newsletters
Find Linux and Open Source Jobs
Subscribe to our ADMIN Newsletters
Support Our Work
Linux Magazine content is made possible with support from readers like you. Please consider contributing when you’ve found an article to be beneficial.
News
-
Linux Kernel 6.13 Offers Improvements for AMD/Apple Users
The latest Linux kernel is now available, and it includes plenty of improvements, especially for those who use AMD or Apple-based systems.
-
Gnome 48 Debuts New Audio Player
To date, the audio player found within the Gnome desktop has been meh at best, but with the upcoming release that all changes.
-
Plasma 6.3 Ready for Public Beta Testing
Plasma 6.3 will ship with KDE Gear 24.12.1 and KDE Frameworks 6.10, along with some new and exciting features.
-
Budgie 10.10 Scheduled for Q1 2025 with a Surprising Desktop Update
If Budgie is your desktop environment of choice, 2025 is going to be a great year for you.
-
Firefox 134 Offers Improvements for Linux Version
Fans of Linux and Firefox rejoice, as there's a new version available that includes some handy updates.
-
Serpent OS Arrives with a New Alpha Release
After months of silence, Ikey Doherty has released a new alpha for his Serpent OS.
-
HashiCorp Cofounder Unveils Ghostty, a Linux Terminal App
Ghostty is a new Linux terminal app that's fast, feature-rich, and offers a platform-native GUI while remaining cross-platform.
-
Fedora Asahi Remix 41 Available for Apple Silicon
If you have an Apple Silicon Mac and you're hoping to install Fedora, you're in luck because the latest release supports the M1 and M2 chips.
-
Systemd Fixes Bug While Facing New Challenger in GNU Shepherd
The systemd developers have fixed a really nasty bug amid the release of the new GNU Shepherd init system.
-
AlmaLinux 10.0 Beta Released
The AlmaLinux OS Foundation has announced the availability of AlmaLinux 10.0 Beta ("Purple Lion") for all supported devices with significant changes.