/* Copyright 2012 10gen Inc. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License, version 3, * as published by the Free Software Foundation. * * 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 Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * * As a special exception, the copyright holders give permission to link the * code of portions of this program with the OpenSSL library under certain * conditions as described in each individual source file and distribute * linked combinations including the program with the OpenSSL library. You * must comply with the GNU Affero General Public License in all respects * for all of the code used other than as permitted herein. If you modify * file(s) with this exception, you may extend this exception to your * version of the file(s), but you are not obligated to do so. If you do not * wish to do so, delete this exception statement from your version. If you * delete this exception statement from all source files in the program, * then also delete it in the license file. */ /** * Unit tests of the InitializerDependencyGraph type. */ #include "mongo/base/init.h" #include "mongo/base/initializer_dependency_graph.h" #include "mongo/base/make_string_vector.h" #include "mongo/unittest/unittest.h" #define ADD_INITIALIZER(GRAPH, NAME, FN, PREREQS, DEPS) \ (GRAPH).addInitializer( \ (NAME), (FN), MONGO_MAKE_STRING_VECTOR PREREQS, MONGO_MAKE_STRING_VECTOR DEPS) #define ASSERT_ADD_INITIALIZER(GRAPH, NAME, FN, PREREQS, DEPS) \ ASSERT_EQUALS(Status::OK(), ADD_INITIALIZER(GRAPH, NAME, FN, PREREQS, DEPS)) #define ASSERT_EXACTLY_N_IN_CONTAINER(N, CONTAINER, THING) \ ASSERT_EQUALS(N, std::count((CONTAINER).begin(), (CONTAINER).end(), (THING))) #define ASSERT_AT_LEAST_N_IN_CONTAINER(N, CONTAINER, THING) \ ASSERT_LESS_THAN_OR_EQUALS(N, std::count((CONTAINER).begin(), (CONTAINER).end(), (THING))) #define ASSERT_EXACTLY_ONE_IN_CONTAINER(CONTAINER, THING) \ ASSERT_EXACTLY_N_IN_CONTAINER(1, CONTAINER, THING) namespace mongo { namespace { Status doNothing(InitializerContext*) { return Status::OK(); } TEST(InitializerDependencyGraphTest, InsertNullFunctionFails) { InitializerDependencyGraph graph; ASSERT_EQUALS( ErrorCodes::BadValue, ADD_INITIALIZER( graph, "A", InitializerFunction(), MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS)); } TEST(InitializerDependencyGraphTest, InsertSameNameTwiceFails) { InitializerDependencyGraph graph; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_EQUALS( ErrorCodes::DuplicateKey, ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS)); } TEST(InitializerDependencyGraphTest, TopSortEmptyGraph) { InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(0U, nodeNames.size()); } TEST(InitializerDependencyGraphTest, TopSortGraphNoDeps) { InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(3U, nodeNames.size()); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "A"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "B"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "C"); } TEST(InitializerDependencyGraphTest, TopSortWithDiamondPrerequisites) { /* * This tests top-sorting a simple diamond, specified using prerequisites: * * B * / ^ * v \ * A D * ^ / * \ v * C */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "D", doNothing, ("B", "C"), MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, ("A"), MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, ("A"), MONGO_NO_DEPENDENTS); ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(4U, nodeNames.size()); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "A"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "B"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "C"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "D"); ASSERT_EQUALS("A", nodeNames.front()); ASSERT_EQUALS("D", nodeNames.back()); } TEST(InitializerDependencyGraphTest, TopSortWithDiamondDependents) { /* * This tests top-sorting a simple diamond, specified using dependents: * * B * / ^ * v \ * A D * ^ / * \ v * C */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, ("B", "C")); ASSERT_ADD_INITIALIZER(graph, "D", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, MONGO_NO_PREREQUISITES, ("D")); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, MONGO_NO_PREREQUISITES, ("D")); ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(4U, nodeNames.size()); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "A"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "B"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "C"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "D"); ASSERT_EQUALS("A", nodeNames.front()); ASSERT_EQUALS("D", nodeNames.back()); } TEST(InitializerDependencyGraphTest, TopSortWithDiamondGeneral1) { /* * This tests top-sorting a simple diamond, where B and C specify all prerequisites and * dependents. * * B * / ^ * v \ * A D * ^ / * \ v * C */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "D", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, ("A"), ("D")); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, ("A"), ("D")); ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(4U, nodeNames.size()); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "A"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "B"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "C"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "D"); ASSERT_EQUALS("A", nodeNames.front()); ASSERT_EQUALS("D", nodeNames.back()); } TEST(InitializerDependencyGraphTest, TopSortWithDiamondGeneral2) { /* * This tests top-sorting a simple diamond, where A and D specify all prerequisites and * dependents. * * B * / ^ * v \ * A D * ^ / * \ v * C */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, ("B", "C")); ASSERT_ADD_INITIALIZER(graph, "D", doNothing, ("C", "B"), MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(4U, nodeNames.size()); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "A"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "B"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "C"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "D"); ASSERT_EQUALS("A", nodeNames.front()); ASSERT_EQUALS("D", nodeNames.back()); } TEST(InitializerDependencyGraphTest, TopSortWithDiamondGeneral3) { /* * This tests top-sorting a simple diamond, where A and D specify all prerequisites and * dependents, but so do B and C. * * B * / ^ * v \ * A D * ^ / * \ v * C */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, ("B", "C")); ASSERT_ADD_INITIALIZER(graph, "D", doNothing, ("C", "B"), MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, ("A"), ("D")); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, ("A"), ("D")); ASSERT_EQUALS(Status::OK(), graph.topSort(&nodeNames)); ASSERT_EQUALS(4U, nodeNames.size()); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "A"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "B"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "C"); ASSERT_EXACTLY_ONE_IN_CONTAINER(nodeNames, "D"); ASSERT_EQUALS("A", nodeNames.front()); ASSERT_EQUALS("D", nodeNames.back()); } TEST(InitializerDependencyGraphTest, TopSortWithDiamondAndCycle) { /* * This tests top-sorting a graph with a cycle, which should fail.. * * B <- E * / ^ ^ * v \ / * A D * ^ / * \ v * C */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, ("B", "C")); ASSERT_ADD_INITIALIZER(graph, "D", doNothing, ("C", "B"), MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "B", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "C", doNothing, MONGO_NO_PREREQUISITES, MONGO_NO_DEPENDENTS); ASSERT_ADD_INITIALIZER(graph, "E", doNothing, ("D"), ("B")); ASSERT_EQUALS(ErrorCodes::GraphContainsCycle, graph.topSort(&nodeNames)); ASSERT_EQUALS(4U, nodeNames.size()); ASSERT_EQUALS(nodeNames.front(), nodeNames.back()); ASSERT_AT_LEAST_N_IN_CONTAINER(1, nodeNames, "D"); ASSERT_AT_LEAST_N_IN_CONTAINER(1, nodeNames, "E"); ASSERT_AT_LEAST_N_IN_CONTAINER(1, nodeNames, "B"); ASSERT_EXACTLY_N_IN_CONTAINER(0, nodeNames, "A"); ASSERT_EXACTLY_N_IN_CONTAINER(0, nodeNames, "C"); } TEST(InitializerDependencyGraphTest, TopSortFailsWhenMissingPrerequisite) { /* * If a node names a never-declared prerequisite, topSort should fail. */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "B", doNothing, ("A"), MONGO_NO_DEPENDENTS); auto status = graph.topSort(&nodeNames); ASSERT_EQUALS(ErrorCodes::BadValue, status); ASSERT_STRING_CONTAINS(status.reason(), "depends on missing initializer A"); } TEST(InitializerDependencyGraphTest, TopSortFailsWhenMissingDependent) { /* * If a node names a never-declared dependent, topSort should fail. */ InitializerDependencyGraph graph; std::vector nodeNames; ASSERT_ADD_INITIALIZER(graph, "A", doNothing, MONGO_NO_PREREQUISITES, ("B")); auto status = graph.topSort(&nodeNames); ASSERT_EQUALS(ErrorCodes::BadValue, status); ASSERT_STRING_CONTAINS(status.reason(), "No implementation provided for initializer B"); } } // namespace } // namespace mongo