Automating Our Workstation Setup

Learn how to automate our workstation setup.

Throughout this course, we’ve created a customized environment. We’ve installed some tools on our machine, created some directories, and wrote some configuration files. Once we get something just how we like it, we might want to keep it that way. We can create a script that’ll set things up for us. That way, when we get a new machine, we can run our script and have everything set up exactly how we want it.

Let’s create a script that does just that. We’ll install the handful of utilities we’ve installed throughout the course and create a ~/.bashrc file with some basic options. To keep this exercise short, the script won’t contain everything we’ve done, but it’ll contain enough that we’ll be able to add everything else on our own. We’ll start by creating a script to set up an Ubuntu environment, which we can test with the Ubuntu virtual machine we created. If you’re on a Mac, follow along anyway because once we’re done, we’ll alter the script so that it works on macOS.

To build this script, we’ll leverage many of the techniques we learned in this course, along with a few new ones. We’ll use some conditional statements to check exit codes from commands, we’ll install programs with the package manager, and we’ll create files with cat. We’ll also use variable expansion throughout the script, a small bit of command substitution, and a for loop to iterate over a collection of packages.

Setting up the OS

Let’s create a new file named ubuntu_setup.sh and make it executable:

touch ubuntu_setup.sh
chmod +x ubuntu_setup.sh

We start our script with the shebang line and declare a variable that holds a datestamp to use as a suffix when creating backups of files. Then, we add a call to sudo, which will cause the script to prompt for the sudo password right away:

#!/usr/bin/env bash
datestamp=$(date +"%Y%m%d%H%M") 
sudo -v

We’re not using set -e this time because we’re going to trap errors ourselves.

Setting output colors

The commands in this script will generate a lot of text output. Let’s colorize information messages, error messages, and success messages so they’ll stand out from the other text.We add this code to create variables to hold the color values:

ERRORCOLOR=$(tput setaf 1) # Red
SUCCESSCOLOR=$(tput setaf 2) # Green 
INFOCOLOR=$(tput setaf 3) # Yellow 
RESET=$(tput sgr0)

This is similar to how we defined the colors for our shell prompt in our .bashrc file, except this time, we don’t need to include square braces in the output since these won’t be used as part of the prompt. And just like with the prompt, we include a variable named RESET so we can put the shell colors back to normal.

To use these colors, we print them out using variable expansion inside of echo statements. That’ll get messy quickly, so we define functions we can call instead to print info, success, and error messages:

function info() { echo "${INFOCOLOR}${@}${RESET}"; } 
function success() { echo "${SUCCESSCOLOR}${@}${RESET}"; } 
function error() { echo "${ERRORCOLOR}${@}${RESET}" >&2; }

Each function prints the color code to the terminal, then prints all the arguments passed to the function (${@}), and then resets the terminal’s text settings. The error function is slightly different, though. It redirects the output to STDERR instead of STDOUT, the default output of echo.

Starting message

Next, we display a message that informs the user we’re starting the script. We use the info function we just defined and use command substitution with the date command to get the date to display:

info "Starting install at $(date)"

Installing packages

Now, we add this code to the script. It installs some of the software packages we used in this course.

declare -a apps=(tree curl unzip make) info "Updating Package list"
sudo apt-get update
info "Installing apps:" 
for app in ${apps[@]}; do
info "Installing ${app}" 
sudo apt-get -y install $app 
result=$?
if [ $result -ne 0 ]; then
error "Error: failed to install ${app}"
exit 1
else
success "Installed ${app}"
fi 
done

First, we define a variable named apps, which contains a list of applications we want to install. We then iterate over each entry in the list using a for loop, displaying the application’s name and passing the application to the apt-get command to install it. The -y option tells apt-get to install the program without confirmation, which is perfect for a script.

Throughout the course, we’ve used apt install instead of apt-get install. The apt command is a newer high-level user interface for installing packages interactively. It sets some default options for package installation and looks a little nicer. The apt-get command is the legacy command and is a good fit for scripts because it has a more stable user interface. In fact, if we use apt in a script, we see this warning:

WARNING: apt does not have a stable CLI interface. Use with caution in scripts.

We could install these packages without using the list of packages and the for loop because the apt-get command supports supplying a list of packages to install all at once. But with this structure, we have a little more control.

Let’s add installing Node.js to this setup script. We use curl to grab the installation script and execute it directly this time. This “pipe a script from the internet to Bash” approach does pose a security risk. Still, we’re going to do it here because we’re trying to automate the process anyway, so downloading it and inspecting it as we did previously doesn’t make sense. We have to trust that the servers holding the script haven’t been compromised.

We add this code to the script:

echo "Installing Node.js from Nodesource"
curl -sL https://deb.nodesource.com/setup_11.x | sudo -E bash - sudo apt-get -y install nodejs
result=$?
if [ $result -ne 0 ]; then
error "Error: Couldn't install Node.js."
exit 1
else
success "Installed Node.js" 
fi

