diff --git a/Foundation/Foundation_vs160.vcxproj b/Foundation/Foundation_vs160.vcxproj index 442fa14e2..32ac92c19 100644 --- a/Foundation/Foundation_vs160.vcxproj +++ b/Foundation/Foundation_vs160.vcxproj @@ -873,6 +873,7 @@ + @@ -1637,6 +1638,7 @@ + diff --git a/Foundation/Foundation_vs160.vcxproj.filters b/Foundation/Foundation_vs160.vcxproj.filters index dbaf9eb24..fa7089179 100644 --- a/Foundation/Foundation_vs160.vcxproj.filters +++ b/Foundation/Foundation_vs160.vcxproj.filters @@ -940,6 +940,9 @@ Text\Utf8Proc Source Files + + Logging\Source Files + @@ -1896,6 +1899,9 @@ Text\Utf8Proc Header Files + + Logging\Header Files + diff --git a/Foundation/Foundation_vs170.vcxproj b/Foundation/Foundation_vs170.vcxproj index c6232b03b..0689fc4f9 100644 --- a/Foundation/Foundation_vs170.vcxproj +++ b/Foundation/Foundation_vs170.vcxproj @@ -1275,6 +1275,7 @@ + @@ -2255,6 +2256,7 @@ + diff --git a/Foundation/Foundation_vs170.vcxproj.filters b/Foundation/Foundation_vs170.vcxproj.filters index b6d0235e9..c9fc58917 100644 --- a/Foundation/Foundation_vs170.vcxproj.filters +++ b/Foundation/Foundation_vs170.vcxproj.filters @@ -940,6 +940,9 @@ Text\Utf8Proc Source Files + + Logging\Source Files + @@ -1896,6 +1899,9 @@ Text\Utf8Proc Header Files + + Logging\Header Files + diff --git a/Foundation/Makefile b/Foundation/Makefile index 3b6b6e93a..86fe425db 100644 --- a/Foundation/Makefile +++ b/Foundation/Makefile @@ -19,7 +19,7 @@ objects = ArchiveStrategy Ascii ASCIIEncoding AsyncChannel AsyncNotificationCent NestedDiagnosticContext Notification NotificationCenter \ NotificationQueue PriorityNotificationQueue TimedNotificationQueue \ NullStream NumberFormatter NumberParser NumericString AbstractObserver \ - Path PatternFormatter PIDFile Process ProcessRunner PurgeStrategy RWLock Random RandomStream \ + Path PatternFormatter JSONFormatter PIDFile Process ProcessRunner PurgeStrategy RWLock Random RandomStream \ DirectoryIteratorStrategy RegularExpression RefCountedObject Runnable RotateStrategy \ SHA1Engine SHA2Engine Semaphore SharedLibrary SimpleFileChannel \ SignalHandler SplitterChannel SortedDirectoryIterator Stopwatch StreamChannel \ diff --git a/Foundation/include/Poco/JSONFormatter.h b/Foundation/include/Poco/JSONFormatter.h new file mode 100644 index 000000000..5edc6e16d --- /dev/null +++ b/Foundation/include/Poco/JSONFormatter.h @@ -0,0 +1,109 @@ +// +// JSONFormatter.h +// +// Library: Foundation +// Package: Logging +// Module: JSONFormatter +// +// Definition of the JSONFormatter class. +// +// Copyright (c) 2024, Applied Informatics Software Engineering GmbH. +// and Contributors. +// +// SPDX-License-Identifier: BSL-1.0 +// + + +#ifndef Foundation_JSONFormatter_INCLUDED +#define Foundation_JSONFormatter_INCLUDED + + +#include "Poco/Foundation.h" +#include "Poco/Formatter.h" +#include "Poco/Message.h" +#include + + +namespace Poco { + + +class Foundation_API JSONFormatter: public Formatter + /// This formatter formats log messages as compact + /// (no unnecessary whitespace) single-line JSON strings. + /// + /// The following JSON schema is used: + /// { + /// "timestamp": "2024-09-26T13:41:23.324461Z", + /// "source": "sample", + /// "level": "information", + /// "message": "This is a test message.", + /// "thread": 12, + /// "file": "source.cpp", + /// "line": 456, + /// "params": { + /// "prop1": "value1" + /// } + /// } + /// + /// The "file" and "line" properties will only be included if the log + /// message contains a file name and line number. + /// + /// The "params" object will only be included if custom parameters + /// have been added to the Message. +{ +public: + using Ptr = AutoPtr; + + JSONFormatter() = default; + /// Creates a JSONFormatter. + + ~JSONFormatter() = default; + /// Destroys the JSONFormatter. + + void format(const Message& msg, std::string& text); + /// Formats the message as a JSON string. + + void setProperty(const std::string& name, const std::string& value); + /// Sets the property with the given name to the given value. + /// + /// The following properties are supported: + /// + /// * times: Specifies whether times are adjusted for local time + /// or taken as they are in UTC. Supported values are "local" and "UTC". + /// * thread: Specifies the value given for the thread. Can be + /// "none" (excluded), "name" (thread name), "id" (POCO thread ID) or "osid" + /// (operating system thread ID). + /// + /// If any other property name is given, a PropertyNotSupported + /// exception is thrown. + + std::string getProperty(const std::string& name) const; + /// Returns the value of the property with the given name or + /// throws a PropertyNotSupported exception if the given + /// name is not recognized. + + static const std::string PROP_TIMES; + static const std::string PROP_THREAD; + +protected: + std::string getThread(const Message& message) const; + static const std::string& getPriorityName(int prio); + + enum ThreadFormat + { + JSONF_THREAD_NONE = 0, + JSONF_THREAD_NAME = 1, + JSONF_THREAD_ID = 2, + JSONF_THREAD_OS_ID = 3 + }; + +private: + bool _localTime = false; + ThreadFormat _threadFormat = JSONF_THREAD_ID; +}; + + +} // namespace Poco + + +#endif // Foundation_JSONFormatter_INCLUDED diff --git a/Foundation/src/JSONFormatter.cpp b/Foundation/src/JSONFormatter.cpp new file mode 100644 index 000000000..9b14487a9 --- /dev/null +++ b/Foundation/src/JSONFormatter.cpp @@ -0,0 +1,182 @@ +// +// JSONFormatter.cpp +// +// Library: Foundation +// Package: Logging +// Module: JSONFormatter +// +// Copyright (c) 2024, Applied Informatics Software Engineering GmbH. +// and Contributors. +// +// SPDX-License-Identifier: BSL-1.0 +// + + +#include "Poco/JSONFormatter.h" +#include "Poco/JSONString.h" +#include "Poco/Message.h" +#include "Poco/String.h" +#include "Poco/JSONString.h" +#include "Poco/NumberFormatter.h" +#include "Poco/DateTimeFormatter.h" +#include "Poco/DateTimeFormat.h" +#include "Poco/Timezone.h" + + +namespace Poco { + + +const std::string JSONFormatter::PROP_TIMES("times"); +const std::string JSONFormatter::PROP_THREAD("thread"); + + +void JSONFormatter::format(const Message& msg, std::string& text) +{ + Timestamp timestamp = msg.getTime(); + int tzd = DateTimeFormatter::UTC; + if (_localTime) + { + tzd = Timezone::utcOffset(); + tzd += Timezone::dst(); + timestamp += tzd*Timestamp::resolution(); + } + + text += '{'; + text += "\"timestamp\":\""; + text += Poco::DateTimeFormatter::format(timestamp, Poco::DateTimeFormat::ISO8601_FRAC_FORMAT, tzd); + text += "\",\"source\":"; + text += toJSON(msg.getSource()); + text += ",\"level\":\""; + text += getPriorityName(msg.getPriority()); + text += "\",\"message\":"; + text += toJSON(msg.getText()); + if (_threadFormat != JSONF_THREAD_NONE) + { + text += ",\"thread\":"; + text += getThread(msg); + } + if (msg.getSourceFile()) + { + text += ",\"file\":"; + text += toJSON(msg.getSourceFile()); + } + if (msg.getSourceLine()) + { + text += ",\"line\":\""; + text += Poco::NumberFormatter::format(msg.getSourceLine()); + text += "\""; + } + if (!msg.getAll().empty()) + { + text += ",\"params\":{"; + const auto& props = msg.getAll(); + bool first = true; + for (const auto& p: props) + { + if (!first) + text += ','; + else + first = false; + text += toJSON(p.first); + text += ':'; + text += toJSON(p.second); + } + text += '}'; + } + text += '}'; +} + + +void JSONFormatter::setProperty(const std::string& name, const std::string& value) +{ + if (name == PROP_TIMES) + { + if (Poco::icompare(value, "local"s) == 0) + _localTime = true; + else if (Poco::icompare(value, "utc"s) == 0) + _localTime = false; + else + throw Poco::InvalidArgumentException("Invalid times value (must be local or UTC)"s, value); + } + else if (name == PROP_THREAD) + { + if (Poco::icompare(value, "none"s) == 0) + _threadFormat = JSONF_THREAD_NONE; + else if (Poco::icompare(value, "name"s) == 0) + _threadFormat = JSONF_THREAD_NAME; + else if (Poco::icompare(value, "id"s) == 0) + _threadFormat = JSONF_THREAD_ID; + else if (Poco::icompare(value, "osid"s) == 0) + _threadFormat = JSONF_THREAD_OS_ID; + else + throw Poco::InvalidArgumentException("Invalid thread value (must be name, id or osID)"s, value); + } + else throw Poco::PropertyNotSupportedException(name); +} + + +std::string JSONFormatter::getProperty(const std::string& name) const +{ + if (name == PROP_TIMES) + { + return _localTime ? "local"s : "UTC"s; + } + else if (name == PROP_THREAD) + { + switch (_threadFormat) + { + case JSONF_THREAD_NONE: + return "none"s; + case JSONF_THREAD_NAME: + return "name"s; + case JSONF_THREAD_ID: + return "id"s; + case JSONF_THREAD_OS_ID: + return "osID"s; + default: + return "invalid"s; + } + } + else throw Poco::PropertyNotSupportedException(name); +} + + +std::string JSONFormatter::getThread(const Message& message) const +{ + switch (_threadFormat) + { + case JSONF_THREAD_NONE: + return ""s; + case JSONF_THREAD_NAME: + return toJSON(message.getThread()); + case JSONF_THREAD_ID: + return Poco::NumberFormatter::format(message.getTid()); + case JSONF_THREAD_OS_ID: + return Poco::NumberFormatter::format(message.getOsTid()); + default: + return ""s; + } +} + + +const std::string& JSONFormatter::getPriorityName(int prio) +{ + static const std::string PRIORITY_NAMES[] = { + "none"s, + "fatal"s, + "critical"s, + "error"s, + "warning"s, + "notice"s, + "information"s, + "debug"s, + "trace" + }; + + poco_assert (prio >= Message::PRIO_FATAL && prio <= Message::PRIO_TRACE); + + return PRIORITY_NAMES[prio]; +} + + +} // namespace Poco diff --git a/Foundation/src/LoggingFactory.cpp b/Foundation/src/LoggingFactory.cpp index 049c49e50..6563d22b2 100644 --- a/Foundation/src/LoggingFactory.cpp +++ b/Foundation/src/LoggingFactory.cpp @@ -29,6 +29,7 @@ #include "Poco/WindowsConsoleChannel.h" #endif #include "Poco/PatternFormatter.h" +#include "Poco/JSONFormatter.h" using namespace std::string_literals; @@ -112,6 +113,7 @@ void LoggingFactory::registerBuiltins() #endif _formatterFactory.registerClass("PatternFormatter"s, new Instantiator); + _formatterFactory.registerClass("JSONFormatter"s, new Instantiator); } diff --git a/Foundation/testsuite/Makefile-Driver b/Foundation/testsuite/Makefile-Driver index 264fe2c15..0eb16a9f9 100644 --- a/Foundation/testsuite/Makefile-Driver +++ b/Foundation/testsuite/Makefile-Driver @@ -20,8 +20,8 @@ objects = ActiveMethodTest ActivityTest ActiveDispatcherTest \ NDCTest NotificationCenterTest NotificationQueueTest \ PriorityNotificationQueueTest TimedNotificationQueueTest \ NotificationsTestSuite NullStreamTest NumberFormatterTest \ - NumberParserTest PathTest PatternFormatterTest PBKDF2EngineTest ProcessRunnerTest RWLockTest \ - RandomStreamTest RandomTest RegularExpressionTest SHA1EngineTest SHA2EngineTest \ + NumberParserTest PathTest PatternFormatterTest JSONFormatterTest PBKDF2EngineTest ProcessRunnerTest \ + RWLockTest RandomStreamTest RandomTest RegularExpressionTest SHA1EngineTest SHA2EngineTest \ SemaphoreTest ConditionTest SharedLibraryTest SharedLibraryTestSuite \ SimpleFileChannelTest StopwatchTest \ StreamConverterTest StreamCopierTest StreamTokenizerTest \ diff --git a/Foundation/testsuite/TestSuite_vs160.vcxproj b/Foundation/testsuite/TestSuite_vs160.vcxproj index 1da0e5e61..41a1fa60a 100644 --- a/Foundation/testsuite/TestSuite_vs160.vcxproj +++ b/Foundation/testsuite/TestSuite_vs160.vcxproj @@ -674,6 +674,7 @@ + @@ -817,6 +818,7 @@ + diff --git a/Foundation/testsuite/TestSuite_vs160.vcxproj.filters b/Foundation/testsuite/TestSuite_vs160.vcxproj.filters index 0c534981e..e26995529 100644 --- a/Foundation/testsuite/TestSuite_vs160.vcxproj.filters +++ b/Foundation/testsuite/TestSuite_vs160.vcxproj.filters @@ -603,6 +603,9 @@ Threading\Source Files + + Logging\Source Files + @@ -1028,5 +1031,8 @@ Threading\Header Files + + Logging\Header Files + \ No newline at end of file diff --git a/Foundation/testsuite/TestSuite_vs170.vcxproj b/Foundation/testsuite/TestSuite_vs170.vcxproj index d91f6c533..91e3d7910 100644 --- a/Foundation/testsuite/TestSuite_vs170.vcxproj +++ b/Foundation/testsuite/TestSuite_vs170.vcxproj @@ -1007,6 +1007,7 @@ + @@ -1150,6 +1151,7 @@ + diff --git a/Foundation/testsuite/TestSuite_vs170.vcxproj.filters b/Foundation/testsuite/TestSuite_vs170.vcxproj.filters index 0c534981e..e26995529 100644 --- a/Foundation/testsuite/TestSuite_vs170.vcxproj.filters +++ b/Foundation/testsuite/TestSuite_vs170.vcxproj.filters @@ -603,6 +603,9 @@ Threading\Source Files + + Logging\Source Files + @@ -1028,5 +1031,8 @@ Threading\Header Files + + Logging\Header Files + \ No newline at end of file diff --git a/Foundation/testsuite/src/JSONFormatterTest.cpp b/Foundation/testsuite/src/JSONFormatterTest.cpp new file mode 100644 index 000000000..d409f675b --- /dev/null +++ b/Foundation/testsuite/src/JSONFormatterTest.cpp @@ -0,0 +1,94 @@ +// +// JSONFormatterTest.cpp +// +// Copyright (c) 2024, Applied Informatics Software Engineering GmbH. +// and Contributors. +// +// SPDX-License-Identifier: BSL-1.0 +// + + +#include "JSONFormatterTest.h" +#include "CppUnit/TestCaller.h" +#include "CppUnit/TestSuite.h" +#include "Poco/JSONFormatter.h" +#include "Poco/Message.h" +#include "Poco/DateTime.h" + + +using Poco::JSONFormatter; +using Poco::Message; +using Poco::DateTime; + + +JSONFormatterTest::JSONFormatterTest(const std::string& name): CppUnit::TestCase(name) +{ +} + + +JSONFormatterTest::~JSONFormatterTest() +{ +} + + +void JSONFormatterTest::testJSONFormatter() +{ + Message msg; + JSONFormatter fmt; + msg.setSource("TestSource"); + msg.setText("Test message text"); + msg.setPid(1234); + msg.setTid(1); + msg.setThread("TestThread"); + msg.setPriority(Message::PRIO_ERROR); + msg.setTime(DateTime(2005, 1, 1, 14, 30, 15, 500).timestamp()); + + std::string result; + fmt.format(msg, result); + assertTrue (result == "{\"timestamp\":\"2005-01-01T14:30:15.500000Z\",\"source\":\"TestSource\",\"level\":\"error\",\"message\":\"Test message text\",\"thread\":1}"); + + msg.setText("Multi\nline\ntext"); + result.clear(); + fmt.format(msg, result); + assertTrue (result == "{\"timestamp\":\"2005-01-01T14:30:15.500000Z\",\"source\":\"TestSource\",\"level\":\"error\",\"message\":\"Multi\\nline\\ntext\",\"thread\":1}"); + + fmt.setProperty("thread", "none"); + result.clear(); + fmt.format(msg, result); + assertTrue (result == "{\"timestamp\":\"2005-01-01T14:30:15.500000Z\",\"source\":\"TestSource\",\"level\":\"error\",\"message\":\"Multi\\nline\\ntext\"}"); + + msg.set("p1", "v1"); + result.clear(); + fmt.format(msg, result); + assertTrue (result == "{\"timestamp\":\"2005-01-01T14:30:15.500000Z\",\"source\":\"TestSource\",\"level\":\"error\",\"message\":\"Multi\\nline\\ntext\",\"params\":{\"p1\":\"v1\"}}"); + + msg.set("p2", "v2"); + result.clear(); + fmt.format(msg, result); + assertTrue (result == "{\"timestamp\":\"2005-01-01T14:30:15.500000Z\",\"source\":\"TestSource\",\"level\":\"error\",\"message\":\"Multi\\nline\\ntext\",\"params\":{\"p1\":\"v1\",\"p2\":\"v2\"}}"); + + fmt.setProperty("thread", "name"); + result.clear(); + fmt.format(msg, result); + assertTrue (result == "{\"timestamp\":\"2005-01-01T14:30:15.500000Z\",\"source\":\"TestSource\",\"level\":\"error\",\"message\":\"Multi\\nline\\ntext\",\"thread\":\"TestThread\",\"params\":{\"p1\":\"v1\",\"p2\":\"v2\"}}"); +} + + +void JSONFormatterTest::setUp() +{ +} + + +void JSONFormatterTest::tearDown() +{ +} + + +CppUnit::Test* JSONFormatterTest::suite() +{ + CppUnit::TestSuite* pSuite = new CppUnit::TestSuite("JSONFormatterTest"); + + CppUnit_addTest(pSuite, JSONFormatterTest, testJSONFormatter); + + return pSuite; +} diff --git a/Foundation/testsuite/src/JSONFormatterTest.h b/Foundation/testsuite/src/JSONFormatterTest.h new file mode 100644 index 000000000..ae18b5dbc --- /dev/null +++ b/Foundation/testsuite/src/JSONFormatterTest.h @@ -0,0 +1,38 @@ +// +// JSONFormatterTest.h +// +// Definition of the JSONFormatterTest class. +// +// Copyright (c) 2024, Applied Informatics Software Engineering GmbH. +// and Contributors. +// +// SPDX-License-Identifier: BSL-1.0 +// + + +#ifndef JSONFormatterTest_INCLUDED +#define JSONFormatterTest_INCLUDED + + +#include "Poco/Foundation.h" +#include "CppUnit/TestCase.h" + + +class JSONFormatterTest: public CppUnit::TestCase +{ +public: + JSONFormatterTest(const std::string& name); + ~JSONFormatterTest(); + + void testJSONFormatter(); + + void setUp(); + void tearDown(); + + static CppUnit::Test* suite(); + +private: +}; + + +#endif // JSONFormatterTest_INCLUDED diff --git a/Foundation/testsuite/src/LoggingTestSuite.cpp b/Foundation/testsuite/src/LoggingTestSuite.cpp index e55e9feb7..2e365fdf8 100644 --- a/Foundation/testsuite/src/LoggingTestSuite.cpp +++ b/Foundation/testsuite/src/LoggingTestSuite.cpp @@ -12,6 +12,7 @@ #include "LoggerTest.h" #include "ChannelTest.h" #include "PatternFormatterTest.h" +#include "JSONFormatterTest.h" #include "FileChannelTest.h" #include "SimpleFileChannelTest.h" #include "LoggingFactoryTest.h" @@ -26,6 +27,7 @@ CppUnit::Test* LoggingTestSuite::suite() pSuite->addTest(LoggerTest::suite()); pSuite->addTest(ChannelTest::suite()); pSuite->addTest(PatternFormatterTest::suite()); + pSuite->addTest(JSONFormatterTest::suite()); pSuite->addTest(FileChannelTest::suite()); pSuite->addTest(SimpleFileChannelTest::suite()); pSuite->addTest(LoggingFactoryTest::suite());