2017-03-31 15:50:31 +13:00
# Tests for Tests for source4/dsdb/samdb/ldb_modules/password_hash.c
#
# Copyright (C) Catalyst IT Ltd. 2017
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
"""
Tests for source4 / dsdb / samdb / ldb_modules / password_hash . c
These tests need to be run in an environment in which
io - > ac - > gpg_key_ids == NULL , so that the gpg supplemental credentials
are not generated . And also need to be in an environment with a
functional level less than 2008 to ensure the kerberos newer keys are not
generated
"""
from samba . tests . password_hash import (
PassWordHashTests ,
get_package ,
USER_PASS
)
from samba . ndr import ndr_unpack
from samba . dcerpc import drsblobs
import binascii
class PassWordHashFl2003Tests ( PassWordHashTests ) :
def setUp ( self ) :
super ( PassWordHashFl2003Tests , self ) . setUp ( )
def test_default_supplementalCredentials ( self ) :
2017-04-12 09:12:56 +12:00
self . add_user ( options = [ ( " password hash userPassword schemes " , " " ) ] )
2017-03-31 15:50:31 +13:00
sc = self . get_supplemental_creds ( )
# Check that we got all the expected supplemental credentials
# And they are in the expected order.
size = len ( sc . sub . packages )
self . assertEquals ( 3 , size )
( pos , package ) = get_package ( sc , " Primary:Kerberos " )
self . assertEquals ( 1 , pos )
self . assertEquals ( " Primary:Kerberos " , package . name )
( pos , package ) = get_package ( sc , " Packages " )
self . assertEquals ( 2 , pos )
self . assertEquals ( " Packages " , package . name )
( pos , package ) = get_package ( sc , " Primary:WDigest " )
self . assertEquals ( 3 , pos )
self . assertEquals ( " Primary:WDigest " , package . name )
# Check that the WDigest values are correct.
#
digests = ndr_unpack ( drsblobs . package_PrimaryWDigestBlob ,
binascii . a2b_hex ( package . data ) )
self . check_wdigests ( digests )
2017-04-12 09:12:56 +12:00
def test_userPassword_sha256 ( self ) :
self . add_user ( options = [ ( " password hash userPassword schemes " ,
" CryptSHA256 " ) ] )
sc = self . get_supplemental_creds ( )
# Check that we got all the expected supplemental credentials
# And they are in the expected order.
size = len ( sc . sub . packages )
self . assertEquals ( 4 , size )
( pos , package ) = get_package ( sc , " Primary:Kerberos " )
self . assertEquals ( 1 , pos )
self . assertEquals ( " Primary:Kerberos " , package . name )
( pos , wd_package ) = get_package ( sc , " Primary:WDigest " )
self . assertEquals ( 2 , pos )
self . assertEquals ( " Primary:WDigest " , wd_package . name )
( pos , package ) = get_package ( sc , " Packages " )
self . assertEquals ( 3 , pos )
self . assertEquals ( " Packages " , package . name )
( pos , up_package ) = get_package ( sc , " Primary:userPassword " )
self . assertEquals ( 4 , pos )
self . assertEquals ( " Primary:userPassword " , up_package . name )
# Check that the WDigest values are correct.
#
digests = ndr_unpack ( drsblobs . package_PrimaryWDigestBlob ,
binascii . a2b_hex ( wd_package . data ) )
self . check_wdigests ( digests )
# Check that the userPassword hashes are computed correctly
#
up = ndr_unpack ( drsblobs . package_PrimaryUserPasswordBlob ,
binascii . a2b_hex ( up_package . data ) )
self . checkUserPassword ( up , [ ( " {CRYPT} " , " 5 " , None ) ] )
self . checkNtHash ( USER_PASS , up . current_nt_hash . hash )
2017-03-31 15:50:31 +13:00
def test_supplementalCredentials_cleartext ( self ) :
2017-04-12 09:12:56 +12:00
self . add_user ( clear_text = True ,
options = [ ( " password hash userPassword schemes " , " " ) ] )
2017-03-31 15:50:31 +13:00
sc = self . get_supplemental_creds ( )
# Check that we got all the expected supplemental credentials
# And they are in the expected order.
size = len ( sc . sub . packages )
self . assertEquals ( 4 , size )
( pos , package ) = get_package ( sc , " Primary:Kerberos " )
self . assertEquals ( 1 , pos )
self . assertEquals ( " Primary:Kerberos " , package . name )
( pos , wd_package ) = get_package ( sc , " Primary:WDigest " )
self . assertEquals ( 2 , pos )
self . assertEquals ( " Primary:WDigest " , wd_package . name )
( pos , package ) = get_package ( sc , " Packages " )
self . assertEquals ( 3 , pos )
self . assertEquals ( " Packages " , package . name )
( pos , ct_package ) = get_package ( sc , " Primary:CLEARTEXT " )
self . assertEquals ( 4 , pos )
self . assertEquals ( " Primary:CLEARTEXT " , ct_package . name )
2017-04-12 09:12:56 +12:00
# Check that the WDigest values are correct.
#
digests = ndr_unpack ( drsblobs . package_PrimaryWDigestBlob ,
binascii . a2b_hex ( wd_package . data ) )
self . check_wdigests ( digests )
# Check the clear text value is correct.
ct = ndr_unpack ( drsblobs . package_PrimaryCLEARTEXTBlob ,
binascii . a2b_hex ( ct_package . data ) )
self . assertEquals ( USER_PASS . encode ( ' utf-16-le ' ) , ct . cleartext )
def test_userPassword_cleartext_sha512 ( self ) :
self . add_user ( clear_text = True ,
options = [ ( " password hash userPassword schemes " ,
" CryptSHA512:rounds=10000 " ) ] )
sc = self . get_supplemental_creds ( )
# Check that we got all the expected supplemental credentials
# And they are in the expected order.
size = len ( sc . sub . packages )
self . assertEquals ( 5 , size )
( pos , package ) = get_package ( sc , " Primary:Kerberos " )
self . assertEquals ( 1 , pos )
self . assertEquals ( " Primary:Kerberos " , package . name )
( pos , wd_package ) = get_package ( sc , " Primary:WDigest " )
self . assertEquals ( 2 , pos )
self . assertEquals ( " Primary:WDigest " , wd_package . name )
( pos , ct_package ) = get_package ( sc , " Primary:CLEARTEXT " )
self . assertEquals ( 3 , pos )
self . assertEquals ( " Primary:CLEARTEXT " , ct_package . name )
( pos , package ) = get_package ( sc , " Packages " )
self . assertEquals ( 4 , pos )
self . assertEquals ( " Packages " , package . name )
( pos , up_package ) = get_package ( sc , " Primary:userPassword " )
self . assertEquals ( 5 , pos )
self . assertEquals ( " Primary:userPassword " , up_package . name )
2017-03-31 15:50:31 +13:00
# Check that the WDigest values are correct.
#
digests = ndr_unpack ( drsblobs . package_PrimaryWDigestBlob ,
binascii . a2b_hex ( wd_package . data ) )
self . check_wdigests ( digests )
# Check the clear text value is correct.
ct = ndr_unpack ( drsblobs . package_PrimaryCLEARTEXTBlob ,
binascii . a2b_hex ( ct_package . data ) )
self . assertEquals ( USER_PASS . encode ( ' utf-16-le ' ) , ct . cleartext )
2017-04-12 09:12:56 +12:00
# Check that the userPassword hashes are computed correctly
#
up = ndr_unpack ( drsblobs . package_PrimaryUserPasswordBlob ,
binascii . a2b_hex ( up_package . data ) )
2018-07-30 18:19:05 +12:00
self . checkUserPassword ( up , [ ( " {CRYPT} " , " 6 " , 10000 ) ] )
2017-04-12 09:12:56 +12:00
self . checkNtHash ( USER_PASS , up . current_nt_hash . hash )