mirror of
https://github.com/realm/SwiftLint.git
synced 2026-06-06 20:18:40 +00:00
449190d324
A rule must conform to ManuallyTestedExamplesRule to skip generation of a test for its examples.
178 lines
7.2 KiB
Swift
178 lines
7.2 KiB
Swift
import Foundation
|
|
import SourceKittenFramework
|
|
|
|
private func wrapInSwitch(
|
|
variable: String = "foo",
|
|
_ str: String,
|
|
file: StaticString = #file, line: UInt = #line) -> Example {
|
|
return Example(
|
|
"""
|
|
switch \(variable) {
|
|
\(str): break
|
|
}
|
|
""", file: file, line: line)
|
|
}
|
|
|
|
private func wrapInFunc(_ str: String, file: StaticString = #file, line: UInt = #line) -> Example {
|
|
return Example("""
|
|
func example(foo: Foo) {
|
|
switch foo {
|
|
case \(str):
|
|
break
|
|
}
|
|
}
|
|
""", file: file, line: line)
|
|
}
|
|
|
|
public struct EmptyEnumArgumentsRule: SubstitutionCorrectableASTRule, ConfigurationProviderRule {
|
|
public var configuration = SeverityConfiguration(.warning)
|
|
|
|
public init() {}
|
|
|
|
public static let description = RuleDescription(
|
|
identifier: "empty_enum_arguments",
|
|
name: "Empty Enum Arguments",
|
|
description: "Arguments can be omitted when matching enums with associated values if they are not used.",
|
|
kind: .style,
|
|
nonTriggeringExamples: [
|
|
wrapInSwitch("case .bar"),
|
|
wrapInSwitch("case .bar(let x)"),
|
|
wrapInSwitch("case let .bar(x)"),
|
|
wrapInSwitch(variable: "(foo, bar)", "case (_, _)"),
|
|
wrapInSwitch("case \"bar\".uppercased()"),
|
|
wrapInSwitch(variable: "(foo, bar)", "case (_, _) where !something"),
|
|
wrapInSwitch("case (let f as () -> String)?"),
|
|
wrapInSwitch("default"),
|
|
Example("if case .bar = foo {\n}"),
|
|
Example("guard case .bar = foo else {\n}"),
|
|
Example("if foo == .bar() {}"),
|
|
Example("guard foo == .bar() else { return }"),
|
|
Example("""
|
|
if case .appStore = self.appInstaller, !UIDevice.isSimulator() {
|
|
viewController.present(self, animated: false)
|
|
} else {
|
|
UIApplication.shared.open(self.appInstaller.url)
|
|
}
|
|
""")
|
|
],
|
|
triggeringExamples: [
|
|
wrapInSwitch("case .bar↓(_)"),
|
|
wrapInSwitch("case .bar↓()"),
|
|
wrapInSwitch("case .bar↓(_), .bar2↓(_)"),
|
|
wrapInSwitch("case .bar↓() where method() > 2"),
|
|
wrapInFunc("case .bar↓(_)"),
|
|
Example("if case .bar↓(_) = foo {\n}"),
|
|
Example("guard case .bar↓(_) = foo else {\n}"),
|
|
Example("if case .bar↓() = foo {\n}"),
|
|
Example("guard case .bar↓() = foo else {\n}"),
|
|
Example("""
|
|
if case .appStore↓(_) = self.appInstaller, !UIDevice.isSimulator() {
|
|
viewController.present(self, animated: false)
|
|
} else {
|
|
UIApplication.shared.open(self.appInstaller.url)
|
|
}
|
|
""")
|
|
],
|
|
corrections: [
|
|
wrapInSwitch("case .bar↓(_)"): wrapInSwitch("case .bar"),
|
|
wrapInSwitch("case .bar↓()"): wrapInSwitch("case .bar"),
|
|
wrapInSwitch("case .bar↓(_), .bar2↓(_)"): wrapInSwitch("case .bar, .bar2"),
|
|
wrapInSwitch("case .bar↓() where method() > 2"): wrapInSwitch("case .bar where method() > 2"),
|
|
wrapInFunc("case .bar↓(_)"): wrapInFunc("case .bar"),
|
|
Example("if case .bar↓(_) = foo {"): Example("if case .bar = foo {"),
|
|
Example("guard case .bar↓(_) = foo else {"): Example("guard case .bar = foo else {")
|
|
]
|
|
)
|
|
|
|
public func validate(file: SwiftLintFile, kind: StatementKind,
|
|
dictionary: SourceKittenDictionary) -> [StyleViolation] {
|
|
return violationRanges(in: file, kind: kind, dictionary: dictionary).map {
|
|
StyleViolation(ruleDescription: Self.description,
|
|
severity: configuration.severity,
|
|
location: Location(file: file, characterOffset: $0.location))
|
|
}
|
|
}
|
|
|
|
public func substitution(for violationRange: NSRange, in file: SwiftLintFile) -> (NSRange, String)? {
|
|
return (violationRange, "")
|
|
}
|
|
|
|
public func violationRanges(in file: SwiftLintFile, kind: StatementKind,
|
|
dictionary: SourceKittenDictionary) -> [NSRange] {
|
|
guard kind == .case || kind == .if || kind == .guard else {
|
|
return []
|
|
}
|
|
|
|
let needsCase = kind == .if || kind == .guard
|
|
let contents = file.stringView
|
|
let callsRanges = dictionary.methodCallRanges(in: file)
|
|
|
|
return dictionary.elements.flatMap { subDictionary -> [NSRange] in
|
|
guard (subDictionary.kind == "source.lang.swift.structure.elem.pattern" ||
|
|
subDictionary.kind == "source.lang.swift.structure.elem.condition_expr"),
|
|
let byteRange = subDictionary.byteRange,
|
|
let caseRange = contents.byteRangeToNSRange(byteRange)
|
|
else {
|
|
return []
|
|
}
|
|
|
|
let emptyArgumentRegex = regex(#"\.\S+\s*(\([,\s_]*\))"#)
|
|
return emptyArgumentRegex.matches(in: file.contents, options: [], range: caseRange).compactMap { match in
|
|
let parenthesesRange = match.range(at: 1)
|
|
|
|
// avoid matches after `where` keyworkd
|
|
if let whereRange = file.match(pattern: "where", with: [.keyword], range: caseRange).first {
|
|
if whereRange.location < parenthesesRange.location {
|
|
return nil
|
|
}
|
|
|
|
// avoid matches in "(_, _) where"
|
|
if let whereByteRange = contents.NSRangeToByteRange(start: whereRange.location,
|
|
length: whereRange.length),
|
|
case let length = whereByteRange.location - byteRange.location,
|
|
case let byteRange = ByteRange(location: byteRange.location, length: length),
|
|
Set(file.syntaxMap.kinds(inByteRange: byteRange)) == [.keyword] {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
if needsCase, file.match(pattern: "\\bcase\\b", with: [.keyword], range: caseRange).isEmpty {
|
|
return nil
|
|
}
|
|
|
|
if callsRanges.contains(where: parenthesesRange.intersects) {
|
|
return nil
|
|
}
|
|
|
|
return parenthesesRange
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension SourceKittenDictionary {
|
|
func methodCallRanges(in file: SwiftLintFile) -> [NSRange] {
|
|
return substructure
|
|
.flatMap { dict -> [SourceKittenDictionary] in
|
|
// In Swift >= 5.6, calls are embedded in a `source.lang.swift.expr.argument` entry
|
|
guard SwiftVersion.current >= .fiveDotSix, dict.expressionKind == .argument else {
|
|
return [dict]
|
|
}
|
|
|
|
return dict.substructure
|
|
}
|
|
.compactMap { dict -> NSRange? in
|
|
guard dict.expressionKind == .call,
|
|
let byteRange = dict.byteRange,
|
|
let range = file.stringView.byteRangeToNSRange(byteRange),
|
|
let name = dict.name,
|
|
!name.starts(with: ".")
|
|
else {
|
|
return nil
|
|
}
|
|
|
|
return range
|
|
}
|
|
}
|
|
}
|