Commit d1c34f8e authored by Zeyi (Rice) Fan's avatar Zeyi (Rice) Fan Committed by Facebook Github Bot

add level settings to handler

Summary:
This diff adds an option "level" to standard log handler configuration so handlers can have individual logging level set.

This allows us to have multiple handler that handles messages with different level settings.

For example,

> eden=DBG3:default:other; default=stream; other=stream:level=WARN;

This configuration creates two log handlers "default" and "other". Both of the handlers will be receiving log messages level "DBG3" from "eden" log category. In this example, "other" handler will only be receiving log messages that is >= "WARN" while the "default" handler still receives all messages >= "DBG3".

However, one catch of this diff is that, when the log category is set to a higher logging level and the log handler has a lower logging level. The log handler will NOT receive any log messages that is lower than the log category logging level. The test `LogLevelReverseTest` illustrate this situation.

Reviewed By: simpkins

Differential Revision: D15541101

fbshipit-source-id: 741aa97d2d5e93c40324bf6a3dd1d9e38f4dc6a5
parent d7e9357c
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
#include <folly/String.h> #include <folly/String.h>
#include <folly/logging/CustomLogFormatter.h> #include <folly/logging/CustomLogFormatter.h>
#include <folly/logging/GlogStyleFormatter.h> #include <folly/logging/GlogStyleFormatter.h>
#include <folly/logging/LogLevel.h>
#include <folly/logging/LogWriter.h> #include <folly/logging/LogWriter.h>
#include <folly/logging/StandardLogHandler.h> #include <folly/logging/StandardLogHandler.h>
...@@ -106,6 +107,7 @@ std::shared_ptr<StandardLogHandler> StandardLogHandlerFactory::createHandler( ...@@ -106,6 +107,7 @@ std::shared_ptr<StandardLogHandler> StandardLogHandlerFactory::createHandler(
to<string>("unknown log formatter type \"", *formatterType, "\"")); to<string>("unknown log formatter type \"", *formatterType, "\""));
} }
Optional<LogLevel> logLevel;
Optional<LogLevel> syncLevel; Optional<LogLevel> syncLevel;
// Process the log formatter and log handler options // Process the log formatter and log handler options
...@@ -127,8 +129,19 @@ std::shared_ptr<StandardLogHandler> StandardLogHandlerFactory::createHandler( ...@@ -127,8 +129,19 @@ std::shared_ptr<StandardLogHandler> StandardLogHandlerFactory::createHandler(
// We explicitly processed the "formatter" option above. // We explicitly processed the "formatter" option above.
handled |= handled || (entry.first == "formatter"); handled |= handled || (entry.first == "formatter");
if (entry.first == "level") {
try {
logLevel = stringToLogLevel(entry.second);
} catch (const std::exception& ex) {
errors.push_back(to<string>(
"unable to parse value for option \"",
entry.first,
"\": ",
ex.what()));
}
handled = true;
} else if (entry.first == "sync_level") {
// Process the "sync_level" option. // Process the "sync_level" option.
if (entry.first == "sync_level") {
try { try {
syncLevel = stringToLogLevel(entry.second); syncLevel = stringToLogLevel(entry.second);
} catch (const std::exception& ex) { } catch (const std::exception& ex) {
...@@ -155,13 +168,21 @@ std::shared_ptr<StandardLogHandler> StandardLogHandlerFactory::createHandler( ...@@ -155,13 +168,21 @@ std::shared_ptr<StandardLogHandler> StandardLogHandlerFactory::createHandler(
auto writer = writerFactory->createWriter(); auto writer = writerFactory->createWriter();
auto formatter = formatterFactory->createFormatter(writer); auto formatter = formatterFactory->createFormatter(writer);
std::shared_ptr<StandardLogHandler> logHandler;
if (syncLevel) { if (syncLevel) {
return std::make_shared<StandardLogHandler>( logHandler = std::make_shared<StandardLogHandler>(
LogHandlerConfig{type, options}, formatter, writer, *syncLevel); LogHandlerConfig{type, options}, formatter, writer, *syncLevel);
} else { } else {
return std::make_shared<StandardLogHandler>( logHandler = std::make_shared<StandardLogHandler>(
LogHandlerConfig{type, options}, formatter, writer); LogHandlerConfig{type, options}, formatter, writer);
} }
if (logLevel) {
logHandler->setLevel(*logLevel);
}
return logHandler;
} }
} // namespace folly } // namespace folly
/*
* Copyright 2017-present Facebook, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <folly/test/TestUtils.h>
#include <queue>
#include <folly/logging/Init.h>
#include <folly/logging/LogConfigParser.h>
#include <folly/logging/LogHandlerFactory.h>
#include <folly/logging/LogWriter.h>
#include <folly/logging/LoggerDB.h>
#include <folly/logging/StandardLogHandler.h>
#include <folly/logging/StandardLogHandlerFactory.h>
#include <folly/logging/xlog.h>
namespace folly {
class TestLogWriter : public LogWriter {
public:
void writeMessage(folly::StringPiece /* buffer */, uint32_t /* flags */ = 0)
override {
message_count++;
}
void flush() override {}
int message_count{0};
bool ttyOutput() const override {
return false;
}
};
class TestHandlerFactory : public LogHandlerFactory {
public:
explicit TestHandlerFactory(const std::shared_ptr<TestLogWriter> writer)
: writer_(writer) {}
StringPiece getType() const override {
return "test";
}
std::shared_ptr<LogHandler> createHandler(const Options& options) override {
TestWriterFactory writerFactory{writer_};
return StandardLogHandlerFactory::createHandler(
getType(), &writerFactory, options);
}
private:
std::shared_ptr<TestLogWriter> writer_;
class TestWriterFactory : public StandardLogHandlerFactory::WriterFactory {
public:
explicit TestWriterFactory(std::shared_ptr<TestLogWriter> writer)
: writer_(writer) {}
bool processOption(StringPiece /* name */, StringPiece /* value */)
override {
return false;
}
std::shared_ptr<LogWriter> createWriter() override {
return writer_;
}
private:
std::shared_ptr<TestLogWriter> writer_;
};
};
} // namespace folly
using namespace folly;
namespace {
class StandardLogHandlerFactoryTest : public testing::Test {
public:
StandardLogHandlerFactoryTest() {
writer = std::make_shared<TestLogWriter>();
db.registerHandlerFactory(
std::make_unique<TestHandlerFactory>(writer), true);
}
LoggerDB db{LoggerDB::TESTING};
std::shared_ptr<TestLogWriter> writer;
};
} // namespace
TEST_F(StandardLogHandlerFactoryTest, LogLevelTest) {
Logger logger{&db, "test"};
db.resetConfig(parseLogConfig("test=DBG4:default; default=test:level=WARN"));
FB_LOG(logger, DBG9) << "DBG9";
FB_LOG(logger, DBG3) << "DBG3";
FB_LOG(logger, WARN) << "WARN 1";
FB_LOG(logger, WARN) << "WARN 2";
EXPECT_EQ(writer->message_count, 2);
}
TEST_F(StandardLogHandlerFactoryTest, LogLevelReverseTest) {
Logger logger{&db, "test"};
// log category level is WARN, log handler level is DBG3
db.resetConfig(parseLogConfig("test=WARN:default; default=test:level=DBG3"));
FB_LOG(logger, DBG9) << "DBG9";
FB_LOG(logger, DBG3) << "DBG3";
FB_LOG(logger, DBG2) << "DBG2";
FB_LOG(logger, WARN) << "WARN 1";
EXPECT_EQ(writer->message_count, 1);
}
TEST_F(StandardLogHandlerFactoryTest, MultipleLoggerTest) {
Logger logger{&db, "test"};
// log category level is DBG4
// log handler "default" level is DBG3
// log handler "other" level is WARN
db.resetConfig(
parseLogConfig("test=DBG4:default:other;"
"default=test:level=DBG3;"
"other=test:level=WARN"));
FB_LOG(logger, DBG9) << "DBG9"; // no handler should log this
FB_LOG(logger, DBG3) << "DBG3"; // only "default" logs this
FB_LOG(logger, WARN) << "WARN 1"; // both log handlers log this
EXPECT_EQ(writer->message_count, 3);
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment