forked from realm/SwiftLint
-
Notifications
You must be signed in to change notification settings - Fork 0
/
EmptyEnumArgumentsRule.swift
141 lines (121 loc) · 5.63 KB
/
EmptyEnumArgumentsRule.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
//
// EmptyEnumArgumentsRule.swift
// SwiftLint
//
// Created by Marcelo Fabri on 05/01/17.
// Copyright © 2017 Realm. All rights reserved.
//
import Foundation
import SourceKittenFramework
public struct EmptyEnumArgumentsRule: ASTRule, ConfigurationProviderRule, CorrectableRule {
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 types if they are not used.",
kind: .style,
nonTriggeringExamples: [
"switch foo {\n case .bar: break\n}",
"switch foo {\n case .bar(let x): break\n}",
"switch foo {\n case let .bar(x): break\n}",
"switch (foo, bar) {\n case (_, _): break\n}",
"switch foo {\n case \"bar\".uppercased(): break\n}"
],
triggeringExamples: [
"switch foo {\n case .bar↓(_): break\n}",
"switch foo {\n case .bar↓(): break\n}",
"switch foo {\n case .bar↓(_), .bar2↓(_): break\n}",
"switch foo {\n case .bar↓() where method() > 2: break\n}"
],
corrections: [
"switch foo {\n case .bar↓(_): break\n}":
"switch foo {\n case .bar: break\n}",
"switch foo {\n case .bar↓(): break\n}":
"switch foo {\n case .bar: break\n}",
"switch foo {\n case .bar↓(_), .bar2↓(_): break\n}":
"switch foo {\n case .bar, .bar2: break\n}",
"switch foo {\n case .bar↓() where method() > 2: break\n}":
"switch foo {\n case .bar where method() > 2: break\n}"
]
)
public func validate(file: File, kind: StatementKind,
dictionary: [String: SourceKitRepresentable]) -> [StyleViolation] {
return violationRanges(in: file, kind: kind, dictionary: dictionary).map {
StyleViolation(ruleDescription: type(of: self).description,
severity: configuration.severity,
location: Location(file: file, characterOffset: $0.location))
}
}
private func violationRanges(in file: File, kind: StatementKind,
dictionary: [String: SourceKitRepresentable]) -> [NSRange] {
guard kind == .case else {
return []
}
let contents = file.contents.bridge()
let callsRanges = dictionary.substructure.flatMap { dict -> NSRange? in
guard dict.kind.flatMap(SwiftExpressionKind.init) == .call,
let offset = dict.offset,
let length = dict.length,
let range = contents.byteRangeToNSRange(start: offset, length: length) else {
return nil
}
return range
}
return dictionary.elements.flatMap { subDictionary -> [NSRange] in
guard subDictionary.kind == "source.lang.swift.structure.elem.pattern",
let offset = subDictionary.offset,
let length = subDictionary.length,
let caseRange = contents.byteRangeToNSRange(start: offset, length: length) else {
return []
}
return file.match(pattern: "\\([,\\s_]*\\)", range: caseRange).flatMap { range, kinds in
guard Set(kinds).isSubset(of: [.keyword]),
case let byteRange = NSRange(location: offset, length: length),
Set(file.syntaxMap.kinds(inByteRange: byteRange)) != [.keyword] else {
return nil
}
// avoid matches after `where` keyworkd
if let whereMatch = file.match(pattern: "where", with: [.keyword], range: caseRange).first,
whereMatch.location < range.location {
return nil
}
if callsRanges.first(where: range.intersects) != nil {
return nil
}
return range
}
}
}
private func violationRanges(in file: File, dictionary: [String: SourceKitRepresentable]) -> [NSRange] {
return dictionary.substructure.flatMap { subDict -> [NSRange] in
guard let kindString = subDict.kind,
let kind = StatementKind(rawValue: kindString) else {
return []
}
return violationRanges(in: file, dictionary: subDict) +
violationRanges(in: file, kind: kind, dictionary: subDict)
}
}
private func violationRanges(in file: File) -> [NSRange] {
return violationRanges(in: file, dictionary: file.structure.dictionary).sorted { lhs, rhs in
lhs.location < rhs.location
}
}
public func correct(file: File) -> [Correction] {
let violatingRanges = file.ruleEnabled(violatingRanges: violationRanges(in: file), for: self)
var correctedContents = file.contents
var adjustedLocations = [Int]()
for violatingRange in violatingRanges.reversed() {
if let indexRange = correctedContents.nsrangeToIndexRange(violatingRange) {
correctedContents = correctedContents.replacingCharacters(in: indexRange, with: "")
adjustedLocations.insert(violatingRange.location, at: 0)
}
}
file.write(correctedContents)
return adjustedLocations.map {
Correction(ruleDescription: type(of: self).description,
location: Location(file: file, characterOffset: $0))
}
}
}