/*
* This file is part of Cockpit.
*
* Copyright (C) 2014 Red Hat, Inc.
*
* Cockpit 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.
*
* Cockpit 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 Cockpit; If not, see .
*/
#include "config.h"
#include "cockpitpaths.h"
#include "common/cockpittest.h"
#include
typedef struct {
const gchar *a;
const gchar *b;
gboolean expect;
const gchar *name;
} CmpFixture;
static const CmpFixture fixtures_has_parent[] = {
{ "/c", "/c", FALSE, "equal" },
{ "/c", "/c/d", FALSE, "child" },
{ "/c", "/c/d/e", FALSE, "grand-child" },
{ "/c/d", "/c", TRUE, "parent" },
{ "/c/d/e", "/c", FALSE, "grand-parent" },
{ "/c", "/peer", FALSE, "peer-after" },
{ "/c", "/a", FALSE, "peer-before" },
{ "/d", "/door", FALSE, "peer-prefix" },
{ "/cat", "/c", FALSE, "peer-truncated" },
{ "/", "/c", FALSE, "root-child" },
{ "/", "/c/d", FALSE, "root-grand-child" },
{ "/c", "/", TRUE, "root-parent" },
{ "/c/d", "/", FALSE, "root-grand-parent" },
};
static void
test_path_has_parent (gconstpointer data)
{
const CmpFixture *fixture = data;
g_assert (cockpit_path_has_parent (fixture->a, fixture->b) == fixture->expect);
}
static const CmpFixture fixtures_has_ancestor[] = {
{ "/c", "/c", FALSE, "equal" },
{ "/c", "/c/d", FALSE, "child" },
{ "/c", "/c/d/e", FALSE, "grand-child" },
{ "/c/d", "/c", TRUE, "parent" },
{ "/c/d/e", "/c", TRUE, "grand-parent" },
{ "/c", "/peer", FALSE, "peer-after" },
{ "/c", "/a", FALSE, "peer-before" },
{ "/d", "/door", FALSE, "peer-prefix" },
{ "/cat", "/c", FALSE, "peer-truncated" },
{ "/", "/c", FALSE, "root-child" },
{ "/", "/c/d", FALSE, "root-grand-child" },
{ "/c", "/", TRUE, "root-parent" },
{ "/c/d", "/", TRUE, "root-grand-parent" },
};
static void
test_path_has_ancestor (gconstpointer data)
{
const CmpFixture *fixture = data;
g_assert (cockpit_path_has_ancestor (fixture->a, fixture->b) == fixture->expect);
}
static const CmpFixture fixtures_equal_or_ancestor[] = {
{ "/c", "/c", TRUE, "equal" },
{ "/c", "/c/d", FALSE, "child" },
{ "/c", "/c/d/e", FALSE, "grand-child" },
{ "/c/d", "/c", TRUE, "parent" },
{ "/c/d/e", "/c", TRUE, "grand-parent" },
{ "/c", "/peer", FALSE, "peer-after" },
{ "/c", "/a", FALSE, "peer-before" },
{ "/d", "/door", FALSE, "peer-prefix" },
{ "/cat", "/c", FALSE, "peer-truncated" },
{ "/", "/c", FALSE, "root-child" },
{ "/", "/c/d", FALSE, "root-grand-child" },
{ "/c", "/", TRUE, "root-parent" },
{ "/c/d", "/", TRUE, "root-grand-parent" },
};
static void
test_path_equal_or_ancestor (gconstpointer data)
{
const CmpFixture *fixture = data;
g_assert (cockpit_path_equal_or_ancestor (fixture->a, fixture->b) == fixture->expect);
}
static void
test_paths_add_remove (void)
{
GTree *paths;
const gchar *value;
const gchar *check;
paths = cockpit_paths_new ();
g_assert_cmpstr (cockpit_paths_contain (paths, "/one"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/two"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/three/3"), ==, NULL);
g_assert_cmpint (g_tree_nnodes (paths), ==, 0);
/* Add first value */
value = "/one";
check = cockpit_paths_add (paths, value);
g_assert (value != check); /* reallocated */
g_assert_cmpstr (cockpit_paths_contain (paths, "/one"), ==, "/one");
g_assert (cockpit_paths_contain (paths, "/one") == check);
g_assert_cmpstr (cockpit_paths_contain (paths, "/two"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/three/3"), ==, NULL);
g_assert_cmpint (g_tree_nnodes (paths), ==, 1);
/* Add another value */
value = cockpit_paths_add (paths, "/two");
g_assert (value != NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/one"), ==, "/one");
g_assert_cmpstr (cockpit_paths_contain (paths, "/two"), ==, "/two");
g_assert_cmpstr (cockpit_paths_contain (paths, "/three/3"), ==, NULL);
g_assert_cmpint (g_tree_nnodes (paths), ==, 2);
/* Add same one again */
check = cockpit_paths_add (paths, "/two");
g_assert (check == NULL); /* Already present */
g_assert_cmpstr (cockpit_paths_contain (paths, "/one"), ==, "/one");
g_assert_cmpstr (cockpit_paths_contain (paths, "/two"), ==, "/two");
g_assert_cmpstr (cockpit_paths_contain (paths, "/three/3"), ==, NULL);
g_assert_cmpint (g_tree_nnodes (paths), ==, 2);
/* Remove first value */
g_assert (cockpit_paths_remove (paths, "/one") == TRUE); /* actually removed */
g_assert_cmpstr (cockpit_paths_contain (paths, "/one"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/two"), ==, "/two");
g_assert_cmpstr (cockpit_paths_contain (paths, "/three/3"), ==, NULL);
g_assert_cmpint (g_tree_nnodes (paths), ==, 1);
/* Remove reference to second value */
g_assert (cockpit_paths_remove (paths, "/two") == TRUE); /* not actually */
g_assert_cmpstr (cockpit_paths_contain (paths, "/one"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/two"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain (paths, "/three/3"), ==, NULL);
g_assert_cmpint (g_tree_nnodes (paths), ==, 0);
/* Add something before destroy, to check destructors */
cockpit_paths_add (paths, "/three/3");
g_tree_destroy (paths);
}
static void
test_paths_ancestor_descendant (void)
{
GTree *paths;
paths = cockpit_paths_new ();
cockpit_paths_add (paths, "/a");
cockpit_paths_add (paths, "/b");
cockpit_paths_add (paths, "/c/3");
g_assert (cockpit_paths_contain_or_descendant (paths, "/0") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/z") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/a") == TRUE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/a/1") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/a1") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/azzzzzz") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/") == TRUE);
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/b"), ==, "/b");
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/b2"), ==, NULL);
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/b/2"), ==, "/b");
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/"), ==, NULL);
g_assert(cockpit_paths_contain_or_descendant (paths, "/c/3/4") == FALSE);
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/c"), ==, NULL);
/* Everything */
cockpit_paths_add (paths, "/");
g_assert (cockpit_paths_contain_or_descendant (paths, "/a") == TRUE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/a/1") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/a1") == FALSE);
g_assert (cockpit_paths_contain_or_descendant (paths, "/") == TRUE);
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/b"), ==, "/b");
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/b2"), ==, "/");
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/b/2"), ==, "/b");
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/"), ==, "/");
g_assert (cockpit_paths_contain_or_descendant (paths, "/c/3/4") == FALSE);
g_assert_cmpstr (cockpit_paths_contain_or_ancestor (paths, "/c"), ==, "/");
g_tree_destroy (paths);
}
int
main (int argc,
char *argv[])
{
gchar *path;
guint i;
cockpit_test_init (&argc, &argv);
for (i = 0; i < G_N_ELEMENTS (fixtures_has_parent); i++)
{
path = g_strdup_printf ("/paths/has-parent/%s", fixtures_has_parent[i].name);
g_test_add_data_func (path, fixtures_has_parent + i, test_path_has_parent);
g_free (path);
}
for (i = 0; i < G_N_ELEMENTS (fixtures_has_parent); i++)
{
path = g_strdup_printf ("/paths/has-ancestor/%s", fixtures_has_ancestor[i].name);
g_test_add_data_func (path, fixtures_has_ancestor + i, test_path_has_ancestor);
g_free (path);
}
for (i = 0; i < G_N_ELEMENTS (fixtures_equal_or_ancestor); i++)
{
path = g_strdup_printf ("/paths/equal-or-ancestor/%s", fixtures_equal_or_ancestor[i].name);
g_test_add_data_func (path, fixtures_equal_or_ancestor + i, test_path_equal_or_ancestor);
g_free (path);
}
g_test_add_func ("/paths/add-remove", test_paths_add_remove);
g_test_add_func ("/paths/ancestor-descendant", test_paths_ancestor_descendant);
return g_test_run ();
}