// // MIT License // // Copyright (c) 2018 Wojciech Nagrodzki // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in all // copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE // SOFTWARE. // import Foundation /// A type able to build custom log message and send it to a logging system. public protocol Logger { /// Builds log a message from passed parameters and sends it to the logging system. /// /// Do not call this method directly. /// /// - Parameters: /// - time: The date log method was called. /// - level: The log level. /// - location: Part of the log message provided by `description(for file: String, line: Int, function: String)` method. /// - message: The message to be logged. func log(time: Date, level: LogLevel, location: String, message: @autoclosure () -> String) /// Transforms passed parameters into location textual representation which will be a part of the log message. /// /// Customization point. Default implementation returns location in format `": "`, /// /// - Parameters: /// - file: The path to the file log method was called from. /// - line: The line number log method was called at. /// - function: The name of the declaration log method was called within. func description(for file: String, line: Int, function: String) -> String } /// Log level controls the conditions under which a message should be logged. /// - note: [RFC5424](https://www.rfc-editor.org/info/rfc5424) public enum LogLevel: Int { /// System is unusable. case emergency = 0 /// Action must be taken immediately. case alert = 1 /// Critical conditions. case critical = 2 /// Error conditions. case error = 3 /// Warning conditions. case warning = 4 /// Normal but significant conditions. case notice = 5 /// Informational messages. case informational = 6 /// Debug-level messages. case debug = 7 } extension LogLevel: LogStringConvertible { public var logDescription: String { switch self { case .emergency: return "emerg" case .alert: return "alert" case .critical: return "crit" case .error: return "err" case .warning: return "warning" case .notice: return "notice" case .informational: return "info" case .debug: return "debug" } } } extension Logger { /// Sends object to the logging system, optionally specifying a custom log level. /// /// - Parameters: /// - message: The message to be logged. /// - level: The log level. /// - file: **Do not provide a custom value.** The path to the file log method is called from. /// - line: **Do not provide a custom value.** The line number log method is called at. /// - function: **Do not provide a custom value.** The name of the declaration log method is called within. public func log(_ message: @autoclosure () -> String, level: LogLevel, file: String = #file, line: Int = #line, function: String = #function) { let now = Date() let location = description(for: file, line: line, function: function) log(time: now, level: level, location: location, message: message()) } /// Returns location in format `": "`. public func description(for file: String, line: Int, function: String) -> String { return filename(fromFilePath: file) + ":\(line) \(function)" } private func filename(fromFilePath path: String) -> String { return URL(fileURLWithPath: path).deletingPathExtension().lastPathComponent } }