Once again, we check the exit status from the apt-get command and display an error message if it failed.

Creating a .bashrc file

Now that the software is installed, let’s create a ~/.bashrc file containing some of the settings we configured in The Shell and Environment. The code that follows won’t have everything we did previously, but we can add more as we work through this. Let’s back up any existing ~/.bashrc file first. We add this code to check if one exists. If it does, we back it up:

info "Setting up ~/.bashrc"
if [ -f ~/.bashrc ]; then
oldbashrc=~/.bashrc.${datestamp}
info "Found existing ~/.bashrc file. Backing up to ${oldbashrc}." 
mv ~/.bashrc ${oldbashrc}
fi

To create the backup file name, we use the datestamp variable we defined at the top of the script as a suffix for the file. Notice that we aren’t using any quotes around the value we are creating. The tilde isn’t expanded to the full path in a double-quoted string. It’s stored as a literal character. If we used double quotes here, the command to move the script would fail because the destination path would be invalid. As an alternative, we could replace all occurrences of the tilde with ${HOME}. That value would be expanded in the double-quoted string.

Now, we use cat to create the new ~/.bashrc file and then check the status of the command:

cat << 'EOF' > ~/.bashrc [ -z "$PS1" ] && return
shopt -s histappend
HISTCONTROL=ignoreboth
HISTSIZE=1000
HISTFILESIZE=2000
HISTIGNORE="exit:clear"
export EDITOR=nano
export VISUAL=nano
export PATH=~/bin:$PATH EOF
result=$?
if [ $result -ne 0 ]; then
error "Error: failed to create ~/.bashrc"
exit 1
else
success "Created ~/.bashrc" 
fi

Using single quotes around the EOF in the cat line ensures that the variables in the body of the heredoc aren’t interpreted.

Finally, we add a message at the end of the script that confirms things are finished and tells the user what to do next:

success "Done! Run source ~/.bashrc to apply changes."

Now, we save our file. To test it out, we run the script with ./ubuntu_setup.sh. It’ll produce a lot of output.

Run the complete code on the terminal below for practice.

cat << "END" > ubuntu_setup.sh
#!/usr/bin/env bash
#---
# Excerpted from "Small, Sharp, Software Tools",
# published by The Pragmatic Bookshelf.
# Copyrights apply to this code. It may not be used to create training material,
# courses, books, articles, and the like. Contact us if you are in doubt.
# We make no guarantees that this code is fit for any purpose.
# Visit http://www.pragmaticprogrammer.com/titles/bhcldev for more book information.
#---
datestamp=$(date +"%Y%m%d%H%M")

sudo -v

ERRORCOLOR=$(tput setaf 1)    # Red
SUCCESSCOLOR=$(tput setaf 2)  # Green
INFOCOLOR=$(tput setaf 3)     # Yellow
RESET=$(tput sgr0)

function info()    { echo "${INFOCOLOR}${@}${RESET}"; }
function success() { echo "${SUCCESSCOLOR}${@}${RESET}"; }
function error()   { echo "${ERRORCOLOR}${@}${RESET}" >&2; }

info "Starting install at $(date)"

declare -a apps=(tree curl unzip make)

info "Updating Package list"
sudo apt-get update

info "Installing apps:"
for app in ${apps[@]}; do
  info "Installing ${app}"
  sudo apt-get -y install $app
  result=$?
  if [ $result -ne 0 ]; then
    error "Error: failed to install ${app}"
    exit 1
  else
    success "Installed ${app}"
  fi
done


echo "Installing Node.js from Nodesource"
curl -sL https://deb.nodesource.com/setup_16.x | sudo -E bash -
sudo apt-get -y install nodejs
result=$?
if [ $result -ne 0 ]; then
  error "Error: Couldn't install Node.js."
  exit 1
else
  success "Installed Node.js"
fi

info "Setting up ~/.bashrc"

if [ -f ~/.bashrc ]; then
  oldbashrc=~/.bashrc.${datestamp}
  info "Found existing ~/.bashrc file. Backing up to ${oldbashrc}."
  mv ~/.bashrc ${oldbashrc}
fi

cat << 'EOF' > ~/.bashrc
[ -z "$PS1" ] && return

shopt -s histappend
HISTCONTROL=ignoreboth
HISTSIZE=1000
HISTFILESIZE=2000
HISTIGNORE="exit:clear"

export EDITOR=nano
export VISUAL=nano

export PATH=~/bin:$PATH
EOF

result=$?
if [ $result -ne 0 ]; then
  error "Error: failed to create ~/.bashrc"
  exit 1
else
  success "Created ~/.bashrc"
fi

success "Done! Run   source ~/.bashrc    to apply changes."
END
chmod +x ubuntu_setup.sh
./ubuntu_setup.sh

Get hands-on with 1400+ tech skills courses.