2019-02-15 20:21:44 +00:00
|
|
|
#!/usr/bin/env perl
|
|
|
|
|
|
|
|
##
|
|
|
|
## Author......: See docs/credits.txt
|
|
|
|
## License.....: MIT
|
|
|
|
##
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
2020-02-10 12:00:44 +00:00
|
|
|
use Digest::SHA qw (sha256);
|
2019-02-15 20:21:44 +00:00
|
|
|
|
2020-02-10 12:00:44 +00:00
|
|
|
sub module_constraints { [[0, 256], [0, 20], [0, 15], [0, 20], [-1, -1]] }
|
2019-02-15 20:21:44 +00:00
|
|
|
|
2019-02-17 08:37:50 +00:00
|
|
|
sub module_generate_hash
|
2019-02-15 20:21:44 +00:00
|
|
|
{
|
2019-02-17 08:37:50 +00:00
|
|
|
my $word = shift;
|
2019-02-15 20:21:44 +00:00
|
|
|
my $salt = shift;
|
2019-02-17 08:37:50 +00:00
|
|
|
my $iter = shift;
|
2019-02-15 20:21:44 +00:00
|
|
|
|
|
|
|
my $hash_buf;
|
|
|
|
|
2019-02-17 08:37:50 +00:00
|
|
|
if (defined $iter)
|
2019-02-15 20:21:44 +00:00
|
|
|
{
|
2020-02-10 12:00:44 +00:00
|
|
|
$hash_buf = sha256crypt ($word, $salt, $iter, 1);
|
2019-02-15 20:21:44 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-02-10 12:00:44 +00:00
|
|
|
$hash_buf = sha256crypt ($word, $salt, 5000, 0);
|
2019-02-15 20:21:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
my $hash = sprintf ("%s", $hash_buf);
|
|
|
|
|
|
|
|
return $hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub module_verify_hash
|
|
|
|
{
|
|
|
|
my $line = shift;
|
|
|
|
|
|
|
|
my $index1 = index ($line, ":", 30);
|
|
|
|
|
|
|
|
return if $index1 < 1;
|
|
|
|
|
|
|
|
my $hash_in = substr ($line, 0, $index1);
|
|
|
|
|
|
|
|
my $word = substr ($line, $index1 + 1);
|
|
|
|
|
|
|
|
$index1 = index ($hash_in, ",", 1);
|
|
|
|
|
|
|
|
my $index2 = index ($hash_in, "\$", 1);
|
|
|
|
|
|
|
|
if ($index1 != -1)
|
|
|
|
{
|
|
|
|
if ($index1 < $index2)
|
|
|
|
{
|
|
|
|
$index2 = $index1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#$param = substr ($hash_in, $index2, 1);
|
|
|
|
|
|
|
|
$index2++;
|
|
|
|
|
|
|
|
# rounds= if available
|
|
|
|
my $iter;
|
|
|
|
|
|
|
|
if (substr ($hash_in, $index2, 7) eq "rounds=")
|
|
|
|
{
|
|
|
|
my $old_index = $index2;
|
|
|
|
|
|
|
|
$index2 = index ($hash_in, "\$", $index2 + 1);
|
|
|
|
|
|
|
|
return if $index2 < 1;
|
|
|
|
|
|
|
|
$iter = substr ($hash_in, $old_index + 7, $index2 - $old_index - 7);
|
|
|
|
|
|
|
|
$index2++;
|
|
|
|
}
|
|
|
|
|
|
|
|
# get salt
|
|
|
|
my $index3 = rindex ($hash_in, "\$");
|
|
|
|
|
|
|
|
return if $index3 < 1;
|
|
|
|
|
|
|
|
my $salt = substr ($hash_in, $index2, $index3 - $index2);
|
|
|
|
|
|
|
|
return unless defined $salt;
|
|
|
|
return unless defined $word;
|
|
|
|
|
|
|
|
$word = pack_if_HEX_notation ($word);
|
|
|
|
|
|
|
|
my $new_hash = module_generate_hash ($word, $salt, $iter);
|
|
|
|
|
|
|
|
return ($new_hash, $word);
|
|
|
|
}
|
|
|
|
|
2020-02-11 10:36:16 +00:00
|
|
|
# This is a modified sha_crypts () function of pass_gen.pl from
|
|
|
|
# https://github.com/magnumripper/JohnTheRipper/blob/bleeding-jumbo/run/pass_gen.pl
|
2020-02-10 12:00:44 +00:00
|
|
|
|
|
|
|
# Copyright: https://github.com/magnumripper/JohnTheRipper/blob/bleeding-jumbo/doc/pass_gen.Manifest
|
|
|
|
# public domain
|
|
|
|
# written by Jim Fougeron
|
|
|
|
|
|
|
|
# updated for new MySQL hashes by philsmd
|
|
|
|
# modified date: February 2020
|
|
|
|
# license: public domain
|
|
|
|
|
|
|
|
my @i64 = ('.', '/', '0'..'9', 'A'..'Z', 'a'..'z');
|
|
|
|
|
|
|
|
sub to64
|
|
|
|
{
|
|
|
|
my $v = shift;
|
|
|
|
my $n = shift;
|
|
|
|
|
|
|
|
my $str;
|
|
|
|
|
|
|
|
while (--$n >= 0)
|
|
|
|
{
|
|
|
|
$str .= $i64[$v & 0x3F];
|
|
|
|
|
|
|
|
$v >>= 6;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $str;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub sha_crypts
|
|
|
|
{
|
|
|
|
my ($func, $bits, $key, $salt, $loops) = @_;
|
|
|
|
|
|
|
|
my $bytes = $bits / 8;
|
|
|
|
|
|
|
|
my $b = $func->($key . $salt . $key);
|
|
|
|
|
|
|
|
# Add for any character in the key one byte of the alternate sum.
|
|
|
|
|
|
|
|
my $tmp = $key . $salt;
|
|
|
|
|
|
|
|
for (my $i = length ($key); $i > 0; $i -= $bytes)
|
|
|
|
{
|
|
|
|
if ($i > $bytes)
|
|
|
|
{
|
|
|
|
$tmp .= $b;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$tmp .= substr ($b, 0, $i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Take the binary representation of the length of the key and for every 1 add the alternate sum, for every 0 the key.
|
|
|
|
|
|
|
|
for (my $i = length ($key); $i > 0; $i >>= 1)
|
|
|
|
{
|
|
|
|
if (($i & 1) != 0)
|
|
|
|
{
|
|
|
|
$tmp .= $b;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$tmp .= $key;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my $a = $func->($tmp);
|
|
|
|
|
|
|
|
# NOTE, this will be the 'initial' $c value in the inner loop.
|
|
|
|
|
|
|
|
# For every character in the password add the entire password. produces DP
|
|
|
|
|
|
|
|
$tmp = "";
|
|
|
|
|
|
|
|
for (my $i = 0; $i < length ($key); $i++)
|
|
|
|
{
|
|
|
|
$tmp .= $key;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $dp = $func->($tmp);
|
|
|
|
|
|
|
|
# Create byte sequence P
|
|
|
|
|
|
|
|
my $p = "";
|
|
|
|
|
|
|
|
for (my $i = length ($key); $i > 0; $i -= $bytes)
|
|
|
|
{
|
|
|
|
if ($i > $bytes)
|
|
|
|
{
|
|
|
|
$p .= $dp;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$p .= substr ($dp, 0, $i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# produce ds
|
|
|
|
|
|
|
|
$tmp = "";
|
|
|
|
|
|
|
|
my $til = 16 + ord (substr ($a, 0, 1));
|
|
|
|
|
|
|
|
for (my $i = 0; $i < $til; $i++)
|
|
|
|
{
|
|
|
|
$tmp .= $salt;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $ds = $func->($tmp);
|
|
|
|
|
|
|
|
# Create byte sequence S
|
|
|
|
|
|
|
|
my $s = "";
|
|
|
|
|
|
|
|
for (my $i = length ($salt); $i > 0; $i -= $bytes)
|
|
|
|
{
|
|
|
|
if ($i > $bytes)
|
|
|
|
{
|
|
|
|
$s .= $ds;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$s .= substr ($ds, 0, $i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my $c = $a; # Ok, we saved this, which will 'seed' our crypt value here in the loop.
|
|
|
|
|
|
|
|
# now we do 5000 iterations of SHA2 (256 or 512)
|
|
|
|
|
|
|
|
for (my $i = 0; $i < $loops; $i++)
|
|
|
|
{
|
|
|
|
if ($i & 1) { $tmp = $p; }
|
|
|
|
else { $tmp = $c; }
|
2019-02-15 20:21:44 +00:00
|
|
|
|
2020-02-10 12:00:44 +00:00
|
|
|
if ($i % 3) { $tmp .= $s; }
|
|
|
|
if ($i % 7) { $tmp .= $p; }
|
|
|
|
|
|
|
|
if ($i & 1) { $tmp .= $c; }
|
|
|
|
else { $tmp .= $p; }
|
|
|
|
|
|
|
|
$c = $func->($tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
my $inc1; my $inc2; my $mod; my $end;
|
|
|
|
|
|
|
|
if ($bits == 256) { $inc1 = 10; $inc2 = 21; $mod = 30; $end = 0; }
|
|
|
|
else { $inc1 = 21; $inc2 = 22; $mod = 63; $end = 21; }
|
|
|
|
|
|
|
|
my $i = 0;
|
|
|
|
$tmp = "";
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
$tmp .= to64 ((ord (substr ($c, $i, 1)) << 16) | (ord (substr ($c, ($i + $inc1) % $mod, 1)) << 8) | ord (substr ($c, ($i + $inc1 * 2) % $mod, 1)), 4);
|
|
|
|
$i = ($i + $inc2) % $mod;
|
|
|
|
} while ($i != $end);
|
|
|
|
|
|
|
|
if ($bits == 256) { $tmp .= to64 ((ord (substr ($c, 31, 1)) << 8) | ord (substr ($c, 30, 1)), 3); }
|
|
|
|
else { $tmp .= to64 (ord (substr ($c, 63, 1)), 2); }
|
|
|
|
|
|
|
|
return $tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub sha256crypt
|
|
|
|
{
|
|
|
|
my $pass = shift;
|
|
|
|
my $salt = shift;
|
|
|
|
my $iter = shift;
|
|
|
|
my $rounds = shift;
|
|
|
|
|
|
|
|
my $bin = sha_crypts (\&sha256, 256, $pass, $salt, $iter);
|
|
|
|
|
|
|
|
if ($rounds == 1)
|
|
|
|
{
|
|
|
|
return "\$5\$rounds=$iter\$" . $salt . "\$$bin";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return "\$5\$" . $salt . "\$$bin";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|