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.

i !
Note

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!

i !
Warning

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.

  1. Search for examples. Many man pages contain examples of their usage. These may be marked by all-caps EXAMPLES in some cases, but sometimes not.
  2. 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 of ssh usage. This one can be helpful when trying to understand how a command is commonly used and which command flags are most useful.
  3. 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.

i !
Note

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 form ssh://[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.

i !
Note

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.

CommandDescription
lslist the files in a directory
cdchange your current directory
catdisplay the contents of a file
filedisplay information about the content of a file
dudisplay the disk space used by files
findsearch 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.

  1. 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 to cat.

    $ cat ./-
    [REDACTED PASSWORD]
    
  2. 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 the pwd 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]
    
  3. 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 to cat.

    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.

i !
Tip

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 the home 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.

  1. 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]
    
  2. 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]
    
  3. 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]
    
i !
Tip

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]
i !
Note

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.

i !
Tip

Some systems will have aliases for the ls command as well, with various flags enabled. Here a few common ones.

  • ll for ls -AlhF to list all files in a human-readable, long format
  • la for ls -A to list all files excluding . and ..
  • l for ls -CF to list files in a column format
  • lsd for ls --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 the reset 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]
i !
Tip

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.

i !
Note

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.

CommandDescription
manaccess the system reference manuals
grepprint lines that match patterns
sortsort lines in text files
uniqremove duplicate lines from a file
stringsprint readable strings from arbitrary files (even binary)
base64encode data into Base64
trtranslate and replace characters
tara utility for archive files
gzipa utility for compressing files
bzip2a utility for compressing files
xxda 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.

i !
Tip

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.

i !
Tip

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]
i !
Note

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"
i !
Tip

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. Use mkdir with a hard to guess directory name. Or better, use the command mktemp -d. Then copy the datafile using cp, and rename it using mv (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.

100000000: 1f8b 0808 dfcd eb66 0203 6461 7461 322e .......f..data2.
200000010: 6269 6e00 013e 02c1 fd42 5a68 3931 4159 bin..>...BZh91AY
300000020: 2653 59ca 83b2 c100 0017 7fff dff3 f4a7 &SY.............
400000030: fc9f fefe f2f3 cffe f5ff ffdd bf7e 5bfe .............~[.
500000040: 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.

i !
Warning

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 user bandit14. 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.

CommandDescription
ssha program for logging into or executing commands on remote machines
telnetcommunicate with another host using the TELNET protocol
ncthe swiss army knife for communicating over the network using TCP, UDP, or Unix-domain sockets
openssla program for using various cryptography functions of the OpenSSL crypto library from the shell
s_clienta program implementing a generic SSL/TLS client
nmapa 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.

i !
Warning

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.

NumberPermission TypeSymbols
0No permissions---
1Execute--x
2Write-w-
3Write + Execute-wx
4Readr--
5Read + Executer-x
6Read + Writerw-
7Read + Write + Executerwx

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.

i !
Note

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 !
Note

I hope you enjoyed the walkthrough. When time permits, I intend to expand this post to include every level of OverTheWire Bandit.

Happy hacking!


  1. Just like almost every cryptographic system, it's possible for SSH to be used incorrectly, thus compromising it's security.