Reversible Rovnix passwords
I got my hands on Rovnix recently.
Not the one who got leaked with Carberp but the ISFB package part (Core, Interceptor, ATS, VNC modules, etc...):
data:image/s3,"s3://crabby-images/11afa/11afa2e5a47058b2aa14f533a4e02c7571ae784d" alt=""
And the panel..
data:image/s3,"s3://crabby-images/20a3c/20a3ce921809c71d3d97bb22bb4c5b6d27b8bb6c" alt=""
Ive already took some screenshots of inside Rovnix, so lets skip about the screenshots part.
Just check this article if you want see pics from the Rovnix C&C:
http://www.xylibox.com/2012/02/win32rovnix.html?spref=tw
The panel come with a sql dump, and a user/password is already defined inside.
The password looks like a MD5 hash and we know nothing about it.
data:image/s3,"s3://crabby-images/30c71/30c71af8cdf793f34995a3ae09f7abbbf50d1131" alt=""
SQL tables:
data:image/s3,"s3://crabby-images/dc605/dc605814629cf61848a8b8b2fff717127936836a" alt=""
By looking the hash on Google we have a correspondence with 21240:
data:image/s3,"s3://crabby-images/3efaa/3efaae4c9ad4203f52f625ab6ad93b23d6a0fa9c" alt=""
A tool confirm also that the hash is good for 21240
data:image/s3,"s3://crabby-images/dee1e/dee1eccd2b6e73acd74e8634e565aa5e5a7acb89" alt=""
But.. there is a problem somewhere:
data:image/s3,"s3://crabby-images/0a30f/0a30fff7ed69bceebe3ec129722dd93d59c52783" alt=""
So we have to check the code to see whats going on.
admin/index.php use a function getMyHash()
data:image/s3,"s3://crabby-images/99397/99397cb9961601405e5e0393dbac4c32ecc12a4e" alt=""
This small function can be found inside mod/main.php:
data:image/s3,"s3://crabby-images/37b13/37b13451398359723e2ebf3df33815246ced921b" alt=""
We have a salt and they use md5() but we have a huge mistake here:
data:image/s3,"s3://crabby-images/1557c/1557cde0929d4dad19b4d611a909a066cac388ab" alt=""
So if we try to hash a password composed only of numbers, we will have a obvious problem.
Like its the case for the default password found inside the sql dump.
If you want an example:
310dcbbf4cce62f762a2aaa148d556bd = getMyHash(123)
310dcbbf4cce62f762a2aaa148d556bd = md5(333)
collision with 2 algo.
We can obtain the password from the hash easily, PoC:
<?php
/**
* Defeat the weak hash function of Rovnix
* to get password from a hash.
*/
$HASH = fbff791ef0770855e599ea6f87d41653;
$value = getNumber($HASH);
$search = search($value, $HASH);
echo(Hash: . $HASH . <br />);
echo(Value: . $value . <br />);
echo(Search: . $search);
// Search an working (number) password
function search($value, $hash) {
$i = 0;
while (true) {
if (getHash($i) == $value)
return $i;
$i++;
}
}
// Get the hashed number
function getNumber($hash) {
$i = 0;
while (true) {
if (md5($i) == $hash)
return $i;
$i++;
}
}
// Hash function without final MD5 (return only numbers)
function getHash($hash) {
$salt = LKJFDJLJkkljKJKJKJkjkj$i%&@(%jkjJn@@j$r@!cdh*!@#$djl1J$r!j@o*$@duJxlJLEKJkJFKJEJ2$jkeJFJLEJFE;
return $hash + $salt + md5($salt) + md5($hash) + $salt[3];
}
?>
/**
* Defeat the weak hash function of Rovnix
* to get password from a hash.
*/
$HASH = fbff791ef0770855e599ea6f87d41653;
$value = getNumber($HASH);
$search = search($value, $HASH);
echo(Hash: . $HASH . <br />);
echo(Value: . $value . <br />);
echo(Search: . $search);
// Search an working (number) password
function search($value, $hash) {
$i = 0;
while (true) {
if (getHash($i) == $value)
return $i;
$i++;
}
}
// Get the hashed number
function getNumber($hash) {
$i = 0;
while (true) {
if (md5($i) == $hash)
return $i;
$i++;
}
}
// Hash function without final MD5 (return only numbers)
function getHash($hash) {
$salt = LKJFDJLJkkljKJKJKJkjkj$i%&@(%jkjJn@@j$r@!cdh*!@#$djl1J$r!j@o*$@duJxlJLEKJkJFKJEJ2$jkeJFJLEJFE;
return $hash + $salt + md5($salt) + md5($hash) + $salt[3];
}
?>
Output for the unknown hash:
data:image/s3,"s3://crabby-images/a4091/a4091acb6d9c4533d0e2819e35eb378c36decd4b" alt=""
So the unknown password for fbff791ef0770855e599ea6f87d41653 is in reality 21173.
Lets try..
data:image/s3,"s3://crabby-images/f36a4/f36a4587952a212011007b17ed8a179ce1c85b3a" alt=""
download file now