mirror of
https://github.com/wnagrodzki/SwiftLogger.git
synced 2025-04-06 12:31:51 +02:00
Added comments
This commit is contained in:
parent
a510af3278
commit
fd7edb0c0d
8 changed files with 106 additions and 0 deletions
|
@ -24,11 +24,15 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// A type with a customized textual representation suitable for logging purposes.
|
||||
public protocol LogStringConvertible {
|
||||
|
||||
/// A textual representation of this instance, suitable for logging.
|
||||
var logDescription: String { get }
|
||||
}
|
||||
|
||||
extension Array: LogStringConvertible where Element: LogStringConvertible {
|
||||
|
||||
public var logDescription: String {
|
||||
let descriptions = map { $0.logDescription }
|
||||
let joinedDescriptions = descriptions.joined(separator: ", ")
|
||||
|
|
|
@ -24,23 +24,75 @@
|
|||
|
||||
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: Part of the log message provided by `description(for date: Date)` method.
|
||||
/// - level: The log level.
|
||||
/// - location: Part of the log message provided by `description(for file: String, line: Int, function: String)` method.
|
||||
/// - object: Part of the log message provided by `description(for object: Any)` method.
|
||||
func log(time: String, level: LogLevel, location: String, object: String)
|
||||
|
||||
/// Transforms `date` into textual representation which will be a part of the log message.
|
||||
///
|
||||
/// Customization point. Default implementation returns date in format `"yyyy-MM-dd HH:mm:ss.SSS"`.
|
||||
///
|
||||
/// - Parameter date: The date log method was called.
|
||||
func description(for date: Date) -> 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 `"<file name>:<line> <function>"`,
|
||||
///
|
||||
/// - 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
|
||||
|
||||
/// Transforms `object` into textual representation which will be a part of the log message.
|
||||
///
|
||||
/// Customization point. Default implementation returns `logDescription` for objects implementing `LogStringConvertible`,
|
||||
/// or `String(describing:object)` otherwise.
|
||||
///
|
||||
/// - Parameter object: The object to be logged.
|
||||
func description(for object: Any) -> String
|
||||
}
|
||||
|
||||
/// Log level controls the conditions under which a message should be logged.
|
||||
public enum LogLevel: String {
|
||||
|
||||
/// Use this level to capture information about things that might result a failure.
|
||||
case `default` = "Default"
|
||||
|
||||
/// Use this level to capture information that may be helpful, but isn’t essential, for troubleshooting errors.
|
||||
case info = "Info"
|
||||
|
||||
/// Use this level to capture information that may be useful during development or while troubleshooting a specific problem.
|
||||
case debug = "Debug"
|
||||
|
||||
/// Use this log level to capture process-level information to report errors in the process.
|
||||
case error = "Error"
|
||||
|
||||
/// Use this level to capture system-level or multi-process information to report system errors.
|
||||
case fault = "Fault"
|
||||
}
|
||||
|
||||
extension Logger {
|
||||
|
||||
/// Sends object to the logging system, optionally specifying a custom log level.
|
||||
///
|
||||
/// - Parameters:
|
||||
/// - object: The object to be logged.
|
||||
/// - level: The log level. If unspecified, the `default` log level is used.
|
||||
/// - 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(_ object: Any, level: LogLevel = .default, file: String = #file, line: Int = #line, function: String = #function) {
|
||||
let now = Date()
|
||||
let time = description(for: now)
|
||||
|
@ -49,14 +101,17 @@ extension Logger {
|
|||
log(time: time, level: level, location: location, object: objectDescription)
|
||||
}
|
||||
|
||||
/// Returns date in format `"yyyy-MM-dd HH:mm:ss.SSS"`.
|
||||
public func description(for date: Date) -> String {
|
||||
return dateFormatter.string(from: date)
|
||||
}
|
||||
|
||||
/// Returns location in format `"<file name>:<line> <function>"`.
|
||||
public func description(for file: String, line: Int, function: String) -> String {
|
||||
return filename(fromFilePath: file) + ":\(line) \(function)"
|
||||
}
|
||||
|
||||
/// Returns `logDescription` for objects implementing `LogStringConvertible`, or `String(describing:object)` otherwise.
|
||||
public func description(for object: Any) -> String {
|
||||
if let logStringConvertible = object as? LogStringConvertible {
|
||||
return logStringConvertible.logDescription
|
||||
|
|
|
@ -24,9 +24,14 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// Logger that forwards messages to all the loggers it is intialized with.
|
||||
public final class AgregateLogger: Logger {
|
||||
|
||||
private let loggers: [Logger]
|
||||
|
||||
/// Initializes new AgregateLogger instance.
|
||||
///
|
||||
/// - Parameter loggers: Array of loggers all messages will be forwarded to.
|
||||
public init(loggers: [Logger]) {
|
||||
self.loggers = loggers
|
||||
}
|
||||
|
|
|
@ -24,6 +24,7 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// Logger that writes messages into the standard output.
|
||||
public final class ConsoleLogger: Logger {
|
||||
public init() {
|
||||
}
|
||||
|
|
|
@ -24,7 +24,9 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// Logger that writes messages into the file at specified URL with log rotation support.
|
||||
public final class DiskLogger: Logger {
|
||||
|
||||
private let fileURL: URL
|
||||
private let fileSizeLimit: UInt64
|
||||
private let rotations: Int
|
||||
|
@ -32,6 +34,12 @@ public final class DiskLogger: Logger {
|
|||
private var buffer = Data()
|
||||
private var fileWriter: FileWriter!
|
||||
|
||||
/// Initializes new DiskLogger instance.
|
||||
///
|
||||
/// - Parameters:
|
||||
/// - fileURL: URL of the log file.
|
||||
/// - fileSizeLimit: Maximum size log file can reach in bytes. Attempt to exceeding that limit triggers log files rotation.
|
||||
/// - rotations: Number of times log files are rotated before being removed.
|
||||
public init(fileURL: URL, fileSizeLimit: UInt64, rotations: Int) {
|
||||
self.fileURL = fileURL
|
||||
self.fileSizeLimit = fileSizeLimit
|
||||
|
|
|
@ -24,20 +24,33 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// Allows writing to a file while respecting allowed size limit.
|
||||
final class FileWriter {
|
||||
|
||||
/// Write failed as allowed size limit would be exceeded for the file.
|
||||
struct FileSizeLimitReached: Error {}
|
||||
|
||||
private let handle: FileHandle
|
||||
private let sizeLimit: UInt64
|
||||
private var currentSize: UInt64
|
||||
|
||||
/// Initializes new FileWriter instance.
|
||||
///
|
||||
/// - Parameters:
|
||||
/// - fileURL: URL of the file.
|
||||
/// - fileSizeLimit: Maximum size the file can reach in bytes.
|
||||
/// - Throws: An error that may occur while the file is being opened for writing.
|
||||
init(fileURL: URL, fileSizeLimit: UInt64) throws {
|
||||
handle = try FileHandle(forWritingTo: fileURL)
|
||||
self.sizeLimit = fileSizeLimit
|
||||
currentSize = handle.seekToEndOfFile()
|
||||
}
|
||||
|
||||
/// Synchronously writes `data` at the end of the file.
|
||||
///
|
||||
/// - Parameter data: The data to be written.
|
||||
/// - Throws: Throws an error if no free space is left on the file system, or if any other writing error occurs.
|
||||
/// Throws `FileSizeLimitReached` if allowed size limit would be exceeded for the file.
|
||||
func write(_ data: Data) throws {
|
||||
let dataSize = UInt64(data.count)
|
||||
guard currentSize + dataSize <= sizeLimit else {
|
||||
|
@ -47,6 +60,7 @@ final class FileWriter {
|
|||
currentSize += dataSize
|
||||
}
|
||||
|
||||
/// Writes all in-memory data to permanent storage and closes the file.
|
||||
func synchronizeAndCloseFile() {
|
||||
handle.synchronizeFile()
|
||||
handle.closeFile()
|
||||
|
|
|
@ -24,16 +24,31 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// Allows log files rotation.
|
||||
final class Logrotate {
|
||||
|
||||
private let fileURL: URL
|
||||
private let rotations: Int
|
||||
|
||||
/// Initializes new Logrotate instance.
|
||||
///
|
||||
/// - Parameters:
|
||||
/// - fileURL: URL of the log file.
|
||||
/// - rotations: Number of times log files are rotated before being removed.
|
||||
init(fileURL: URL, rotations: Int) {
|
||||
precondition(rotations > 0)
|
||||
self.fileURL = fileURL
|
||||
self.rotations = rotations
|
||||
}
|
||||
|
||||
/// Rotates log files `rotations` number of times.
|
||||
///
|
||||
/// First deletes file at `<fileURL>.<rotations>`.
|
||||
/// Next moves files located at:
|
||||
///
|
||||
/// `<fileURL>, <fileURL>.1, <fileURL>.2 ... <fileURL>.<rotations - 1>`
|
||||
///
|
||||
/// to `<fileURL>.1, <fileURL>.2 ... <fileURL>.<rotations>`
|
||||
func rotate() throws {
|
||||
let range = 1...rotations
|
||||
let pathExtensions = range.map { "\($0)" }
|
||||
|
|
|
@ -24,6 +24,7 @@
|
|||
|
||||
import Foundation
|
||||
|
||||
/// Logger that ignores all messages with the intention to minimize observer effect.
|
||||
public class NullLogger: Logger {
|
||||
public init() {
|
||||
}
|
||||
|
@ -32,14 +33,17 @@ public class NullLogger: Logger {
|
|||
// noop
|
||||
}
|
||||
|
||||
/// Returns empty string to minimize observer effect.
|
||||
public func description(for date: Date) -> String {
|
||||
return ""
|
||||
}
|
||||
|
||||
/// Returns empty string to minimize observer effect.
|
||||
public func description(for file: String, line: Int, function: String) -> String {
|
||||
return ""
|
||||
}
|
||||
|
||||
/// Returns empty string to minimize observer effect.
|
||||
public func description(for object: Any) -> String {
|
||||
return ""
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue