Introducing the terminal to developers
The article I wish I had read when I had to open the terminal for the first time.
Introduction
Being a developer can be quite overwhelming these days. Getting familiar with a codebase and the framework(s) and libraries it uses is not the whole story. There is also a real demand of additional skills to get your job done, such as using Git, package managers, and build tooling.
Many of these tools are to be used in the terminal, which may be new and a bit frightening. But don’t worry, we’ve all been there. This article provides an overview, along with practical examples, to get you up to speed with the terminal and some essential tooling.
Why do I need the terminal?
For certain tasks you can get away without using the terminal at all. For example, there are great GUI tools for working with Git. However, getting familiar with tools like Git from the terminal gives you more power and flexibility. In the end, a GUI is a graphical shell in front of a command-line tool. Often limited by screen estate or a minimalistic design, a GUI may feature only a subset of the underlying command-line interface. Being “closer to the metal”, it can also help you to get out of trouble in case a GUI is stuck or messed up.
Frameworks and libraries for programming languages (such as JavaScript or PHP) come and go, but knowing your way around in the terminal is a skill you can use always and everywhere.
What actually is a terminal?
A terminal is text-based, and serves as the command-line interface (CLI) you can type your commands in. A shell takes these commands, and tells the operating system to execute them.
On macOS, the default terminal application is named “Terminal”, and the default shell is Bash. You will also find a “terminal” in Linux distributions like Debian and Ubuntu. On Windows 10, you can install a Linux environment.
For the rest of this article, I’ll be assuming you are using Bash or a similar shell.
Opening a terminal
- On macOS, you can use Spotlight (
⌘+Space
), and type “Terminal” to find and open it. - On Linux, search for the “terminal” app and open it. You can also try the
Ctrl+Alt+T
combo. - On Windows 10, start the “Bash on Ubuntu on Windows” application.
The first thing you will always get is a prompt. It’s what “prompts” you to type
a command. The prompt might include information such as the computer and/or user
name, and usually ends with a $
. Behind the $
is the cursor, where you can
type commands.
If your terminal has a white background color, but you prefer to have a black background color instead: this article contains instructions on how to do this.
Finding your way
In an application like Finder or File Explorer you can navigate your files and
folders. In the terminal we can do the same with commands like ls
and cd
(to
list files and change directories, respectively). For example:
The last line is the default output of ls
. You can pass it extra arguments (or
“options”) to change its behavior. For instance, there is -a
to show all files
(including hidden ones), and -l
to show one file per line:
$ ls -a -l
total 1160
drwx------ 6 lars staff 204 Sep 12 19:52 .
drwxr-xr-x 64 lars staff 2176 Sep 12 22:29 ..
-rw-r--r-- 7 lars staff 42 Sep 12 19:52 .DS_Store
-rw-r--r-- 7 lars staff 81238 Sep 12 19:52 tutorial.pdf
-rw-r--r-- 1 lars staff 11086 Jul 5 22:35 Keynote.pdf
The arguments can also be combined into one: ls -al
means the same.
Here are some essential commands to manage files and directories:
If you want more information about any of these commands, you could either use
the --help
argument, or perhaps google the command. Both ways are not ideal:
the former is correct and complete, but often hard to read. And the latter takes
more time and needs you to switch context between the terminal and a web
browser. Maybe we can improve on this. We all like readable documentation,
right?
Package managers
Let’s use this as a an opportunity to install a package manager, and then install a package with it. In this case, one for simplified documentation.
In a nutshell, package managers are an essential tool to install system-wide (or “global”) software, or local dependencies within a project. You may have heard of Homebrew, npm, Maven, RubyGems, or apt-get. In this article, we’ll be looking at Homebrew and npm.
- On macOS, Homebrew is “the missing package manager”. Find it at brew.sh.
- Most Linux distributions (including Linux on Windows), come with a
pre-installed package manager, such as
apt-get
oryum
.
Are you working with JavaScript? You need npm (the package manager for JavaScript), which comes with Node.js. You can install Node.js via many package managers. This will also make npm available on your system.
Now we can install a package named “tldr” with either one of them. Using npm:
$ npm install --global tdlr
Or, with Hombrew:
$ brew install tldr
Now you can try for instance tdlr cd
to see what I mean with readable
documentation:
$ tldr cd
cd
Change the current working directory.
More information: https://man.archlinux.org/man/cd.n.
- Go to the given directory:
cd path/to/directory
- Go to home directory of current user:
cd
- Go up to the parent of the current directory:
cd ..
- Go to the previously chosen directory:
cd -
We’ll dive into npm in a minute. However, if you have questions or got stuck installing a package manager, then you might want to check out one of the following links:
Git
Git is a program that many projects use for version control. Basically, it allows people to work on code together, while keeping track of changes. Everybody should not be editing the same files directly, and sometimes changes need to be reverted.
Collaborating with Git involves cloning (“downloading”) the code repository of a project, making changes to fix bugs or develop new features, and then pushing back (“uploading”) the changes. It is common to create a separate branch first, so others can review the changes before they are merged back into the master branch.
Let’s install Git (with a package manager, obviously)! Here are some options for various operating systems:
- For macOS with Homebrew:
brew install git
- For Linux (e.g. Debian, Ubuntu):
apt-get install git
- For Windows: use either the Git for Windows installer or Chocolatey:
choco install git
Below is a screenshot after running a few commands as an example:
$ mkdir my-first-repository
$ cd my-first-repository/
$ git init
Initialized empty Git repository in /Users/lars/Projects/my-first-repository/.git/
$ git checkout -b cool-feature
Switched to a new branch 'cool-feature'
$ echo "Some content" > somefile.txt
$ ls
somefile.txt
$ git add somefile.txt
$ git status
On branch cool-feature
No commits yet
Changes to be committed:
(use "git rm --cached <file>..." to unstage)
new file: somefile.txt
$ git commit -m "Add some file"
[cool-feature (root-commit) b706c43] Add some file
1 file changed, 1 insertion(+)
create mode 100644 somefile.txt
$
For more information on working with Git, here are a two excellent tutorials:
npm
npm is the package manager for JavaScript. Even though Node.js itself is a server-side application runtime, and npm was originally built for Node modules, npm proves to be a great dependency manager for JavaScript in general. Next to JavaScript modules, the npm repository contains many tools built on top of Node.js, such as Grunt, Webpack, Babel, UglifyJS, and many more.
To manage JavaScript dependencies with npm, a project has a package.json
file.
It contains meta data about the project, and always includes a project name and
version. Here’s an example:
Let’s clone this project with Git, and install its dependencies:
$ cd Projects/
$ git clone git@github.com:webpro/my-awesome-project.git
Cloning into 'my-awesome-project'...
remote: Enumerating objects: 3, done.
remote: Counting objects: 100% (3/3), done.
remote: Compressing objects: 100% (2/2), done.
remote: Total 3 (delta 0), reused 3 (delta 0), pack-reused 0
Receiving objects: 100% (3/3), done.
$ cd my-awesome-project/
$ ls -a
. .. .git package.json
$ npm install
added 56 packages, and audited 57 packages in 4s
[...]
$ ls node_modules/
asap fbjs js-tokens loose-envify react
balanced-match fs.realpath json3 minimatch react-is
brace-expansion glob lodash minimist safer-buffer
browser-stdout graceful-readlink lodash._baseassign mkdirp setimmediate
commander growl lodash._basecopy mocha supports-color
concat-map has-flag lodash._basecreate ms ua-parser-js
core-js he lodash._getnative node-fetch whatwg-fetch
create-react-class iconv-lite lodash._isiterateecall object-assign wrappy
debug inflight lodash.create once
diff inherits lodash.isarguments path-is-absolute
encoding is-stream lodash.isarray promise
escape-string-regexp isomorphic-fetch lodash.keys prop-types
$
The command npm install
installed the dependencies lodash
and react
(and
their dependencies) locally in the node_modules
directory.
Note that this installs the dependencies locally to the project. This is very
different from installing packages globally, as we did with the tldr
package
(using the --global
or -g
argument). In general, local packages are
dependencies for a single project and global packages are used from the command
line.
If you want to learn more, I can recommend this article about npm dependencies and scripts: Using npm dependencies in npm scripts.
Tips & tricks
In this section I’d like to present a few tips and tricks to make your life in the terminal easier right away.
Change the terminal’s theme
In macOS and some Linux distributions, the default background color of the terminal application is white, and the window might be slightly transparent. Yet many people prefer a black background in the terminal (like the screenshots in this article).
Here’s you can change this for macOS: from the Terminal.app, go to Preferences
(⌘,
), go to Profiles, and select the desired profile (e.g. the “Pro” theme).
Make sure to press the “Default” button to store this for later sessions as
well. In the same screen, you can go into the “Background Color” modal and set
opacity to 100% to remove the transparency.
Shortcuts & commands you should know
The ^
(caret) below represents the “Control” key:
Shortcut | Description |
---|---|
↑ | Show the previous command |
↓ | Show the next command |
^a | Move the cursor to the start of the line |
^e | Move the cursor to the end of the line |
⇥ | (tab) Auto-complete commands, and directory and file names |
!! | Run the previous command again |
^l | Clear the screen (or clear ) |
^c | Cancel the current process (if it hangs or becomes unusable) |
^d | Exit the current terminal (or exit ) |
Aliases and functions
Over time, you will see that you are using the same commands and parameters over and over again. This is where aliases and functions come in. An alias can be used as an abbreviation for a more complex command. For example:
Now you can use ll
as an alias and it will execute the ls
command including
the extra arguments. You can also still add extra arguments to ll
. Use alias
without arguments to get a list of all active aliases.
Functions can contain logic, and a main difference with aliases is that you can pass it arguments. Here’s a trivial example:
You can invoke this function with say_hi John
, and it would print
Hello, John
:
$ say_hi John
Hello, John
This is a very short and superficial introduction to this topic. If you are interested in learning more, please check out the following resources:
Wrapping up
After reading this article I hope you feel slightly more comfortable to use the terminal. I’ve compiled a short list of great resources to get some directions to learn more. What’s next for you?
If you feel there’s something important missing in this article, feel free to let me know (in a comment here, or via Twitter: @webprolific). Thanks for reading!