/** * @file * * CUnit testing framework for libpoldiff's correctness. * * @author Paul Rosenfeld prosenfeld@tresys.com * * Copyright (C) 2007 Tresys Technology, LLC * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include "libpoldiff-tests.h" #include #include #include #include #include #include #include #include #include #include "components-tests.h" #include "rules-tests.h" #include "mls-tests.h" #include "nomls-tests.h" apol_vector_t *string_array_to_vector(char *arr[]) { apol_vector_t *v = apol_vector_create(free); int i; for (i = 0; arr[i] != NULL; ++i) { apol_vector_append(v, strdup(arr[i])); } return v; } char *vector_to_string(const apol_vector_t * v, const char *pre, const char *sep) { char *item = NULL, *str = NULL, *tmp = NULL; size_t i = 0, str_len = 0, tmp_len = 0; size_t num_elements = apol_vector_get_size(v); for (i = 0; v && i < num_elements; i++) { item = apol_vector_get_element(v, i); if (apol_str_appendf(&tmp, &tmp_len, "%s%s", sep, item) < 0) { return NULL; } } apol_str_trim(tmp); if (tmp) { apol_str_appendf(&str, &str_len, "%s%s", pre, tmp); } else { str = strdup(""); } free(tmp); return str; } apol_vector_t *shallow_copy_str_vec_and_sort(const apol_vector_t * v) { apol_vector_t *copy = apol_vector_create_from_vector(v, NULL, NULL, NULL); apol_vector_sort(copy, apol_str_strcmp, NULL); return copy; } void run_test(component_funcs_t * component_funcs, poldiff_test_answers_t * poldiff_test_answers, test_numbers_e test_num) { added_v = apol_vector_create(free); removed_v = apol_vector_create(free); modified_v = apol_vector_create(free); modified_name_only_v = apol_vector_create(free); switch (test_num) { case COMPONENT: build_component_vecs(component_funcs); break; case RULES_AVRULE: build_avrule_vecs(); break; case RULES_TERULE: build_terule_vecs(); break; case RULES_ROLEALLOW: build_roleallow_vecs(); break; case RULES_ROLETRANS: build_roletrans_vecs(); break; case MLS_CATEGORY: build_category_vecs(); break; case MLS_LEVEL: build_level_vecs(); break; case MLS_RANGETRANS: build_rangetrans_vecs(); break; case MLS_USER: build_user_vecs(); break; } size_t first_diff; apol_vector_t *intersect = NULL, *all_changes = NULL; if (!(all_changes = apol_vector_create(NULL))) { goto err; } apol_vector_cat(all_changes, added_v); apol_vector_cat(all_changes, removed_v); apol_vector_cat(all_changes, modified_name_only_v); if (! (intersect = apol_vector_create_from_intersection(all_changes, poldiff_test_answers->correct_unchanged_v, compare_str, NULL))) { goto err; } /* unchanged */ CU_ASSERT_EQUAL(apol_vector_get_size(intersect), 0); /* added */ apol_vector_sort(added_v, compare_str, NULL); apol_vector_sort(poldiff_test_answers->correct_added_v, compare_str, NULL); int test_result; CU_ASSERT_FALSE(test_result = apol_vector_compare(added_v, poldiff_test_answers->correct_added_v, compare_str, NULL, &first_diff)); if (test_result) { print_test_failure(added_v, poldiff_test_answers->correct_added_v, first_diff, "Added"); } /* removed */ apol_vector_sort(removed_v, compare_str, NULL); apol_vector_sort(poldiff_test_answers->correct_removed_v, compare_str, NULL); CU_ASSERT_FALSE(test_result = apol_vector_compare(removed_v, poldiff_test_answers->correct_removed_v, compare_str, NULL, &first_diff)); if (test_result) { print_test_failure(removed_v, poldiff_test_answers->correct_removed_v, first_diff, "Removed"); } /* modified */ apol_vector_sort(modified_v, compare_str, NULL); apol_vector_sort(poldiff_test_answers->correct_modified_v, compare_str, NULL); CU_ASSERT_FALSE(test_result = apol_vector_compare(modified_v, poldiff_test_answers->correct_modified_v, compare_str, NULL, &first_diff)); if (test_result) { print_test_failure(modified_v, poldiff_test_answers->correct_modified_v, first_diff, "Modified"); } apol_vector_destroy(&intersect); apol_vector_destroy(&added_v); apol_vector_destroy(&removed_v); apol_vector_destroy(&modified_name_only_v); apol_vector_destroy(&modified_v); apol_vector_destroy(&all_changes); return; err: apol_vector_destroy(&intersect); apol_vector_destroy(&added_v); apol_vector_destroy(&removed_v); apol_vector_destroy(&modified_name_only_v); apol_vector_destroy(&modified_v); apol_vector_destroy(&all_changes); CU_FAIL_FATAL("Could not initialize vectors for test"); } void print_test_failure(apol_vector_t * actual, apol_vector_t * expected, size_t first_diff, const char *test_name) { printf("\nTEST FAILED\n"); size_t i; printf("--- ACTUAL RESULT (%s) -----\n", test_name); for (i = first_diff; i < apol_vector_get_size(actual); ++i) { char *item = (char *)apol_vector_get_element(actual, i); printf("\t%3d. %s\n", (int)i, item); } printf("--- EXPECTED RESULT (%s) ---\n", test_name); for (i = first_diff; i < apol_vector_get_size(expected); ++i) { char *item = (char *)apol_vector_get_element(expected, i); printf("\t%3d. %s\n", (int)i, item); } } int compare_str(const void *s1, const void *s2, void *debug) { char *str1 = strdup((char *)s1); char *str2 = strdup((char *)s2); apol_str_trim(str1); apol_str_trim(str2); int result = strcmp(str1, str2); free(str1); free(str2); return result; } poldiff_test_answers_t *init_answer_vectors(char *added_arr[], char *removed_arr[], char *unchanged_arr[], char *modified_arr[]) { poldiff_test_answers_t *answers = (poldiff_test_answers_t *) malloc(sizeof(poldiff_test_answers_t)); answers->correct_added_v = string_array_to_vector(added_arr); answers->correct_removed_v = string_array_to_vector(removed_arr); answers->correct_unchanged_v = string_array_to_vector(unchanged_arr); answers->correct_modified_v = string_array_to_vector(modified_arr); return answers; } component_funcs_t *init_test_funcs(poldiff_get_diff_vector get_diff_vector, poldiff_get_name get_name, poldiff_get_form get_form, poldiff_get_added get_added, poldiff_get_removed get_removed) { component_funcs_t *funcs = (component_funcs_t *) malloc(sizeof(component_funcs_t)); funcs->get_diff_vector = get_diff_vector; funcs->get_name = get_name; funcs->get_form = get_form; funcs->get_added = get_added; funcs->get_removed = get_removed; return funcs; } poldiff_t *init_poldiff(char *orig_base_path, char *mod_base_path) { poldiff_t *return_diff = NULL; uint32_t flags = POLDIFF_DIFF_ALL; apol_policy_path_t *mod_pol_path = NULL; apol_policy_path_t *orig_pol_path = NULL; orig_pol_path = apol_policy_path_create(APOL_POLICY_PATH_TYPE_MONOLITHIC, orig_base_path, NULL); if (!orig_pol_path) { ERR(NULL, "%s", strerror(errno)); goto err; } mod_pol_path = apol_policy_path_create(APOL_POLICY_PATH_TYPE_MONOLITHIC, mod_base_path, NULL); if (!mod_pol_path) { ERR(NULL, "%s", strerror(errno)); goto err; } orig_policy = apol_policy_create_from_policy_path(orig_pol_path, 0, NULL, NULL); if (!orig_policy) { ERR(NULL, "%s", strerror(errno)); goto err; } mod_policy = apol_policy_create_from_policy_path(mod_pol_path, 0, NULL, NULL); if (!mod_policy) { ERR(NULL, "%s", strerror(errno)); goto err; } if (!(return_diff = poldiff_create(orig_policy, mod_policy, NULL, NULL))) { ERR(NULL, "%s", strerror(errno)); goto err; } if (poldiff_run(return_diff, flags)) { goto err; } apol_policy_path_destroy(&orig_pol_path); apol_policy_path_destroy(&mod_pol_path); return return_diff; err: apol_policy_destroy(&orig_policy); apol_policy_destroy(&mod_policy); apol_policy_path_destroy(&orig_pol_path); apol_policy_path_destroy(&mod_pol_path); poldiff_destroy(&return_diff); return NULL; } void cleanup_test(poldiff_test_answers_t * answers) { if (answers != NULL) { apol_vector_destroy(&answers->correct_added_v); apol_vector_destroy(&answers->correct_unchanged_v); apol_vector_destroy(&answers->correct_removed_v); apol_vector_destroy(&answers->correct_modified_v); free(answers); } } int poldiff_cleanup() { poldiff_destroy(&diff); return 0; } int main() { if (CU_initialize_registry() != CUE_SUCCESS) { return CU_get_error(); } CU_TestInfo components_tests_arr[] = { {"Attributes", components_attributes_tests} , {"Classes", components_class_tests} , {"Commons", components_commons_tests} , {"Roles", components_roles_tests} , {"Users", components_users_tests} , {"Bools", components_bools_tests} , {"Types", components_types_tests} , CU_TEST_INFO_NULL }; CU_TestInfo rules_tests_arr[] = { {"AV Rules", rules_avrules_tests} , {"TE Rules", rules_terules_tests} , {"Role Allow Rules", rules_roleallow_tests} , {"Role Transition Rules", rules_roletrans_tests} , CU_TEST_INFO_NULL }; CU_TestInfo mls_tests_arr[] = { {"Categories", mls_category_tests} , {"Levels", mls_level_tests} , {"Range Transitions", mls_rangetrans_tests} , {"Users (MLS)", mls_user_tests} , CU_TEST_INFO_NULL }; CU_TestInfo nomls_tests_arr[] = { {"Changed & Unchanged Users", nomls_tests} , CU_TEST_INFO_NULL }; CU_SuiteInfo suites[] = { {"Components", components_test_init, poldiff_cleanup, components_tests_arr} , {"Rules", rules_test_init, poldiff_cleanup, rules_tests_arr} , {"MLS", mls_test_init, poldiff_cleanup, mls_tests_arr} , {"Non-MLS vs. MLS Users", nomls_test_init, poldiff_cleanup, nomls_tests_arr} , CU_SUITE_INFO_NULL }; CU_register_suites(suites); CU_basic_set_mode(CU_BRM_VERBOSE); CU_basic_run_tests(); unsigned int num_failures = CU_get_number_of_failure_records(); CU_cleanup_registry(); return (int)num_failures; }