OverTheWire - Bandit
Introduction
Bandit is just one of several wargames available at overthewire.org. It's aimed at beginners to Linux and CTFs and provides an excellent introduction to the basics of the Linux command line.
In this walkthrough I'm going to show the process for solving each challenge while also providing some insight into what each challenge is trying to teach and why it's useful, so keep an eye out for callouts like those below.
Access to each Bandit level is made over SSH. The username will correspond to the index of the level starting at 0. Always make sure you're logging into the correct Bandit level!
If you're looking for the solution of a particular level, you are highly encouraged to attempt it on your own before following this walkthrough. You will learn some useful things regardless, but attempting it on your own first will help the knowledge stick better. I promise.
I have a few recommendations before getting started. Most of the OverTheWire
Bandit levels provide links to potentially useful manual pages. Unfortunately
the man
pages can be somewhat cryptic, especially for beginners. For that
reason I recommend the following steps to try to get unstuck when slogging
through the man
pages for any particular command.
- Search for examples. Many
man
pages contain examples of their usage. These may be marked by all-capsEXAMPLES
in some cases, but sometimes not. - If skimming through the man pages doesn't surface any apparent examples,
then you may want to refer to cheat.sh. Cheat.sh is a
database of usage examples for thousands of command line programs. It can be
searched from the main page or by simply adding a
/
followed by the command. For example https://cheat.sh/ssh will return examples ofssh
usage. This one can be helpful when trying to understand how a command is commonly used and which command flags are most useful. - If you're struggling to understand some of the examples then paste them into
https://explainshell.com. Explainshell
provides a more readable breakdown of each command line flag and argument. This
tool can save you a lot of time flipping back and forth through the
man
pages.
Please also note that for any terminal output, the command prompt will be
truncated to a single $
for brevity. Just remember that you'll need to be
logged in to the appropriate level before executing any of the commands.
For example:
$ cat hello.txt
Hello there!
Log in
The goal of this level is for you to log into the game using SSH. The host to which you need to connect is bandit.labs.overthewire.org, on port 2220. The username is bandit0 and the password is bandit0. Once logged in, go to the Level 1 page to find out how to beat Level 1.
Bandit 0 provides an introduction to SSH and provides a couple useful links for further research.
Firstly, let's discuss what SSH is and how it's used day-to-day. SSH is a
networking protocol that allows secure communication between networked computer
systems. This is distinct from the ssh
command line program linked above. The
program implements the protocol and allows regular users to utilize the SSH
protocol for secure communication. The security of SSH is based on public-key
cryptography which
you're welcome to read more about, but many of the details are out of scope for
this walkthrough.
The most important thing you need to know about SSH is that it can be used to
login to a computer system with a username
and password
just as if you were
physically present. Except in this case it's another system accessible over the
network. This is how it will be used over the course of the Bandit wargame as
well as some of the other wargames by OverTheWire.
It's worth mentioning that SSH is usually (and preferably) used with an SSH key. More specifically a key-pair. A public key and a private key which are both needed to take advantage of the aforementioned public-key cryptography. I won't go into detail here, but don't worry there are a couple later levels that do utilize SSH keys, so I'll discuss them in more detail when we reach them.
The prompt tells us that both the username
and password
are bandit0
.
Just as important though, is the network location we've been provided,
bandit.labs.overthewire.org
, which we'll use to connect to level 0. Actually,
bandit.labs.overthewire.org
is what's known as a
hostname or domain
name. More specifically, it's a
fully qualified domain name
(FQDN). You can
recognize an FQDN because it will include a top-level domain
(TLD) such as .com
, .net
,
or .org
. This naming system for computers is called the Domain Name System
(DNS). Luckily we don't need
to dig into the details here. Just know that one of these FQDNs will resolve to
an IP address which can (usually)
be used to uniquely identify a computer system on the internet.
This unique name will tell the ssh
program what system to connect to. The
ssh
program expects the user credentials and hostname in the following format
ssh://[user@]hostname[:port]
. This is explained in the first couple
paragraphs of the
ssh man page.
ssh connects and logs into the specified destination, which may be specified as either
[user@]hostname
or a URI of the formssh://[user@]hostname[:port]
.
The elements wrapped in brackets [
and ]
are actually optional. Linux man
pages will commonly express the available
flags and arguments for a command in this way.
With that knowledge we can now connect to the first level with this command.
ssh ssh://bandit0@bandit.labs.overthewire.org:2220
You should be greeted by a banner for the Bandit wargame and a prompt
requesting entry of the bandit0
password. Enter bandit0
and you're ready to
get started.
It isn't necessary to solve any of the Bandit levels, but I highly recommend researching DNS. It's intrinsic to how the internet operates and you'll likely encounter it again and again if you pursue any area of IT, software, or cybersecurity.
Resources:
- What is DNS?: an excellent intro to the basics of DNS by freeCodeCamp
- Implement DNS in a weekend: if you have some programming experience, I highly recommend following this walkthrough by Julia Evans
Walkthrough
Level 0
Now that we're logged in, it's time to solve the first level.
The password for the next level is stored in a file called readme located in the home directory. Use this password to login to
bandit1
using SSH. Whenever you find a password for a level, use SSH (on port 2220) to log into that level and continue the game.
The prompt also provides links to the man pages for several commands.
With perhaps the exception of du
. Each of these commands is fundamental for
any Linux user who wants to effectively use the command line.
Command | Description |
---|---|
ls | list the files in a directory |
cd | change your current directory |
cat | display the contents of a file |
file | display information about the content of a file |
du | display the disk space used by files |
find | search the file system for files with various parameters such as filename, file type, file size, etc. |
You are highly encouraged to review each of the commands. At the very least
check out the cheat.sh
page for each. Regardless these are all pretty common,
so you'll get plenty of experience with these commands in later levels.
For now, it should be apparent that one of the above commands should serve to
show us the contents of the readme
file mentioned in the prompt. The humble
cat
command.
It takes a filename as a parameter. So providing it with the filename readme
should print it's contents.
$ cat readme
Congratulations on your first steps into the bandit game!!
Please make sure you have read the rules at https://overthewire.org/rules/
If you are following a course, workshop, walkthrough or other educational activity,
please inform the instructor about the rules as well and encourage them to
contribute to the OverTheWire community so we can keep these games free!
The password you are looking for is: [REDACTED PASSWORD]
I won't be providing any of the passwords throughout this walkthrough per the OverTheWire rules. So get out your notepad and copy those passwords. You'll need them to return to the last level you've completed without completing each level again. Once you've done that, you're ready to continue to Level 1.
Level 1
The password for the next level is stored in a file called
-
located in the home directory
The prompt also provides links to the man pages for the same commands as level 0.
Naturally, you might think to try the cat
command again. Afterall, we're told
the password in in the file called -
. Let's try it and see what happens.
$ cat -
█
Strange, you're left with a blank line on the terminal now. And, if you type
some text and press the Enter
key, the text is repeated back to the
terminal like so.
$ cat -
hello there
hello there
The reason for this will become clear if you read the description from the
cat
man page.
NAME
cat - concatenate files and print on the standard output
SYNOPSIS
cat [OPTION]... [FILE]...
DESCRIPTION
Concatenate FILE(s) to standard output.
With no FILE, or when FILE is -, read standard input.
So, according to the description, the parameter -
, makes cat
read from
standard input. You may have guessed this already, but standard input or STDIN
is usually what's entered by the user in the terminal; however, STDIN doesn't
explicitly refer to input from a user. Rather, it refers to a stream of data
that is being sent to a program, so it may also refer to files or even the
output of other programs that is being passed to other programs.
You'll most likely hear of this concept of input and output referred to as STDIO or standard input and output. If you'd like to read more, there is an excellent article by freeCodeCamp explaining more about it and many other useful concepts.
There will be opportunities to demonstrate STDIO and IO redirection later, but
for this level, all you need to recognize is that -
is a special character
that tells cat to read input from STDIN instead of a file as we saw before. So
to properly refer to the -
file, it must be referenced by some other means
than the simple filename. There are several ways to accomplish that.
-
Use
./<filename>
where<filename>
would be-
for this example. The.
is a special character that is interpreted as the current directory. This is usually implied when we just enter a file by it's name. However, stating it explicitly allows us to circumvent the special case of using-
as an argument tocat
.$ cat ./- [REDACTED PASSWORD]
-
Use the full path. On Linux and other Unix-based systems the root of the file system can be specified with a
/
. To use this method though we'll need to know the full path of the-
file. To get that we can use thepwd
command, which is short for "print working directory".$ pwd /home/bandit1
To complete the full path for
-
we just need to append/<filename>
.$ cat /home/bandit1/- [REDACTED PASSWORD]
-
We can also use what is called a glob (
*
). The glob can be used to execute commands over multiple files at once. For example using the following command will print all files in the current working directory.$ cat ./* [REDACTED PASSWORD]
In this example the only file in our current directory is the password file
-
. But just like the other examples it circumvents the special-
argument tocat
.The glob is actually a part of a larger set of filename expansion features available in Bash. They're quite useful and we may see more of them in future levels.
The main takeway from this level is that there are special characters that may
change how commands are interpreted on the command line. Some of them will be
built-in to whatever shell you're using, but some may just be conventions like
the -
character for STDIN and won't apply to every program. Here is a
breakdown of the special characters used in the Bash shell that you might want
to watch out for.
https://mywiki.wooledge.org/BashGuide/SpecialCharacters
Level 2
The password for the next level is stored in a file called
spaces in this filename
located in thehome
directory
This level is similar to Level 1, except that the file is not a special character. Instead it contains special characters, the space █. The space character is essential for the shell to interpret the input text.
$ cat spaces in this filename
cat: spaces: No such file or directory
cat: in: No such file or directory
cat: this: No such file or directory
cat: filename: No such file or directory
As you can see from above, entering the name as it's written will cause the
cat
command to interpret each word in the file as a separate filename.
Just as before, there are a few ways to get around this.
-
Escape the space characters. In Bash, the backslash
\
is used as an escape character. When the\
is used, the following character is interpreted literally. This allows the space characters of the filename to be "escaped".$ cat spaces\ in\ this\ filename [REDACTED PASSWORD]
-
Instead of escaping the spaces individually, the filename can also be surrounded by single quotes. Every character between two single quotes is interpreted literally.
$ cat 'spaces in this filename' [REDACTED PASSWORD]
-
Similarly, double quotes can also be used to interpret the surrounded characters literally. However, there are some exceptions, and double quotes allow some special characters to be interpreted. But the space character is not one those, so it is functionally the same as the above example in this case.
$ cat "spaces in this filename" [REDACTED PASSWORD]
Recognize that it's sometimes necessary to escape characters within filenames and other arguments.
Most Linux users will avoid naming files with any special characters, but sometimes you'll still run into them. This is especially true with files created on Windows where spaces inside file and directory names are much more common.
Level 3
The password for the next level is stored in a hidden file in the
inhere
directory.
$ ls ./inhere
According to the ls
command, there don't appear to be any files in the
inhere
directory. That's because ls
doesn't display hidden files by
default.
To show hidden files with ls
, it's necessary to use the --all
or -a
flag.
$ ls -a ./inhere
. .. ...Hiding-From-You
With the --all
flag enabled, ls
now shows all the files in the inhere
directory. Including the hidden files. In Linux and other Unix-based systems, a
leading dot .
in a filename is used to indicate a hidden file. As such, you
may here them referred to as dotfiles.
Now that we know the name of the hidden file, cat
can be used to print the
contents just as we've seen before.
$ cat inhere/...Hiding-From-You
[REDACTED PASSWORD]
The term "dotfile" is also frequently used to refer to a user's personal
configuration files. This is because many configuration files follow the
leading dot .
convention, so they don't clutter up directory listings.
Some systems will have aliases for the ls
command as well, with various flags
enabled. Here a few common ones.
ll
forls -AlhF
to list all files in a human-readable, long formatla
forls -A
to list all files excluding.
and..
l
forls -CF
to list files in a column formatlsd
forls --group-directories-first
See this DigitalOcean article to learn more about common aliases and how to configure your own.
Level 4
The password for the next level is stored in the only human-readable file in the
inhere
directory. Tip: if your terminal is messed up, try thereset
command.
If you haven't done so already, now would be a good time to read through the manual for each of the commands that were already mentioned in Level 0.
If you've read through the description for each, you should have a pretty good idea of which command you'll need to solve this one.
This level is asking us to identify which files in inhere
are human readable.
The file
command can get the information we need.
From the file
manual page we know that we can expect a response of text
for
any files that only contain typical ASCII
characters.
The type printed will usually contain one of the words `text` (the file
contains only printing characters and a few common control characters and is
probably safe to read on an ASCII terminal), `executable` (the file contains
the result of compiling a program in a form understandable to some UNIX
kernel or another), or `data` meaning anything else (data is usually “binary”
or non-printable).
One option is to execute file
on each file in inhere
one-by-one like so.
$ file inhere/-file00
inhere/-file00: data
A better way is to use the glob *
that we've seen previously. This allows us
to run the file command on all the files at once.
$ file ./inhere/*
./inhere/-file00: data
./inhere/-file01: data
./inhere/-file02: data
./inhere/-file03: data
./inhere/-file04: data
./inhere/-file05: data
./inhere/-file06: data
./inhere/-file07: ASCII text
./inhere/-file08: data
./inhere/-file09: data
From here it's obvious that we want the file with the text data at ./inhere/-file07
.
$ cat ./inhere/-file07
[REDACTED PASSWORD]
Always keep an eye out for opportunities to use globbing. Especially if you're running commands over a bunch of files.
Level 5
The password for the next level is stored in a file somewhere under the
inhere
directory and has all of the following properties:
- human-readable
- 1033 bytes in size
- not executable
This level requires a precise approach to locate the correct file. While it's
technically possible to solve with just the ls
and file
commands. It would
require some tedious manual searching to find the files that match each of the
criteria. Luckily the find
command is capable of locating files with all the
above criteria. You just need to know the right flags.
The find
command is essential to efficiently locating files on Linux systems.
It has several flags that can be used to refine its search. Most importantly
for this level are -readable
, -size
and -executable
.
$ find -readable -size 1033c -not -executable
./inhere/maybehere07/.file2
Note -size
and -not
flags. The c
suffix for the -size
argument is used
to indicate a size in bytes. The other available suffixes are all available in
the find man
page.
Additionally, the -not
flag negates the next expression, thus locating any
files that aren't executable in this example.
In this case the content and size of the file are sufficient to uniquely
identify the file and the -not -executable
isn't strictly necessary.
$ find -readable -size 1033c
./inhere/maybehere07/.file2
Once again, cat
the file to get the password.
$ cat ./inhere/maybehere07/.file2
[REDACTED PASSWORD]
Level 6
The password for the next level is stored somewhere on the server and has all of the following properties:
- owned by user bandit7
- owned by group bandit6
- 33 bytes in size
This level is very similar to Level 5 with a couple minor
differences. First, the file is "stored somewhere on the server" instead of in
the inhere
directory. That just means we'll need to run the find
command
from the root of the file system to ensure the file isn't missed. Secondly, the
file is specified by two new parameters. The user and
group that own the file.
Searching the find
manpage you can find the two flags -user
and -group
to
filter for files owned by the bandit7
user and the bandit6
group as
specified by the prompt. The final parameter is for a -size
of 33 bytes which
we already saw in level 5.
Here's an example of the output from find
with all the required arguments.
$ find / -user bandit7 -group bandit6 -size 33c
find: ‘/drifter/drifter14_src/axTLS’: Permission denied
find: ‘/root’: Permission denied
find: ‘/snap’: Permission denied
find: ‘/tmp’: Permission denied
find: ‘/proc/tty/driver’: Permission denied
find: ‘/proc/250118/task/250118/fd/6’: No such file or directory
find: ‘/proc/250118/task/250118/fdinfo/6’: No such file or directory
find: ‘/proc/250118/fd/5’: No such file or directory
...
[TRUNCATED OUTPUT]
Unfortunately searching with find
from /
has a side effect. Any files or
directories that the current user is not allowed to read will print an error to
the terminal. This makes it pretty difficult to parse the output for any
resulting files that match our search. To avoid this flood of errors, a common
solution is to redirect the standard error stream to
/dev/null
This is actually just a file that discards anything written to it.
To redirect a data stream in Bash we must specify its file
descriptor, which is an
integer. The file descriptor is followed by a greater than sign >
which
indacates that the stream should be redirected to a target file. Here is the
same command as above with all errors redirect to /dev/null
.
$ find / -user bandit7 -group bandit6 -size 33c 2>/dev/null
/var/lib/dpkg/info/bandit7.password
As you can see, the output is considerably easier to understand. Read the file
at /var/lib/dpkg/info/bandit7.password
to get the next password.
Level 7
The password for the next level is stored in the file
data.txt
next to the word "millionth".
This is the first level that OverTheWire introduces some new recommended commands since level 0.
Command | Description |
---|---|
man | access the system reference manuals |
grep | print lines that match patterns |
sort | sort lines in text files |
uniq | remove duplicate lines from a file |
strings | print readable strings from arbitrary files (even binary) |
base64 | encode data into Base64 |
tr | translate and replace characters |
tar | a utility for archive files |
gzip | a utility for compressing files |
bzip2 | a utility for compressing files |
xxd | a tool for creating a hex dump of a file |
Once again, I highly recommend at least reading through the introduction for each of these commands and checking out the examples at cheat.sh.
If you've followed the above advice, there should really only be one contender
to solve this level. The grep
command.
NAME
grep - print lines that match patterns
SYNOPSIS
grep [OPTION...] PATTERNS [FILE...]
grep [OPTION...] -e PATTERNS ... [FILE...]
grep [OPTION...] -f PATTERN_FILE ... [FILE...]
DESCRIPTION
grep searches for PATTERNS in each FILE. PATTERNS is one or more
patterns separated by newline characters, and grep prints each line that
matches a pattern. Typically PATTERNS should be quoted when grep is used in a
shell command.
According to the syntax description we should be able to search for patterns in a file with the following syntax.
grep <PATTERN> <FILE>
Swapping in the values mentioned in the prompt will return the line we're looking for.
$ grep "millionth" data.txt
millionth [REDACTED PASSWORD]
Regex
Grep and more generally regular expressions (regex) are extremely useful. You'll find many applications have integrated support for text search via regex. In particular text editors, word processors, and programming IDEs.
To explore regex more I highly recommend reading through the Regex Quick Start Guide from regular-expression.info to get a feel for what's possible with regex and then follow that up with some experimentation on regex101.com. This is a tool that visualizes regex matches. Drop any text you want into it and try out all kinds of search patterns. Try to match words, letters, various combinations of upper and lowercase letters, punctuation, etc. Seeing regex in action and observing precisely what matches with different patterns will give you a much better intuition for what's possible than anything I could write here.
Regex can get very complicated very quickly, so regex101 is also a great tool for debugging your regex. Definitely give it a bookmark, it'll be a life saver. Trust me.
Level 8
The password for the next level is stored in the file
data.txt
and is the only line of text that occurs only once
This level is the first that seriously benefits from chaining two commands
together. In Bash this is done with the pipe |
character. The pipe, when
placed after a command will pass all of the output (stdout) into the input
(stdin) of the command that follows it.
For example, we can combine the ls
and grep
commands to list only those
files that contain "bash" in the name.
$ ls -a | grep bash
.bash_logout
.bashrc
Remember that the -a
flag is necessary to list hidden or dot files.
Read through the Piping and Redirection article provided under the helpful reading material section to learn more about piping.
Reviewing the recommended commands, one should stick out.
NAME
uniq - report or omit repeated lines
SYNOPSIS
uniq [OPTION]... [INPUT [OUTPUT]]
DESCRIPTION
Filter adjacent matching lines from INPUT (or standard input), writing
to OUTPUT (or standard output).
With no options, matching lines are merged to the first occurrence.
The uniq
command is able to filter matching or repeated lines. Since we're
looking for a unique line in a file, this will be helpful. However, there is
one caveat when using uniq
that you must be aware of. The uniq
command
filters adjacent matching lines. This means that any matching lines that
aren't directly adjacent, won't be filtered. So the first step must be to
organize the file such that matching lines are adjacent. In other words the
file should be sorted. The sort
program is designed precisely for this use
case.
For example, here's the first 20 lines of data.txt
when sorted.
$ sort data.txt | head -n20
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
As mentioned before, the pipe |
is useful here to send the output of the
sort
command to uniq
.
$ sort data.txt | uniq | head -n20
0BKVRLEJQcpNx8wnSPxDLFnFKlQafKK6
0eJPctF8gK96ykGBBaKydhJgxSpTlJtz
0kJ7XHD4gVtNSZIpqyP1V45sfz9OBLFo
0lPOvKhpHZebxji0gdjtGCd5GWiZnNBj
0REUhKk0yMqQOwei6NK9ZqIpE5dVlWWM
1jfUH1m4XCjr7eWAeleGdaNSxFXRtX0l
1VKPEkd0bCtIRwMFVQfY7InulwOFyDsn
2u8fvAzvnaFlvQG3iPt4Wc1TFhPcGxhH
35l6mr3f6TvlJyDwU6aUgJX07cLhr6t9
3FIgajXBiaQAiTMVGo1gxRDSiACNyvvJ
3mNA2le0gfURQKNHVIhGkMNLqLwjyyLN
4CKMh1JI91bUIZZPXDqGanal4xvAg0JM
4P8FsHcdr7d5WKnPtAaXY5SslKICd2gL
5EmwMKZHwF6Lwq5jHUaDlfFJBeHbcX0b
5hYz0028e1Q2TrtPVz5GZbpMzZNjebhh
5I2jWpqjtVp576xXI2TLh1UCyXJtGQ78
6Boy6esAjnIxCYn8uI6KZ7VD7zysDM8i
7cP8ssLElERHXqOJc9T84bxsmJBjNXk2
7qHmEo1FEbzthgyNpKc38YofXjYKZv18
8FCtUQlFXsJnNeyiDY5KfE3vRy6sZFEJ
Well that's strange. Why has uniq
returned all those lines? If you read the
description for uniq
carefully, then one line explains this.
With no options, matching lines are merged to the first occurrence.
So essentially what we have here is a file with each unique line of the file
where adjacent duplicates have been compressed into a single line. However, we
want to list only the unique lines from the input. For that, the -u
flag
will meet our needs. The -u
flag tells sort to only print unique lines. In
other words, only lines without any duplicates in the input.
So finally, we have a solution.
$ sort data.txt | uniq -u
[REDACTED PASSWORD]
Level 9
The password for the next level is stored in the file
data.txt
in one of the few human-readable strings, preceded by several ‘=’ characters.
Once again we're searching a file, so exploring our options with grep
may be a
good idea.
Trying a basic grep
for several =
characters doesn't seem to work.
$ grep '=== \w*' data.txt
grep: data.txt: binary file matches
The \w
is a shortcut for any word characters which equivalent to
[a-zA-Z0-9_]
.
We can see here, that grep found some matches, but data.txt
was interpreted
as a binary file. We can force grep to process the file as if it were text with
the -a
flag.
$ grep -a '=== \w*' data.txt
D]
h#!QJsVzl7POl%Y]Ha^UvToD|@T^N8g}b}?
Q#gm1x}========== theѦ+idW^)F1>)٘SK3PZt&xs肉WB/2ÜB Ź/Bjɢ<7<u/d|
-n
#iu=
7֣n)Uջش5bBKK}x>}:4Rl_7gHD:274CFy
6!&zB$l_GphqI.02H$Twm⧫o3mt0p~L3JprD========== passwordi L ~ˏ<@Ȅh$%Q5Dk |3
~Tf;o9sP#t+Pe쵟
OqDf.8Czmnf&vl:FXKbM
CIBi>Y
Еk $nXT=~}*4a2?TO"'&J~fDV3========== isd5z(#&s!10&poq
nR F
z|!(if+A64+'FTb5A}
éT:kAU2Qcɐ%#g+;YA_ekrX53|f8+e~&Oiu?VhM}^Qp^G==6!sT: "uVa-t\fg
](.ۍg:7nnp CD`voSQ-<]`@#H UumBiAj堵!O&D9========== [REDACTED PASSWORD]
On the last line of the above output, you'll find the password to the next level.
It's possible for grep
to output the precise matching text instead of each
entire line. The -o
flag is needed to do this. It tells grep to only output
the matched pattern.
1 $ grep -a -o '=== \w*' data.txt
2 === the
3 === passwordi
4 === is
5 === [REDACTED PASSWORD]
This gives a much clearer picture of the password without all the surrounding binary data.
Level 10
The password for the next level is stored in the file
data.txt
, which contains base64 encoded data
This one is pretty straight forward. The prompt gives it away by mentioning that the data is Base64 encoded.
Check the manual for the base64
command and you'll find one of the first
flags is -d
for --decode
.
$ base64 -d data.txt
The password is [REDACTED PASSWORD]
Be sure you understand what encoding is in this context.
Base64 is just one scheme of many to
convert binary data into printable characters i.e. alphanumeric characters with
the addition of the equal sign =
. In fact, that equal sign =
is used for
padding the end of encoded text, so it's often a dead giveaway that some text
was Base64 encoded. For example, the word "password" when Base64 encoded is
cGFzc3dvcmQ=
.
Base64 encoded text is extremely common on the web and you're likely to come across it at some point, so it may behoove you to read up on it. FreeCodeCamp has an excellent article that gives a good overview of how Base64 works and what it's used for.
You're less likely see other forms of binary to text encoding, but feel free to read more.
Level 11
The password for the next level is stored in the file data.txt, where all lowercase (a-z) and uppercase (A-Z) letters have been rotated by 13 positions
This challenge is describing what's commonly known as a shift cipher or Caesar cipher. It's not actually used in modern times for any meaningful attempts at securing messages, but it's somewhat popular in CTFs and wargames.
To keep within the spirit of the wargame, let's first go over how you might
solve this challenge in the terminal. One of the recommended commands is tr
which can "[t]ranslate, squeeze, and/or delete characters" according to the
description. Here's a few different ways to go about it.
# Rotate the alphabet by 13 letters where both input and output are explicit
cat data.txt | tr "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ" "nopqrstuvwxyzabcdefghijklmNOPQRSTUVWXYZABCDEFGHIJKLM"
# Similar to above except the input is defined by a regular expression
cat data.txt | tr "a-zA-Z" "nopqrstuvwxyzabcdefghijklmNOPQRSTUVWXYZABCDEFGHIJKLM"
# Similar to above except both the input and output are defined by a regular expression
# Note: the odd arrangement for the output is the necessary since Regex doesn't allow
# the letter ranges to wrap around
cat data.txt | tr "a-zA-Z" "n-za-mN-ZA-M"
While the tr
command is cool and all, there's an even cooler tool you should
be using when investigating any challenge related to cryptography. And that's
CyberChef. CyberChef has a huge number of
useful features for transforming data and supports hundreds of data formats and
encoding schemes.
Check out the CyberChef ROT13 cipher solver.
Level 12
The password for the next level is stored in the file
data.txt
, which is a hexdump of a file that has been repeatedly compressed. For this level it may be useful to create a directory under/tmp
in which you can work. Usemkdir
with a hard to guess directory name. Or better, use the commandmktemp -d
. Then copy the datafile usingcp
, and rename it usingmv
(read the manpages!)
This challenge is quite tedious. As the prompt mentions, it's a good idea to
create a temp directory to work with all the files. For example mkdir /tmp/my-super-secret-directory
followed by mv ~/data.txt /tmp/my-super-secret-directory
. Now we're ready to begin.
The first step is to recognize the format of the data.txt
.
Taking a look at the first few lines shows that this file isn't just a text
file. It's a hexdump. Read the first few lines with head -n5 data.txt
.
1 00000000: 1f8b 0808 dfcd eb66 0203 6461 7461 322e .......f..data2.
2 00000010: 6269 6e00 013e 02c1 fd42 5a68 3931 4159 bin..>...BZh91AY
3 00000020: 2653 59ca 83b2 c100 0017 7fff dff3 f4a7 &SY.............
4 00000030: fc9f fefe f2f3 cffe f5ff ffdd bf7e 5bfe .............~[.
5 00000040: faff dfbe 97aa 6fff f0de edf7 b001 3b56 ......o.......;V
Don't panic! You don't need to be able to read this stuff right away. Just recognizing it as a hexdump is enough. Fortunately, one of the recommended commands is made specifically to handle hexdumps. The xxd command.
Using xxd
with the -r
flag can reverse the hexdump into a binary file.
xxd -r data.txt > data
This output data
file is now in it's original format and can be examined with
file data
to determine it's type.
data: gzip compressed data, was "data2.bin", last modified: Thu Sep 19 07:08:15 2024, max compression, from Unix, original size modulo 2^32 574
The result from file
identifies it as "gzip compressed data". To decompress
the archive, use the gunzip
command.
Be aware, that the archive utilities like gunzip
may require particular file
extensions when decompressing files. For example, .gzip
or .gz
. Otherwise
you may get an error like this.
gzip: data: unknown suffix -- ignored
To complete this challenge, you must repeat this process of decompressing or
extracting data into a new format, then verifying the new format with file
eight times to reach the original flag file content.
Below is a script describing each step of the decompression. You could run the script directly on the Bandit host to get the flag, but I encourage you to walk through each decompression step manually and observe the different flags being used for each command.
#!/bin/sh
# extract.sh
xxd -r data.txt > f1.gz; # extract first gzip archive from hexdump
gunzip -c f1.gz > f2.bz2; # extract bzip2 archive from f1.gz
bunzip2 -c f2.bz2 > f3.gz; # extract gzip archive from f2.bz2
gunzip -c f3.gz > f4.tar; # extract tar archive from f3.gz
tar -xOf f4.tar > f5.tar; # extract tar archive from f4.tar
tar -xOf f5.tar > f6.bz2; # extract bzip2 archive from f5.tar
bunzip2 -c f6.bz2 > f7.tar; # extract tar archive from f6.bz2
tar -xOf f7.tar > f8.gz; # extract gzip archive from f7.tar
gunzip -c f8.gz > flag; # extract plaintext flag file from f8.gz
cat flag;
Level 13
The password for the next level is stored in
/etc/bandit_pass/bandit14
and can only be read by userbandit14
. For this level, you don’t get the next password, but you get a private SSH key that can be used to log into the next level.Note:
localhost
is a hostname that refers to the machine you are working on
Once again, OverTheWire has provided some new recommended commands to investigate.
Command | Description |
---|---|
ssh | a program for logging into or executing commands on remote machines |
telnet | communicate with another host using the TELNET protocol |
nc | the swiss army knife for communicating over the network using TCP, UDP, or Unix-domain sockets |
openssl | a program for using various cryptography functions of the OpenSSL crypto library from the shell |
s_client | a program implementing a generic SSL/TLS client |
nmap | a network scanner for network exploration and security auditing |
Each program is worth exploring, but for this challenge we'll only need ssh
.
However, it won't be used quite the same as before. This time an SSH private
key is required to login to the next level.
After logging in to bandit13
, you should find the private SSH key mentioned
in the prompt at /home/bandit13/sshkey.private
. Copy that file to your
primary host. You can simply copy paste the text if you wish, or use something
like scp.
Watch out! The ssh
command requires private key files to have appropriate
permissions. If they aren't correct you may receive an error message like this.
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@ WARNING: UNPROTECTED PRIVATE KEY FILE! @
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
Permissions 0644 for 'sshkey.private' are too open.
It is required that your private key files are NOT accessible by others.
This private key will be ignored.
Load key "sshkey.private": bad permissions
A brief reminder of the octal permissions.
Number | Permission Type | Symbols |
---|---|---|
0 | No permissions | --- |
1 | Execute | --x |
2 | Write | -w- |
3 | Write + Execute | -wx |
4 | Read | r-- |
5 | Read + Execute | r-x |
6 | Read + Write | rw- |
7 | Read + Write + Execute | rwx |
As the error mentions, the permissions 0644
are too open. That's because
private key files should only be readable and/or writeable by the user they
belong to. Usually that means either 600
or 400
, though 400
is a bit
strict and won't allow the file to edited.
So, setting the permissions to 600
would give the key file read and write
access for the user, and no permissions for either the group or others.
chmod 600 sshkey.private
Now we're ready to connect to bandit14
.
To use a key file with ssh
, the -i
flag can be used.
E.g.
ssh user@host -i private_key
To connect to bandi14
use the following command.
ssh bandit14@bandit.labs.overthewire.org -p 2220 -i sshkey.private
SSH is the most common protocol used for remotely administrating Linux and Unix-like systems, and using a key file as we've done here is by far the most common way it's used. It's not strictly necessary, but I highly recommend reading up on the fundamentals of public-key cryptography. It's how SSH guarantees[1] it's security.
In the solution above, we just used the -i
flag to specify the private key
file. However, anyone using ssh
on a regular basis will rightly tell you to
consider configuring ssh on your system for a much simpler workflow.
If you're connecting to many hosts via ssh, it is much more convenient to
configure the
ssh-agent
to handle your ssh keys automatically so you don't need to specifiy the key
file with the -i
flag each time.
You may also want to combine this with host-specific configurations. This can
be done with a config
file usually at ~/.ssh/config
. Check out the
ssh_config
manual
for more details.
To be continued
I hope you enjoyed the walkthrough. When time permits, I intend to expand this post to include every level of OverTheWire Bandit.
Happy hacking!
-
Just like almost every cryptographic system, it's possible for SSH to be used incorrectly, thus compromising it's security. ↩