From 993378506057cc6fae9d738571861aa6654f9f6b Mon Sep 17 00:00:00 2001 From: Jeff Ward Date: Mon, 14 Nov 2022 14:08:58 -0500 Subject: [PATCH 1/4] Add methods to pass error info to logs as decomposed strings This is mostly for Cross Platform SDKs that need to send in serialized error information. --- CHANGELOG.md | 2 ++ Sources/Datadog/Logger.swift | 31 ++++++++++++++++ Sources/Datadog/Logging/ConsoleLogger.swift | 35 +++++++++++++++---- Sources/Datadog/Logging/RemoteLogger.swift | 16 +++++++-- Tests/DatadogTests/Datadog/LoggerTests.swift | 25 +++++++++++++ .../Datadog/Logging/ConsoleLoggerTests.swift | 33 +++++++++++++++++ 6 files changed, 133 insertions(+), 9 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7bda1a4198..58cd236042 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,7 @@ # Unreleased +- [IMPROVEMENT] Add a method for sending error attributes on logs as strings. + # 1.13.0 / 08-11-2022 - [IMPROVEMENT] Improve console logs when using `DDNoopRUMMonitor`. See [#1007][] (Thanks [@dfed][]) diff --git a/Sources/Datadog/Logger.swift b/Sources/Datadog/Logger.swift index 9950854abc..007aea207a 100644 --- a/Sources/Datadog/Logger.swift +++ b/Sources/Datadog/Logger.swift @@ -49,6 +49,24 @@ public protocol LoggerProtocol { /// the same key already exist in this logger, it will be overridden (only for this message). func log(level: LogLevel, message: String, error: Error?, attributes: [String: Encodable]?) + /// General purpose logging method. + /// Sends a log with certain `level`, `message`, `errorKind`, `errorMessage`, `stackTrace` and `attributes`. + /// + /// This method is meant for non-native or cross platform frameworks (such as React Native of Flutter) to send error information + /// to Datadog. Although it can be used directly, it is recommended to use other methods declared on `Logger`. + /// + /// - Parameters: + /// - level: the log level + /// - message: the message to be logged + /// - errorKind: the kind of error reported + /// - errorMessage: the message attached to the error + /// - stackTrace: a string representation of the error's stack trace + /// - attributes: a dictionary of attributes (optional) to add for this message. If an attribute with + /// the same key already exist in this logger, it will be overridden (only for this message). + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, + stackTrace: String?, attributes: [String: Encodable]?) + + // MARK: - Attributes /// Adds a custom attribute to all future logs sent by this logger. @@ -186,6 +204,12 @@ public class Logger: LoggerProtocol { v2Logger.log(level: level, message: message, error: error, attributes: attributes) } + public func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, + stackTrace: String?, attributes: [String: Encodable]?) { + v2Logger.log(level: level, message: message, errorKind: errorKind, errorMessage: errorMessage, + stackTrace: stackTrace, attributes: attributes) + } + public func addAttribute(forKey key: AttributeKey, value: AttributeValue) { v2Logger.addAttribute(forKey: key, value: value) } @@ -421,6 +445,12 @@ internal struct CombinedLogger: LoggerProtocol { combinedLoggers.forEach { $0.log(level: level, message: message, error: error, attributes: attributes) } } + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, + stackTrace: String?, attributes: [String : Encodable]?) { + combinedLoggers.forEach { $0.log(level: level, message: message, errorKind: errorKind, errorMessage: errorMessage, + stackTrace: stackTrace, attributes: attributes) } + } + func addAttribute(forKey key: AttributeKey, value: AttributeValue) { combinedLoggers.forEach { $0.addAttribute(forKey: key, value: value) } } @@ -448,6 +478,7 @@ internal struct CombinedLogger: LoggerProtocol { internal struct NOPLogger: LoggerProtocol { func log(level: LogLevel, message: String, error: Error?, attributes: [String: Encodable]?) {} + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String : Encodable]?) {} func addAttribute(forKey key: AttributeKey, value: AttributeValue) {} func removeAttribute(forKey key: AttributeKey) {} func addTag(withKey key: String, value: String) {} diff --git a/Sources/Datadog/Logging/ConsoleLogger.swift b/Sources/Datadog/Logging/ConsoleLogger.swift index 8eb2da46a9..6ea183d2f4 100644 --- a/Sources/Datadog/Logging/ConsoleLogger.swift +++ b/Sources/Datadog/Logging/ConsoleLogger.swift @@ -45,24 +45,45 @@ internal final class ConsoleLogger: LoggerProtocol { // MARK: - Logging func log(level: LogLevel, message: String, error: Error?, attributes: [String: Encodable]?) { + var errorString: String? = nil + if let error = error { + let ddError = DDError(error: error) + errorString = buildErrorString(error: ddError) + } + + internalLog(level: level, message: message, errorString: errorString) + } + + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String : Encodable]?) { + var errorString: String? = nil + if errorKind != nil || errorMessage != nil || stackTrace != nil { + // Cross platform frameworks don't necessarilly send all values for errors. Send empty strings + // for any values that are empty. + let ddError = DDError(type: errorKind ?? "", message: errorMessage ?? "", stack: stackTrace ?? "") + errorString = buildErrorString(error: ddError) + } + + internalLog(level: level, message: message, errorString: errorString) + } + + private func internalLog(level: LogLevel, message: String, errorString: String?) { let time = timeFormatter.string(from: dateProvider.now) let status = level.asLogStatus.rawValue.uppercased() var log = "\(self.prefix)\(time) [\(status)] \(message)" - if let error = error { - log += "\n\nError details:\n\(buildErrorString(error: error))" + if let errorString = errorString { + log += "\n\nError details:\n\(errorString)" } printFunction(log) } - private func buildErrorString(error: Error) -> String { - let dderror = DDError(error: error) + private func buildErrorString(error: DDError) -> String { return """ - → type: \(dderror.type) - → message: \(dderror.message) - → stack: \(dderror.stack) + → type: \(error.type) + → message: \(error.message) + → stack: \(error.stack) """ } diff --git a/Sources/Datadog/Logging/RemoteLogger.swift b/Sources/Datadog/Logging/RemoteLogger.swift index 036a27a761..dd24c58c4b 100644 --- a/Sources/Datadog/Logging/RemoteLogger.swift +++ b/Sources/Datadog/Logging/RemoteLogger.swift @@ -95,9 +95,21 @@ internal final class RemoteLogger: LoggerProtocol { // MARK: - Logging func log(level: LogLevel, message: String, error: Error?, attributes: [String: Encodable]?) { - guard configuration.sampler.sample() else { - return + internalLog(level: level, message: message, error: error.map { DDError(error: $0) }, attributes: attributes) + } + + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String : Encodable]?) { + var ddError: DDError? + if errorKind != nil || errorMessage != nil || stackTrace != nil { + // Cross platform frameworks don't necessarilly send all values for errors. Send empty strings + // for any values that are empty. + ddError = DDError(type: errorKind ?? "", message: errorMessage ?? "", stack: stackTrace ?? "") } + + internalLog(level: level, message: message, error: ddError, attributes: attributes) + } + + func internalLog(level: LogLevel, message: String, error: DDError?, attributes: [String : Encodable]?) { guard level.rawValue >= configuration.threshold.rawValue else { return } diff --git a/Tests/DatadogTests/Datadog/LoggerTests.swift b/Tests/DatadogTests/Datadog/LoggerTests.swift index 9fb96f3c7f..1f421b88ef 100644 --- a/Tests/DatadogTests/Datadog/LoggerTests.swift +++ b/Tests/DatadogTests/Datadog/LoggerTests.swift @@ -185,6 +185,31 @@ class LoggerTests: XCTestCase { } } + func testLoggingErrorStrings() throws { + core.context = .mockAny() + + let feature: LoggingFeature = .mockByRecordingLogMatchers() + core.register(feature: feature) + + + let logger = Logger.builder.build(in: core) + let errorKind = String.mockRandom() + let errorMessage = String.mockRandom() + let stackTrace = String.mockRandom() + logger.log(level: .info, message: .mockAny(), errorKind: errorKind, errorMessage: errorMessage, + stackTrace: stackTrace, attributes: nil) + + + let logMatchers = try feature.waitAndReturnLogMatchers(count: 1) + let logMatcher = logMatchers.first + XCTAssertNotNil(logMatcher) + if let logMatcher = logMatcher { + logMatcher.assertValue(forKeyPath: "error.kind", equals: errorKind) + logMatcher.assertValue(forKeyPath: "error.message", equals: errorMessage) + logMatcher.assertValue(forKeyPath: "error.stack", equals: stackTrace) + } + } + // MARK: - Sampling func testSamplingEnabled() { diff --git a/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift b/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift index fb65b576fd..164ff4bddd 100644 --- a/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift +++ b/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift @@ -86,4 +86,37 @@ class ConsoleLoggerTests: XCTestCase { } XCTAssertEqual(mock.printedMessages.count, 6) } + + func testItPrintsErrorStringsWithExpectedFormat() { + // Given + let logger = ConsoleLogger( + configuration: .init( + timeZone: .UTC, + format: .short + ), + dateProvider: RelativeDateProvider( + using: .mockDecember15th2019At10AMUTC() + ), + printFunction: mock.print + ) + + let message = String.mockRandom() + let errorKind = String.mockRandom() + let errorMessage = String.mockRandom() + let stackTrace = String.mockRandom() + + logger.log(level: .info, message: message, errorKind: errorKind, errorMessage: errorMessage, + stackTrace: stackTrace, attributes: nil) + // Then + let expectedMessage = """ + 10:00:00.000 [INFO] \(message) + + Error details: + → type: \(errorKind) + → message: \(errorMessage) + → stack: \(stackTrace) + """ + XCTAssertEqual(mock.printedMessages.first, expectedMessage) + XCTAssertEqual(mock.printedMessages.count, 1) + } } From 140f4376d2610aa4a902ba33e78fc44d34be7591 Mon Sep 17 00:00:00 2001 From: Jeff Ward Date: Tue, 15 Nov 2022 08:27:35 -0500 Subject: [PATCH 2/4] Update Sources/Datadog/Logger.swift Co-authored-by: Pedro Lousada --- Sources/Datadog/Logger.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/Datadog/Logger.swift b/Sources/Datadog/Logger.swift index 007aea207a..a30584d3a2 100644 --- a/Sources/Datadog/Logger.swift +++ b/Sources/Datadog/Logger.swift @@ -52,7 +52,7 @@ public protocol LoggerProtocol { /// General purpose logging method. /// Sends a log with certain `level`, `message`, `errorKind`, `errorMessage`, `stackTrace` and `attributes`. /// - /// This method is meant for non-native or cross platform frameworks (such as React Native of Flutter) to send error information + /// This method is meant for non-native or cross platform frameworks (such as React Native or Flutter) to send error information /// to Datadog. Although it can be used directly, it is recommended to use other methods declared on `Logger`. /// /// - Parameters: From f9738c207418f23d73aa8d4c5290c858964c5247 Mon Sep 17 00:00:00 2001 From: Jeff Ward Date: Tue, 15 Nov 2022 08:41:14 -0500 Subject: [PATCH 3/4] Swiftlint fixes --- Sources/Datadog/Logger.swift | 53 ++++++++++++++----- Sources/Datadog/Logging/ConsoleLogger.swift | 2 +- Sources/Datadog/Logging/RemoteLogger.swift | 7 ++- Tests/DatadogTests/Datadog/LoggerTests.swift | 11 ++-- .../Datadog/Logging/ConsoleLoggerTests.swift | 11 +++- 5 files changed, 63 insertions(+), 21 deletions(-) diff --git a/Sources/Datadog/Logger.swift b/Sources/Datadog/Logger.swift index a30584d3a2..135f871cc2 100644 --- a/Sources/Datadog/Logger.swift +++ b/Sources/Datadog/Logger.swift @@ -63,9 +63,14 @@ public protocol LoggerProtocol { /// - stackTrace: a string representation of the error's stack trace /// - attributes: a dictionary of attributes (optional) to add for this message. If an attribute with /// the same key already exist in this logger, it will be overridden (only for this message). - func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, - stackTrace: String?, attributes: [String: Encodable]?) - + func log( + level: LogLevel, + message: String, + errorKind: String?, + errorMessage: String?, + stackTrace: String?, + attributes: [String: Encodable]? + ) // MARK: - Attributes @@ -204,10 +209,21 @@ public class Logger: LoggerProtocol { v2Logger.log(level: level, message: message, error: error, attributes: attributes) } - public func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, - stackTrace: String?, attributes: [String: Encodable]?) { - v2Logger.log(level: level, message: message, errorKind: errorKind, errorMessage: errorMessage, - stackTrace: stackTrace, attributes: attributes) + public func log( + level: LogLevel, + message: String, + errorKind: String?, + errorMessage: String?, + stackTrace: String?, + attributes: [String: Encodable]?) { + v2Logger.log( + level: level, + message: message, + errorKind: errorKind, + errorMessage: errorMessage, + stackTrace: stackTrace, + attributes: attributes + ) } public func addAttribute(forKey key: AttributeKey, value: AttributeValue) { @@ -445,10 +461,23 @@ internal struct CombinedLogger: LoggerProtocol { combinedLoggers.forEach { $0.log(level: level, message: message, error: error, attributes: attributes) } } - func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, - stackTrace: String?, attributes: [String : Encodable]?) { - combinedLoggers.forEach { $0.log(level: level, message: message, errorKind: errorKind, errorMessage: errorMessage, - stackTrace: stackTrace, attributes: attributes) } + func log( + level: LogLevel, + message: String, + errorKind: String?, + errorMessage: String?, + stackTrace: String?, + attributes: [String: Encodable]?) { + combinedLoggers.forEach { + $0.log( + level: level, + message: message, + errorKind: errorKind, + errorMessage: errorMessage, + stackTrace: stackTrace, + attributes: attributes + ) + } } func addAttribute(forKey key: AttributeKey, value: AttributeValue) { @@ -478,7 +507,7 @@ internal struct CombinedLogger: LoggerProtocol { internal struct NOPLogger: LoggerProtocol { func log(level: LogLevel, message: String, error: Error?, attributes: [String: Encodable]?) {} - func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String : Encodable]?) {} + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String: Encodable]?) {} func addAttribute(forKey key: AttributeKey, value: AttributeValue) {} func removeAttribute(forKey key: AttributeKey) {} func addTag(withKey key: String, value: String) {} diff --git a/Sources/Datadog/Logging/ConsoleLogger.swift b/Sources/Datadog/Logging/ConsoleLogger.swift index 6ea183d2f4..d6f1b879ee 100644 --- a/Sources/Datadog/Logging/ConsoleLogger.swift +++ b/Sources/Datadog/Logging/ConsoleLogger.swift @@ -54,7 +54,7 @@ internal final class ConsoleLogger: LoggerProtocol { internalLog(level: level, message: message, errorString: errorString) } - func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String : Encodable]?) { + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String: Encodable]?) { var errorString: String? = nil if errorKind != nil || errorMessage != nil || stackTrace != nil { // Cross platform frameworks don't necessarilly send all values for errors. Send empty strings diff --git a/Sources/Datadog/Logging/RemoteLogger.swift b/Sources/Datadog/Logging/RemoteLogger.swift index dd24c58c4b..fa8b8e81db 100644 --- a/Sources/Datadog/Logging/RemoteLogger.swift +++ b/Sources/Datadog/Logging/RemoteLogger.swift @@ -98,7 +98,7 @@ internal final class RemoteLogger: LoggerProtocol { internalLog(level: level, message: message, error: error.map { DDError(error: $0) }, attributes: attributes) } - func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String : Encodable]?) { + func log(level: LogLevel, message: String, errorKind: String?, errorMessage: String?, stackTrace: String?, attributes: [String: Encodable]?) { var ddError: DDError? if errorKind != nil || errorMessage != nil || stackTrace != nil { // Cross platform frameworks don't necessarilly send all values for errors. Send empty strings @@ -109,7 +109,10 @@ internal final class RemoteLogger: LoggerProtocol { internalLog(level: level, message: message, error: ddError, attributes: attributes) } - func internalLog(level: LogLevel, message: String, error: DDError?, attributes: [String : Encodable]?) { + func internalLog(level: LogLevel, message: String, error: DDError?, attributes: [String: Encodable]?) { + guard configuration.sampler.sample() else { + return + } guard level.rawValue >= configuration.threshold.rawValue else { return } diff --git a/Tests/DatadogTests/Datadog/LoggerTests.swift b/Tests/DatadogTests/Datadog/LoggerTests.swift index 1f421b88ef..6aaa67679d 100644 --- a/Tests/DatadogTests/Datadog/LoggerTests.swift +++ b/Tests/DatadogTests/Datadog/LoggerTests.swift @@ -191,14 +191,17 @@ class LoggerTests: XCTestCase { let feature: LoggingFeature = .mockByRecordingLogMatchers() core.register(feature: feature) - let logger = Logger.builder.build(in: core) let errorKind = String.mockRandom() let errorMessage = String.mockRandom() let stackTrace = String.mockRandom() - logger.log(level: .info, message: .mockAny(), errorKind: errorKind, errorMessage: errorMessage, - stackTrace: stackTrace, attributes: nil) - + logger.log(level: .info, + message: .mockAny(), + errorKind: errorKind, + errorMessage: errorMessage, + stackTrace: stackTrace, + attributes: nil + ) let logMatchers = try feature.waitAndReturnLogMatchers(count: 1) let logMatcher = logMatchers.first diff --git a/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift b/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift index 164ff4bddd..c751430751 100644 --- a/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift +++ b/Tests/DatadogTests/Datadog/Logging/ConsoleLoggerTests.swift @@ -105,8 +105,15 @@ class ConsoleLoggerTests: XCTestCase { let errorMessage = String.mockRandom() let stackTrace = String.mockRandom() - logger.log(level: .info, message: message, errorKind: errorKind, errorMessage: errorMessage, - stackTrace: stackTrace, attributes: nil) + logger.log( + level: .info, + message: message, + errorKind: errorKind, + errorMessage: errorMessage, + stackTrace: stackTrace, + attributes: nil + ) + // Then let expectedMessage = """ 10:00:00.000 [INFO] \(message) From 396b512ab62633b62928066d915e7ea75b17e50d Mon Sep 17 00:00:00 2001 From: Jeff Ward Date: Wed, 16 Nov 2022 11:51:35 -0500 Subject: [PATCH 4/4] Fix merge conflict --- Sources/Datadog/Logging/RemoteLogger.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/Datadog/Logging/RemoteLogger.swift b/Sources/Datadog/Logging/RemoteLogger.swift index fa8b8e81db..33771be0d8 100644 --- a/Sources/Datadog/Logging/RemoteLogger.swift +++ b/Sources/Datadog/Logging/RemoteLogger.swift @@ -150,7 +150,7 @@ internal final class RemoteLogger: LoggerProtocol { date: date, level: level, message: message, - error: error.map { DDError(error: $0) }, + error: error, attributes: .init( userAttributes: userAttributes, internalAttributes: internalAttributes