import Foundation /// All possible SwiftLint issues which are printed as warnings by default. public enum Issue: LocalizedError, Equatable { /// The configuration didn't match internal expectations. case invalidConfiguration(ruleID: String) /// Issued when an option is deprecated. Suggests an alternative optionally. case deprecatedConfigurationOption(ruleID: String, key: String, alternative: String? = nil) /// Used in configuration parsing when no changes have been applied. Use only internally! case nothingApplied(ruleID: String) /// Rule is listed multiple times in the configuration. case listedMultipleTime(ruleID: String, times: Int) /// An identifier `old` has been renamed to `new`. case renamedIdentifier(old: String, new: String) /// Some configuration keys are invalid. case invalidConfigurationKeys(ruleID: String, keys: Set) /// Used rule IDs are invalid. case invalidRuleIDs(Set) /// Found a rule configuration for a rule that is not present in `only_rules`. case ruleNotPresentInOnlyRules(ruleID: String) /// Found a rule configuration for a rule that is disabled. case ruleDisabledInDisabledRules(ruleID: String) /// Found a rule configuration for a rule that is disabled in the parent configuration. case ruleDisabledInParentConfiguration(ruleID: String) /// Found a rule configuration for a rule that is not enabled in `opt_in_rules`. case ruleNotEnabledInOptInRules(ruleID: String) /// Found a rule configuration for a rule that is not enabled in parent `only_rules`. case ruleNotEnabledInParentOnlyRules(ruleID: String) /// A generic warning specified by a string. case genericWarning(String) /// A generic error specified by a string. case genericError(String) /// A deprecation warning for a rule. case ruleDeprecated(ruleID: String) /// The initial configuration file was not found. case initialFileNotFound(path: String) /// A file at specified path was not found. case fileNotFound(path: String) /// The file at `path` is not readable or cannot be opened. case fileNotReadable(path: String?, ruleID: String) /// The file at `path` is not writable. case fileNotWritable(path: String) /// The file at `path` cannot be indexed by a specific rule. case indexingError(path: String?, ruleID: String) /// No arguments were provided to compile a file at `path` within a specific rule. case missingCompilerArguments(path: String?, ruleID: String) /// Cursor information cannot be extracted from a specific location. case missingCursorInfo(path: String?, ruleID: String) /// An error that occurred when parsing YAML. case yamlParsing(String) /// The baseline file at `path` is not readable or cannot be opened. case baselineNotReadable(path: String) /// Flag to enable warnings for deprecations being printed to the console. Printing is enabled by default. public static var printDeprecationWarnings = true /// Hook used to capture all messages normally printed to stdout and return them back to the caller. /// /// > Warning: Shall only be used in tests to verify console output. /// /// - parameter runner: The code to run. Messages printed during the execution are collected. /// /// - returns: The collected messages produced while running the code in the runner. static func captureConsole(runner: () throws -> Void) rethrows -> String { var console = "" messageConsumer = { console += (console.isEmpty ? "" : "\n") + $0 } defer { messageConsumer = nil } try runner() return console } private static var messageConsumer: ((String) -> Void)? /// Wraps any `Error` into a `SwiftLintError.genericWarning` if it is not already a `SwiftLintError`. /// /// - parameter error: Any `Error`. /// /// - returns: A `SwiftLintError.genericWarning` containing the message of the `error` argument. static func wrap(error: some Error) -> Self { error as? Issue ?? Self.genericWarning(error.localizedDescription) } /// Make this issue an error. var asError: Self { Self.genericError(message) } /// The issues description which is ready to be printed to the console. var errorDescription: String { switch self { case .genericError: return "error: \(message)" case .genericWarning: return "warning: \(message)" default: return Self.genericWarning(message).errorDescription } } /// Print the issue to the console. public func print() { if case .ruleDeprecated = self, !Self.printDeprecationWarnings { return } if let consumer = Self.messageConsumer { consumer(errorDescription) } else { queuedPrintError(errorDescription) } } private var message: String { switch self { case let .invalidConfiguration(id): return "Invalid configuration for '\(id)' rule. Falling back to default." case let .deprecatedConfigurationOption(id, key, alternative): let baseMessage = "Configuration option '\(key)' in '\(id)' rule is deprecated." if let alternative { return baseMessage + " Use the option '\(alternative)' instead." } return baseMessage case let .nothingApplied(ruleID: id): return Self.invalidConfiguration(ruleID: id).message case let .listedMultipleTime(id, times): return "'\(id)' is listed \(times) times in the configuration." case let .renamedIdentifier(old, new): return "'\(old)' has been renamed to '\(new)' and will be completely removed in a future release." case let .invalidConfigurationKeys(id, keys): return "Configuration for '\(id)' rule contains the invalid key(s) \(keys.formatted)." case let .invalidRuleIDs(ruleIDs): return "The key(s) \(ruleIDs.formatted) used as rule identifier(s) is/are invalid." case let .ruleNotPresentInOnlyRules(id): return "Found a configuration for '\(id)' rule, but it is not present in " + "'\(Configuration.Key.onlyRules.rawValue)'." case let .ruleDisabledInDisabledRules(id): return "Found a configuration for '\(id)' rule, but it is disabled in " + "'\(Configuration.Key.disabledRules.rawValue)'." case let .ruleDisabledInParentConfiguration(id): return "Found a configuration for '\(id)' rule, but it is disabled in a parent configuration." case let .ruleNotEnabledInOptInRules(id): return "Found a configuration for '\(id)' rule, but it is not enabled in " + "'\(Configuration.Key.optInRules.rawValue)'." case let .ruleNotEnabledInParentOnlyRules(id): return "Found a configuration for '\(id)' rule, but it is not present in the parent's " + "'\(Configuration.Key.onlyRules.rawValue)'." case let .genericWarning(message), let .genericError(message): return message case let .ruleDeprecated(id): return """ The `\(id)` rule is now deprecated and will be \ completely removed in a future release. """ case let .initialFileNotFound(path): return "Could not read file at path '\(path)'." case let .fileNotFound(path): return "File at path '\(path)' not found." case let .fileNotReadable(path, id): return "Cannot open or read file at path '\(path ?? "...")' within '\(id)' rule." case let .fileNotWritable(path): return "Cannot write to file at path '\(path)'." case let .indexingError(path, id): return "Cannot index file at path '\(path ?? "...")' within '\(id)' rule." case let .missingCompilerArguments(path, id): return """ Attempted to lint file at path '\(path ?? "...")' within '\(id)' rule \ without any compiler arguments. """ case let .missingCursorInfo(path, id): return "Cannot get cursor info from file at path '\(path ?? "...")' within '\(id)' rule." case let .yamlParsing(message): return "Cannot parse YAML file: \(message)" case let .baselineNotReadable(path): return "Cannot open or read the baseline file at path '\(path)'." } } } private extension Set where Element == String { var formatted: String { sorted() .map { "'\($0)'" } .joined(separator: ", ") } }