How I save literally one minute of work every day with a simple bash script

Andy Abi Haidar
5 min readJul 13, 2023

--

I’ve recently made the jump to NeoVim, and I’m currently at the phase where I feel the need to tell everyone that I recently made the jump to NeoVim, because why else would you move to it.

But, with that, I’ve started to change my workflow and how I do specific things with the aim of shaving off as many seconds/minutes as possible, and focus on not doing any repeatable tasks that over time just become annoying.

As a developer, I’m working on a lot of different projects that are written in different languages, and require unique commands to initialize the projects, or to stop them.

My current workflow

Hypothetical example, but say I want to switch from project A that runs Django and React, to project B, which runs on Laravel and Vue. Here’s how I would do that:

  1. Go to my terminal, CMD-C all the node processes that are running ( npm run watch, etc…)
  2. Go to my other terminal, CMD-C the Django server
  3. Exit the docker container that’s running Django, run docker-compose stop
  4. Wait until that’s over, then cd into my other’s project’s directory
  5. Run sail up -d
  6. Switch to another tab, cd into whatever folder holds my frontend stuff
  7. Run npm run watch
  8. Switch to VSCode, and either use some Project Manager plugin to switch projects, or just CMD-O and choose the folder.

Now sure, there are some ways to shave off seconds here. This is just an example of a typical process I would take.

…but we’re developers. We get bored, find a stupid problem that makes us waste 5 extra minutes a week, and what do we do? We decide to spend 5 hours finding a solution for that.

My current workflow

All my projects now have a .catapult.yml file that looks something like this:

up: docker-compose up -d
down: docker-compose stop
bg:
- docker-compose exec -it django bash
- docker-compose exec -it django bash -c "python manage.py runserver; bash"
- nvm use 16 && cd assets/frontend && npm run dev

What this basically does is tell Catapult that you need to run whatever is in up whenever you want to launch the project, and then down whenever you want to stop it. It also instructs it to start a separate bg tmux window with 3 panes, each running its own command for background processes like npm, Django server, etc…

So now my whole workflow went from the 8 steps you saw above, to:

  1. p run projectb

And this will kill the old project and all its dependencies, start docker and everything needed to run, then put me into NeoVim with the code open.

Some cool things as well, like giving it multiple projects p run projecta projectb say for something like a backend project and then a frontend project.

This the p bash script that I use. This was my first complex-ish bash script, so be gentle.

#!/bin/bash

# The file the stores the name of the current projects
CURRENT_PROJECTS_FILE="$HOME/.current_projects"

# Development directory that holds all the project folders
DEV_DIR="$HOME/dev"

# A python script that parses the .catapult.yml file and returns the command to run
PARSE_YAML_SCRIPT="$HOME/parse_yaml.py"

# Name of the configuration file that will be present in each project folder
CONFIG_FILE=".catapult.yml"

# Name of the code window
CODE_WINDOW="code"

# Name of the background window
BG_WINDOW="bg"

# Name of the keys in the .catapult.yml file
UP_KEY="up"
DOWN_KEY="down"
BG_KEY="bg"

# Name of the code editor to use
CODE_EDITOR="nvim"

function run() {
# Allow support for parallel projects with the -p flag.
# This is useful when we don't want to stop whatever project is already running
local parallel=0
if [[ $1 == "-p" ]]; then
parallel=1
shift
fi
if [[ "${parallel}" -eq 0 ]]; then
stop
fi

project_to_attach=""
for project in "$@"; do
local project_dir="${DEV_DIR}/${project}"
local config_file="${project_dir}/${CONFIG_FILE}"

if [ ! -d "${project_dir}" ]; then
echo "Project ${1} does not exist."
return 1
fi

if [ ! -f "${config_file}" ]; then
echo "${confing_file} file for project ${1} does not exist."
return 1
fi

# Running the "up" command to initialize the project
local run_command=$(python3 "${PARSE_YAML_SCRIPT}" "${config_file}" "${UP_KEY}")
echo "Running ${run_command}..."
cd "${project_dir}" && eval "${run_command}"

# Check for tmux session and create it if doesn't exist
tmux has-session -t="${project}" 2>/dev/null
if [ $? != 0 ]; then
tmux new-session -d -s "${project}"
echo "Created new tmux session ${project}."
else
echo "Tmux session ${project} already exists."
fi

# Create a new window for background processes in the session and send commands to it
# We create a new vertical pane for every entry
tmux list-windows -t="${project}" | grep -q "${BG_WINDOW}"
if [ $? != 0 ]; then
tmux new-window -n "${BG_WINDOW}" -t="${project}"
local bg_commands=$(python3 "${PARSE_YAML_SCRIPT}" "${config_file}" "${BG_KEY}")
IFS=$'\n'
for cmd in $bg_commands
do
echo "Running ${cmd}..."
tmux split-window -h -t="${project}:${BG_WINDOW}"
tmux send-keys -t="${project}:${BG_WINDOW}" "${cmd}" ENTER
done
tmux kill-pane -t="${project}:${BG_WINDOW}.0" # Remove the first, empty pane
tmux select-pane -t="${project}:${BG_WINDOW}.0"
else
echo "Tmux window ${project}:${BG_WINDOW} already exists."
fi

echo "${project}" >> "${CURRENT_PROJECTS_FILE}"

# Create the base window for the code editor, and focus it
tmux rename-window -t="${project}:0" "${CODE_WINDOW}"
tmux send-keys -t="${project}:${CODE_WINDOW}" "${CODE_EDITOR} ." ENTER
tmux select-window -t"${project}:${CODE_WINDOW}"

project_to_attach="${project}"
done

if [[ -n "${project_to_attach}" ]]; then
tmux attach-session -t="${project_to_attach}"
fi
}

function stop() {
local current_projects=$(cat "${CURRENT_PROJECTS_FILE}")

if [[ -n "${current_projects}" ]]; then
for project in ${current_projects}; do
if [[ -d "${DEV_DIR}/${project}" ]]; then
local down_command=$(python3 "${PARSE_YAML_SCRIPT}" "${DEV_DIR}/${project}/${CONFIG_FILE}" "${DOWN_KEY}")

if [[ -n "${down_command}" ]]; then
echo "Running ${down_command}..."
cd "${DEV_DIR}/${project}" && eval "${down_command}"
fi

tmux kill-session -t="${project}"
echo "Stopped ${project}."
else
echo "Project ${project} does not exist."
fi
done
else
echo "No projects running."
fi

echo "" > "${CURRENT_PROJECTS_FILE}"
}

"$@"

If you are interested in having a similar workflow, feel free to clone https://github.com/andyabih/catapult and do your thing.

Disclaimer: I am not an expert at neither tmux, NeoVim, or Bash for that matter. So if you see any improvements to the code I would love to hear it.

--

--

Andy Abi Haidar
Andy Abi Haidar

Written by Andy Abi Haidar

Coder by day... and by night too to be honest.

No responses yet