mirror of
https://github.com/realm/SwiftLint.git
synced 2026-06-06 20:18:40 +00:00
b83e0991b9
The MIT license doesn't require that all files be prepended with this licensing or copyright information. Realm confirmed that they're ok with this change. This will enable some companies to contribute to SwiftLint and the date & authorship information will remain accessible via git source control.
91 lines
3.7 KiB
Swift
91 lines
3.7 KiB
Swift
import Foundation
|
|
import SourceKittenFramework
|
|
|
|
public struct PrivateOverFilePrivateRule: Rule, ConfigurationProviderRule, CorrectableRule {
|
|
public var configuration = PrivateOverFilePrivateRuleConfiguration()
|
|
|
|
public init() {}
|
|
|
|
public static let description = RuleDescription(
|
|
identifier: "private_over_fileprivate",
|
|
name: "Private over fileprivate",
|
|
description: "Prefer `private` over `fileprivate` declarations.",
|
|
kind: .idiomatic,
|
|
nonTriggeringExamples: [
|
|
"extension String {}",
|
|
"private extension String {}",
|
|
"public \n enum MyEnum {}",
|
|
"open extension \n String {}",
|
|
"internal extension String {}",
|
|
"extension String {\nfileprivate func Something(){}\n}",
|
|
"class MyClass {\nfileprivate let myInt = 4\n}",
|
|
"class MyClass {\nfileprivate(set) var myInt = 4\n}",
|
|
"struct Outter {\nstruct Inter {\nfileprivate struct Inner {}\n}\n}"
|
|
],
|
|
triggeringExamples: [
|
|
"↓fileprivate enum MyEnum {}",
|
|
"↓fileprivate class MyClass {\nfileprivate(set) var myInt = 4\n}"
|
|
],
|
|
corrections: [
|
|
"↓fileprivate enum MyEnum {}": "private enum MyEnum {}",
|
|
"↓fileprivate class MyClass {\nfileprivate(set) var myInt = 4\n}":
|
|
"private class MyClass {\nfileprivate(set) var myInt = 4\n}"
|
|
]
|
|
)
|
|
|
|
public func validate(file: File) -> [StyleViolation] {
|
|
return violationRanges(in: file).map {
|
|
StyleViolation(ruleDescription: type(of: self).description,
|
|
severity: configuration.severityConfiguration.severity,
|
|
location: Location(file: file, characterOffset: $0.location))
|
|
}
|
|
}
|
|
|
|
private func violationRanges(in file: File) -> [NSRange] {
|
|
let syntaxTokens = file.syntaxMap.tokens
|
|
let contents = file.contents.bridge()
|
|
|
|
return file.structure.dictionary.substructure.compactMap { dictionary -> NSRange? in
|
|
guard let offset = dictionary.offset else {
|
|
return nil
|
|
}
|
|
|
|
if !configuration.validateExtensions &&
|
|
dictionary.kind.flatMap(SwiftDeclarationKind.init) == .extension {
|
|
return nil
|
|
}
|
|
|
|
let parts = syntaxTokens.prefix { offset > $0.offset }
|
|
guard let lastKind = parts.last,
|
|
SyntaxKind(rawValue: lastKind.type) == .attributeBuiltin,
|
|
let aclName = contents.substringWithByteRange(start: lastKind.offset, length: lastKind.length),
|
|
AccessControlLevel(description: aclName) == .fileprivate,
|
|
let range = contents.byteRangeToNSRange(start: lastKind.offset, length: lastKind.length) else {
|
|
return nil
|
|
}
|
|
|
|
return range
|
|
}
|
|
}
|
|
|
|
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: "private")
|
|
adjustedLocations.insert(violatingRange.location, at: 0)
|
|
}
|
|
}
|
|
|
|
file.write(correctedContents)
|
|
|
|
return adjustedLocations.map {
|
|
Correction(ruleDescription: type(of: self).description,
|
|
location: Location(file: file, characterOffset: $0))
|
|
}
|
|
}
|
|
}
|