Files
SwiftLint/Source/SwiftLintFramework/Rules/PrivateOverFilePrivateRule.swift
T
JP Simard b83e0991b9 Remove all file headers
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.
2018-05-04 13:42:02 -07:00

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))
}
}
}