2
0
mirror of https://gitlab.isc.org/isc-projects/bind9 synced 2025-08-23 02:28:55 +00:00
bind/lib/isc/tests/lex_test.c

253 lines
6.1 KiB
C
Raw Normal View History

/*
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, you can obtain one at https://mozilla.org/MPL/2.0/.
*
* See the COPYRIGHT file distributed with this work for additional
* information regarding copyright ownership.
*/
2018-10-23 22:33:45 -07:00
#if HAVE_CMOCKA
#include <sched.h> /* IWYU pragma: keep */
#include <setjmp.h>
2018-10-23 22:33:45 -07:00
#include <stdarg.h>
#include <stddef.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
2018-10-23 22:33:45 -07:00
#define UNIT_TESTING
#include <cmocka.h>
#include <isc/buffer.h>
#include <isc/lex.h>
#include <isc/mem.h>
#include <isc/util.h>
#include "isctest.h"
2020-08-01 01:25:37 +10:00
static bool debug = false;
static int
2020-02-13 14:44:37 -08:00
_setup(void **state) {
isc_result_t result;
UNUSED(state);
result = isc_test_begin(NULL, true, 0);
assert_int_equal(result, ISC_R_SUCCESS);
return (0);
}
static int
2020-02-13 14:44:37 -08:00
_teardown(void **state) {
UNUSED(state);
isc_test_end();
return (0);
}
2018-10-23 22:33:45 -07:00
/* check handling of 0xff */
static void
2020-02-13 14:44:37 -08:00
lex_0xff(void **state) {
isc_result_t result;
2020-02-13 14:44:37 -08:00
isc_lex_t *lex = NULL;
isc_buffer_t death_buf;
2020-02-13 14:44:37 -08:00
isc_token_t token;
unsigned char death[] = { EOF, 'A' };
2018-10-23 22:33:45 -07:00
UNUSED(state);
result = isc_lex_create(test_mctx, 1024, &lex);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
isc_buffer_init(&death_buf, &death[0], sizeof(death));
isc_buffer_add(&death_buf, sizeof(death));
result = isc_lex_openbuffer(lex, &death_buf);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
result = isc_lex_gettoken(lex, 0, &token);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
2018-11-19 07:47:05 +01:00
isc_lex_destroy(&lex);
}
2018-10-23 22:33:45 -07:00
/* check setting of source line */
static void
2020-02-13 14:44:37 -08:00
lex_setline(void **state) {
isc_result_t result;
isc_lex_t *lex = NULL;
unsigned char text[] = "text\nto\nbe\nprocessed\nby\nlexer";
2020-02-13 14:44:37 -08:00
isc_buffer_t buf;
isc_token_t token;
unsigned long line;
2020-02-13 14:44:37 -08:00
int i;
2018-10-23 22:33:45 -07:00
UNUSED(state);
result = isc_lex_create(test_mctx, 1024, &lex);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
isc_buffer_init(&buf, &text[0], sizeof(text));
isc_buffer_add(&buf, sizeof(text));
result = isc_lex_openbuffer(lex, &buf);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
result = isc_lex_setsourceline(lex, 100);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
for (i = 0; i < 6; i++) {
result = isc_lex_gettoken(lex, 0, &token);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_SUCCESS);
line = isc_lex_getsourceline(lex);
2018-10-23 22:33:45 -07:00
assert_int_equal(line, 100U + i);
}
result = isc_lex_gettoken(lex, 0, &token);
2018-10-23 22:33:45 -07:00
assert_int_equal(result, ISC_R_EOF);
line = isc_lex_getsourceline(lex);
2018-10-23 22:33:45 -07:00
assert_int_equal(line, 105U);
2018-11-19 07:47:05 +01:00
isc_lex_destroy(&lex);
}
2020-08-01 01:25:37 +10:00
/*%
* keypair is <string>=<qstring>. This has implications double quotes
* in key names.
*/
static void
lex_keypair(void **state) {
isc_buffer_t buf;
isc_lex_t *lex = NULL;
isc_result_t result;
isc_token_t token;
size_t i;
struct {
const char *text;
const char *value;
isc_result_t result;
isc_tokentype_t type;
} tests[] = {
{ "", "", ISC_R_SUCCESS, isc_tokentype_eof },
{ "1234", "1234", ISC_R_SUCCESS, isc_tokentype_string },
{ "1234=", "1234=", ISC_R_SUCCESS, isc_tokentype_vpair },
{ "1234=foo", "1234=foo", ISC_R_SUCCESS, isc_tokentype_vpair },
{ "1234=\"foo", NULL, ISC_R_UNEXPECTEDEND, 0 },
{ "1234=\"foo\"", "1234=foo", ISC_R_SUCCESS,
isc_tokentype_qvpair },
{ "key", "key", ISC_R_SUCCESS, isc_tokentype_string },
{ "\"key=", "\"key=", ISC_R_SUCCESS, isc_tokentype_vpair },
{ "\"key=\"", NULL, ISC_R_UNEXPECTEDEND, 0 },
{ "key=\"\"", "key=", ISC_R_SUCCESS, isc_tokentype_qvpair },
{ "key=\"a b\"", "key=a b", ISC_R_SUCCESS,
isc_tokentype_qvpair },
{ "key=\"a\tb\"", "key=a\tb", ISC_R_SUCCESS,
isc_tokentype_qvpair },
/* double quote not immediately after '=' is not special. */
{ "key=c\"a b\"", "key=c\"a", ISC_R_SUCCESS,
isc_tokentype_vpair },
/* remove special meaning for '=' by escaping */
{ "key\\=", "key\\=", ISC_R_SUCCESS, isc_tokentype_string },
{ "key\\=\"a\"", "key\\=\"a\"", ISC_R_SUCCESS,
isc_tokentype_string },
{ "key\\=\"a \"", "key\\=\"a", ISC_R_SUCCESS,
isc_tokentype_string },
/* vpair with a key of 'key\=' (would need to be deescaped) */
{ "key\\==", "key\\==", ISC_R_SUCCESS, isc_tokentype_vpair },
/* qvpair with a key of 'key\=' (would need to be deescaped) */
{ "key\\==\"\"", "key\\==", ISC_R_SUCCESS,
isc_tokentype_qvpair },
};
UNUSED(state);
for (i = 0; i < ARRAY_SIZE(tests); i++) {
result = isc_lex_create(test_mctx, 1024, &lex);
assert_int_equal(result, ISC_R_SUCCESS);
isc_buffer_constinit(&buf, tests[i].text,
strlen(tests[i].text));
isc_buffer_add(&buf, strlen(tests[i].text));
result = isc_lex_openbuffer(lex, &buf);
assert_int_equal(result, ISC_R_SUCCESS);
result = isc_lex_setsourceline(lex, 100);
assert_int_equal(result, ISC_R_SUCCESS);
memset(&token, 0, sizeof(token));
result = isc_lex_getmastertoken(lex, &token,
isc_tokentype_qvpair, true);
if (debug) {
fprintf(stdout, "# '%s' -> result=%s/%s, type=%u/%u\n",
tests[i].text, isc_result_toid(result),
isc_result_toid(tests[i].result), token.type,
tests[i].type);
}
assert_int_equal(result, tests[i].result);
if (result == ISC_R_SUCCESS) {
switch (token.type) {
case isc_tokentype_string:
case isc_tokentype_qstring:
case isc_tokentype_vpair:
case isc_tokentype_qvpair:
if (debug) {
#define AS_STR(x) (x).value.as_textregion.base
fprintf(stdout, "# value='%s'\n",
AS_STR(token));
}
assert_int_equal(token.type, tests[i].type);
assert_string_equal(AS_STR(token),
tests[i].value);
break;
default:
assert_int_equal(token.type, tests[i].type);
break;
}
}
isc_lex_destroy(&lex);
}
}
2018-10-23 22:33:45 -07:00
int
2020-08-01 01:25:37 +10:00
main(int argc, char *argv[]) {
2018-10-23 22:33:45 -07:00
const struct CMUnitTest tests[] = {
cmocka_unit_test(lex_0xff),
2020-08-01 01:25:37 +10:00
cmocka_unit_test(lex_keypair),
2018-10-23 22:33:45 -07:00
cmocka_unit_test(lex_setline),
};
2020-08-01 01:25:37 +10:00
UNUSED(argv);
if (argc > 1) {
debug = true;
}
return (cmocka_run_group_tests(tests, _setup, _teardown));
2018-10-23 22:33:45 -07:00
}
#else /* HAVE_CMOCKA */
#include <stdio.h>
int
2020-02-13 14:44:37 -08:00
main(void) {
2018-10-23 22:33:45 -07:00
printf("1..0 # Skipped: cmocka not available\n");
return (SKIPPED_TEST_EXIT_CODE);
}
#endif /* if HAVE_CMOCKA */