1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-10 01:18:15 +03:00
samba-mirror/lib/crypto/test_gkdi.c
Jo Sutton 02d7ab13ee lib:crypto: Add more unit tests for GKDI functions
Signed-off-by: Jo Sutton <josutton@catalyst.net.nz>
Reviewed-by: Andrew Bartlett <abartlet@samba.org>
2024-04-21 22:10:36 +00:00

334 lines
10 KiB
C
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* Unix SMB/CIFS implementation.
*
* Copyright (C) Catalyst.Net Ltd 2024
*
* 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 <https://www.gnu.org/licenses/>.
*/
#include <stdarg.h>
#include <stddef.h>
#include <stdint.h>
#include <setjmp.h>
#include <cmocka.h>
#include "replace.h"
#include <talloc.h>
#include "libcli/util/ntstatus.h"
#include "lib/crypto/gkdi.h"
#include "lib/crypto/gmsa.h"
#include "librpc/ndr/libndr.h"
#include "librpc/gen_ndr/gmsa.h"
#include "librpc/gen_ndr/ndr_gmsa.h"
#include "lib/util/time.h"
#include "libcli/security/dom_sid.h"
static void test_password_based_on_key_id(void **state)
{
static const uint8_t root_key_data[] = {
152, 203, 215, 84, 113, 216, 118, 177, 81, 128, 50, 160, 148,
132, 82, 244, 65, 179, 164, 219, 209, 14, 33, 131, 178, 193,
80, 248, 126, 23, 66, 227, 45, 221, 171, 12, 247, 15, 62,
179, 164, 217, 123, 179, 106, 118, 228, 74, 12, 2, 241, 229,
139, 55, 237, 155, 220, 122, 200, 245, 129, 222, 37, 15};
static const struct ProvRootKey root_key = {
.version = root_key_version_1,
.id = {.time_low = 0x170e972,
.time_mid = 0xa035,
.time_hi_and_version = 0xaf4c,
.clock_seq = {0x99, 0x3b},
.node = {0xe0, 0x52, 0xd5, 0xbd, 0x12, 0x16}},
.data =
{
.data = discard_const_p(uint8_t, root_key_data),
.length = sizeof root_key_data,
},
.create_time = 0,
.use_start_time = 0,
.domain_id = "example.com",
.kdf_algorithm = {
.id = KDF_ALGORITHM_SP800_108_CTR_HMAC,
.param.sp800_108 = KDF_PARAM_SHA512,
}};
static const uint8_t expected[GMSA_PASSWORD_LEN] = {
114, 92, 31, 204, 138, 249, 1, 76, 192, 65, 52, 248, 247,
191, 30, 213, 25, 38, 81, 21, 183, 167, 154, 102, 190, 234,
234, 116, 114, 18, 141, 208, 143, 38, 178, 115, 195, 26, 199,
214, 176, 229, 128, 160, 147, 249, 245, 67, 165, 191, 192, 78,
224, 50, 115, 8, 207, 124, 178, 121, 67, 135, 125, 113, 79,
0, 131, 43, 74, 48, 171, 239, 183, 228, 50, 212, 202, 215,
188, 182, 94, 127, 117, 217, 91, 17, 90, 80, 158, 176, 204,
151, 244, 107, 139, 65, 94, 148, 216, 212, 97, 53, 54, 253,
6, 201, 94, 93, 250, 213, 12, 82, 162, 246, 197, 254, 205,
8, 19, 153, 66, 72, 60, 167, 28, 65, 39, 218, 147, 82,
162, 11, 177, 78, 231, 200, 66, 121, 9, 196, 240, 7, 148,
190, 151, 96, 214, 246, 7, 110, 85, 0, 246, 28, 121, 3,
61, 212, 204, 101, 153, 121, 100, 91, 65, 28, 225, 241, 123,
115, 105, 138, 74, 187, 74, 188, 59, 17, 201, 229, 158, 170,
184, 141, 237, 179, 246, 135, 104, 204, 56, 228, 156, 182, 26,
90, 151, 147, 25, 142, 47, 172, 183, 165, 222, 240, 95, 63,
79, 88, 35, 205, 76, 26, 229, 107, 46, 16, 202, 102, 196,
18, 140, 211, 242, 226, 198, 154, 97, 199, 44, 220, 186, 76,
215, 54, 196, 44, 140, 145, 252, 99, 229, 179, 74, 150, 154,
70, 226, 45, 122, 156, 156, 75, 83, 24};
uint8_t out[GMSA_PASSWORD_NULL_TERMINATED_LEN];
TALLOC_CTX *mem_ctx = NULL;
mem_ctx = talloc_new(NULL);
assert_non_null(mem_ctx);
{
/* An arbitrary GKID. */
const struct Gkid gkid = {362, 0, 23};
/* An arbitrary time in the past. */
const NTTIME current_nt_time = 133524411756072082;
struct dom_sid account_sid;
NTSTATUS status;
bool ok;
ok = dom_sid_parse(
"S-1-5-21-4119591868-3001629103-3445594911-48026",
&account_sid);
assert_true(ok);
/* Derive a password from the root key. */
status = gmsa_password_based_on_key_id(mem_ctx,
gkid,
current_nt_time,
&root_key,
&account_sid,
out);
assert_true(NT_STATUS_IS_OK(status));
assert_memory_equal(expected, out, GMSA_PASSWORD_LEN);
}
{
uint64_t query_interval = 0;
uint64_t unchanged_interval = 0;
const struct MANAGEDPASSWORD_BLOB managed_pwd = {
.passwords = {
.current = out,
.query_interval = &query_interval,
.unchanged_interval = &unchanged_interval}};
DATA_BLOB packed_blob = {};
enum ndr_err_code err;
err = ndr_push_struct_blob(
&packed_blob,
mem_ctx,
&managed_pwd,
(ndr_push_flags_fn_t)ndr_push_MANAGEDPASSWORD_BLOB);
assert_int_equal(NDR_ERR_SUCCESS, err);
}
talloc_free(mem_ctx);
}
static void assert_gkid_equal(const struct Gkid g1, const struct Gkid g2)
{
assert_int_equal(g1.l0_idx, g2.l0_idx);
assert_int_equal(g1.l1_idx, g2.l1_idx);
assert_int_equal(g1.l2_idx, g2.l2_idx);
}
static void test_gkdi_rollover_interval(void **state)
{
NTTIME interval;
bool ok;
ok = gkdi_rollover_interval(0, &interval);
assert_true(ok);
assert_int_equal(0, interval);
ok = gkdi_rollover_interval(1, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(720000000000), interval);
ok = gkdi_rollover_interval(2, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(1440000000000), interval);
ok = gkdi_rollover_interval(3, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(2520000000000), interval);
ok = gkdi_rollover_interval(4, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(3240000000000), interval);
ok = gkdi_rollover_interval(5, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(4320000000000), interval);
ok = gkdi_rollover_interval(-1, &interval);
assert_false(ok);
ok = gkdi_rollover_interval(-2, &interval);
assert_false(ok);
ok = gkdi_rollover_interval(10675199, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(9223371720000000000), interval);
ok = gkdi_rollover_interval(-10675198, &interval);
assert_false(ok);
ok = gkdi_rollover_interval(10675200, &interval);
assert_true(ok);
assert_int_equal(UINT64_C(9223372800000000000), interval);
ok = gkdi_rollover_interval(-10675199, &interval);
assert_false(ok);
ok = gkdi_rollover_interval(21350398, &interval);
/*
* If we accepted this high of an interval, the result would be
* 18446743800000000000.
*/
assert_false(ok);
ok = gkdi_rollover_interval(-21350397, &interval);
assert_false(ok);
ok = gkdi_rollover_interval(21350399, &interval);
assert_false(ok); /* too large to be represented */
ok = gkdi_rollover_interval(-21350398, &interval);
assert_false(ok); /* too small to be represented */
}
static void assert_get_interval_id(const NTTIME time,
const struct Gkid expected_gkid)
{
{
const bool valid = gkid_is_valid(expected_gkid);
assert_true(valid);
}
{
const struct Gkid interval_id = gkdi_get_interval_id(time);
assert_gkid_equal(expected_gkid, interval_id);
}
}
static void test_get_interval_id(void **state)
{
assert_get_interval_id(0, Gkid(0, 0, 0));
assert_get_interval_id(gkdi_key_cycle_duration - 1, Gkid(0, 0, 0));
assert_get_interval_id(gkdi_key_cycle_duration, Gkid(0, 0, 1));
assert_get_interval_id(27 * gkdi_key_cycle_duration, Gkid(0, 0, 27));
assert_get_interval_id((gkdi_l2_key_iteration - 1) *
gkdi_key_cycle_duration,
Gkid(0, 0, gkdi_l2_key_iteration - 1));
assert_get_interval_id(gkdi_l2_key_iteration * gkdi_key_cycle_duration,
Gkid(0, 1, 0));
assert_get_interval_id(17 * gkdi_l2_key_iteration *
gkdi_key_cycle_duration,
Gkid(0, 17, 0));
assert_get_interval_id(((gkdi_l1_key_iteration - 1) *
gkdi_l2_key_iteration +
3) * gkdi_key_cycle_duration,
Gkid(0, gkdi_l1_key_iteration - 1, 3));
assert_get_interval_id(gkdi_l1_key_iteration * gkdi_l2_key_iteration *
gkdi_key_cycle_duration,
Gkid(1, 0, 0));
assert_get_interval_id(((1234 * gkdi_l1_key_iteration + 8) *
gkdi_l2_key_iteration +
13) * gkdi_key_cycle_duration,
Gkid(1234, 8, 13));
assert_get_interval_id(INT64_MAX, Gkid(25019, 31, 29));
assert_get_interval_id(UINT64_MAX, Gkid(50039, 31, 27));
}
static void test_get_key_start_time(void **state)
{
NTTIME start_time = 0;
bool ok;
/* Try passing an invalid GKID. */
ok = gkdi_get_key_start_time(invalid_gkid, &start_time);
assert_false(ok);
/* Try passing an L1 GKID rather than an L2 GKID. */
ok = gkdi_get_key_start_time(Gkid(0, 0, -1), &start_time);
assert_false(ok);
/* Test some L2 GKIDs. */
ok = gkdi_get_key_start_time(Gkid(0, 0, 0), &start_time);
assert_true(ok);
assert_int_equal(0, start_time);
ok = gkdi_get_key_start_time(Gkid(0, 0, 1), &start_time);
assert_true(ok);
assert_int_equal(gkdi_key_cycle_duration, start_time);
ok = gkdi_get_key_start_time(Gkid(123, 18, 2), &start_time);
assert_true(ok);
assert_int_equal(126530 * gkdi_key_cycle_duration, start_time);
ok = gkdi_get_key_start_time(Gkid(25019, 31, 29), &start_time);
assert_true(ok);
assert_int_equal(25620477 * gkdi_key_cycle_duration, start_time);
ok = gkdi_get_key_start_time(Gkid(25019, 31, 30), &start_time);
assert_true(ok);
assert_int_equal(UINT64_C(25620478) * gkdi_key_cycle_duration,
start_time);
ok = gkdi_get_key_start_time(Gkid(50039, 31, 27), &start_time);
assert_true(ok);
assert_int_equal(UINT64_C(51240955) * gkdi_key_cycle_duration,
start_time);
/*
* Test GKIDs so high that their start times cant be represented in
* NTTIME.
*/
ok = gkdi_get_key_start_time(Gkid(50039, 31, 28), &start_time);
assert_false(ok);
ok = gkdi_get_key_start_time(Gkid(INT32_MAX, 31, 31), &start_time);
assert_false(ok);
}
int main(int argc, char *argv[])
{
const struct CMUnitTest tests[] = {
cmocka_unit_test(test_password_based_on_key_id),
cmocka_unit_test(test_gkdi_rollover_interval),
cmocka_unit_test(test_get_interval_id),
cmocka_unit_test(test_get_key_start_time),
};
if (argc == 2) {
cmocka_set_test_filter(argv[1]);
}
cmocka_set_message_output(CM_OUTPUT_SUBUNIT);
return cmocka_run_group_tests(tests, NULL, NULL);
}