123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185 |
- // Copyright (C) 2011 Internet Systems Consortium, Inc. ("ISC")
- //
- // Permission to use, copy, modify, and/or distribute this software for any
- // purpose with or without fee is hereby granted, provided that the above
- // copyright notice and this permission notice appear in all copies.
- //
- // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
- // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
- // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
- // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
- // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
- // PERFORMANCE OF THIS SOFTWARE.
- #include "config.h"
- #include <gtest/gtest.h>
- #include <util/unittests/resource.h>
- #include <log/log_formatter.h>
- #include <log/logger_level.h>
- #include <vector>
- #include <string>
- using namespace std;
- namespace {
- class FormatterTest : public ::testing::Test {
- protected:
- typedef pair<isc::log::Severity, string> Output;
- typedef isc::log::Formatter<FormatterTest> Formatter;
- vector<Output> outputs;
- public:
- void output(const isc::log::Severity& prefix, const string& message) {
- outputs.push_back(Output(prefix, message));
- }
- // Just shortcut for new string
- string* s(const char* text) {
- return (new string(text));
- }
- };
- // Create an inactive formatter and check it doesn't produce any output
- TEST_F(FormatterTest, inactive) {
- Formatter();
- EXPECT_EQ(0, outputs.size());
- }
- // Create an active formatter and check it produces output. Does no arg
- // substitution yet
- TEST_F(FormatterTest, active) {
- Formatter(isc::log::INFO, s("Text of message"), this);
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("Text of message", outputs[0].second);
- }
- // No output even when we have an arg on the inactive formatter
- TEST_F(FormatterTest, inactiveArg) {
- Formatter().arg("Hello");
- EXPECT_EQ(0, outputs.size());
- }
- // Create an active formatter and replace a placeholder with string
- TEST_F(FormatterTest, stringArg) {
- {
- SCOPED_TRACE("C++ string");
- Formatter(isc::log::INFO, s("Hello %1"), this).arg(string("World"));
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("Hello World", outputs[0].second);
- }
- {
- SCOPED_TRACE("C++ string");
- Formatter(isc::log::INFO, s("Hello %1"), this).arg(string("Internet"));
- ASSERT_EQ(2, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[1].first);
- EXPECT_EQ("Hello Internet", outputs[1].second);
- }
- }
- // Test the .deactivate() method
- TEST_F(FormatterTest, deactivate) {
- Formatter(isc::log::INFO, s("Text of message"), this).deactivate();
- // If there was no .deactivate, it should have output it.
- // But not now.
- ASSERT_EQ(0, outputs.size());
- }
- // Can convert to string
- TEST_F(FormatterTest, intArg) {
- Formatter(isc::log::INFO, s("The answer is %1"), this).arg(42);
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("The answer is 42", outputs[0].second);
- }
- // Can use multiple arguments at different places
- TEST_F(FormatterTest, multiArg) {
- Formatter(isc::log::INFO, s("The %2 are %1"), this).arg("switched").
- arg("arguments");
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("The arguments are switched", outputs[0].second);
- }
- #ifdef ENABLE_LOGGER_CHECKS
- TEST_F(FormatterTest, mismatchedPlaceholders) {
- // Throws MismatchedPlaceholders exception if the placeholder is missing
- // for a supplied argument.
- EXPECT_THROW(Formatter(isc::log::INFO, s("Missing the second %1"), this).
- arg("argument").arg("missing"),
- isc::log::MismatchedPlaceholders);
- #ifdef EXPECT_DEATH
- // Likewise, if there's a redundant placeholder (or missing argument), the
- // check detects it and aborts the program. Due to the restriction of the
- // current implementation, it doesn't throw.
- EXPECT_DEATH({
- isc::util::unittests::dontCreateCoreDumps();
- Formatter(isc::log::INFO, s("Too many arguments in %1 %2"), this).
- arg("only one");
- }, ".*");
- #endif /* EXPECT_DEATH */
- // Mixed case of above two: the exception will be thrown due to the missing
- // placeholder. The other check is disabled due to that.
- EXPECT_THROW(Formatter(isc::log::INFO, s("Missing the first %2"), this).
- arg("missing").arg("argument"),
- isc::log::MismatchedPlaceholders);
- }
- #else
- // If logger checks are not enabled, nothing is thrown
- TEST_F(FormatterTest, mismatchedPlaceholders) {
- Formatter(isc::log::INFO, s("Missing the second %1"), this).
- arg("argument").arg("missing");
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("Missing the second argument "
- "@@Missing placeholder %2 for 'missing'@@", outputs[0].second);
- EXPECT_NO_THROW(Formatter(isc::log::INFO,
- s("Too many arguments in %1 %2"), this).
- arg("only one"));
- ASSERT_EQ(2, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[1].first);
- EXPECT_EQ("Too many arguments in only one %2 "
- "@@Excess logger placeholders still exist@@",
- outputs[1].second);
- EXPECT_NO_THROW(Formatter(isc::log::INFO, s("Missing the first %2"), this).
- arg("missing").arg("argument"));
- ASSERT_EQ(3, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[2].first);
- EXPECT_EQ("Missing the first argument "
- "@@Missing placeholder %1 for 'missing'@@", outputs[2].second);
- }
- #endif /* ENABLE_LOGGER_CHECKS */
- // Can replace multiple placeholders
- TEST_F(FormatterTest, multiPlaceholder) {
- Formatter(isc::log::INFO, s("The %1 is the %1"), this).
- arg("first rule of tautology club");
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("The first rule of tautology club is "
- "the first rule of tautology club", outputs[0].second);
- }
- // Test we can cope with replacement containing the placeholder
- TEST_F(FormatterTest, noRecurse) {
- // If we recurse, this will probably eat all the memory and crash
- Formatter(isc::log::INFO, s("%1"), this).arg("%1 %1");
- ASSERT_EQ(1, outputs.size());
- EXPECT_EQ(isc::log::INFO, outputs[0].first);
- EXPECT_EQ("%1 %1", outputs[0].second);
- }
- }
|