Files
2025-05-21 11:34:46 +01:00

213 lines
9.3 KiB
Swift

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, message: String? = nil)
/// Issued when a regular expression pattern is invalid.
case invalidRegexPattern(ruleID: String, pattern: 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<String>)
/// The configuration is inconsistent, that is options are mutually exclusive or one drives other values
/// irrelevant.
case inconsistentConfiguration(ruleID: String, message: String)
/// Used rule IDs are invalid.
case invalidRuleIDs(Set<String>)
/// 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.
package nonisolated(unsafe) static var printDeprecationWarnings = true
@TaskLocal private static var printQueueContinuation: AsyncStream<String>.Continuation?
/// 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.
@MainActor
static func captureConsole(runner: @Sendable () throws -> Void) async rethrows -> String {
let (stream, continuation) = AsyncStream.makeStream(of: String.self)
try $printQueueContinuation.withValue(continuation, operation: runner)
continuation.finish()
return await stream.reduce(into: "") { @Sendable in $0 += $0.isEmpty ? $1 : "\n\($1)" }
}
/// 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.
package static func wrap(error: some Error) -> Self {
error as? Self ?? Self.genericWarning(error.localizedDescription)
}
/// Make this issue an error.
package var asError: Self {
Self.genericError(message)
}
/// The issues description which is ready to be printed to the console.
public 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
}
Self.printQueueContinuation?.yield(localizedDescription)
queuedPrintError(localizedDescription)
}
private var message: String {
switch self {
case let .invalidConfiguration(id, message):
let message = if let message { ": \(message)" } else { "." }
return "Invalid configuration for '\(id)' rule\(message) Falling back to default."
case let .invalidRegexPattern(id, pattern):
return "Invalid regular expression pattern '\(pattern)' used to configure '\(id)' rule."
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 .inconsistentConfiguration(id, message):
return "Inconsistent configuration for '\(id)' rule: \(message)"
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 'only_rules'."
case let .ruleDisabledInDisabledRules(id):
return "Found a configuration for '\(id)' rule, but it is disabled in 'disabled_rules'."
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 'opt_in_rules'."
case let .ruleNotEnabledInParentOnlyRules(id):
return "Found a configuration for '\(id)' rule, but it is not present in the parent's 'only_rules'."
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: ", ")
}
}