mirror of
https://github.com/realm/SwiftLint.git
synced 2026-06-06 20:18:40 +00:00
91 lines
3.8 KiB
Swift
91 lines
3.8 KiB
Swift
//
|
|
// LineLengthRule.swift
|
|
// SwiftLint
|
|
//
|
|
// Created by JP Simard on 5/16/15.
|
|
// Copyright © 2015 Realm. All rights reserved.
|
|
//
|
|
|
|
import SourceKittenFramework
|
|
|
|
public struct LineLengthRule: ConfigurationProviderRule, SourceKitFreeRule {
|
|
public var configuration = SeverityLevelsConfiguration(warning: 120, error: 200)
|
|
|
|
public init() {}
|
|
|
|
public static let description = RuleDescription(
|
|
identifier: "line_length",
|
|
name: "Line Length",
|
|
description: "Lines should not span too many characters.",
|
|
nonTriggeringExamples: [
|
|
String(repeating: "/", count: 120) + "\n",
|
|
String(repeating: "#colorLiteral(red: 0.9607843161, green: 0.7058823705, blue: 0.200000003, alpha: 1)", count: 120) + "\n",
|
|
String(repeating: "#imageLiteral(resourceName: \"image.jpg\")", count: 120) + "\n"
|
|
],
|
|
triggeringExamples: [
|
|
String(repeating: "/", count: 121) + "\n",
|
|
String(repeating: "#colorLiteral(red: 0.9607843161, green: 0.7058823705, blue: 0.200000003, alpha: 1)", count: 121) + "\n",
|
|
String(repeating: "#imageLiteral(resourceName: \"image.jpg\")", count: 121) + "\n"
|
|
]
|
|
)
|
|
|
|
public func validateFile(_ file: File) -> [StyleViolation] {
|
|
let minValue = configuration.params.map({ $0.value }).min(by: <)
|
|
return file.lines.flatMap { line in
|
|
// `line.content.characters.count` <= `line.range.length` is true.
|
|
// So, `check line.range.length` is larger than minimum parameter value.
|
|
// for avoiding using heavy `line.content.characters.count`.
|
|
if line.range.length < minValue! {
|
|
return nil
|
|
}
|
|
|
|
var strippedString = line.content
|
|
strippedString = stripLiterals(fromSourceString: strippedString,
|
|
withDelimiter: "#colorLiteral")
|
|
strippedString = stripLiterals(fromSourceString: strippedString,
|
|
withDelimiter: "#imageLiteral")
|
|
|
|
let length = strippedString.characters.count
|
|
|
|
for param in configuration.params where length > param.value {
|
|
return StyleViolation(ruleDescription: type(of: self).description,
|
|
severity: param.severity,
|
|
location: Location(file: file.path, line: line.index),
|
|
reason: "Line should be \(configuration.warning) characters or less: " +
|
|
"currently \(length) characters")
|
|
}
|
|
return nil
|
|
}
|
|
}
|
|
|
|
/// Takes a string and replaces any literals specified by the `delimiter` parameter with `#`
|
|
///
|
|
/// - parameter sourceString: Original string, possibly containing literals
|
|
/// - parameter delimiter: Delimiter of the literal
|
|
/// (characters before the parentheses, e.g. `#colorLiteral`)
|
|
///
|
|
/// - returns: sourceString with the given literals replaced by `#`
|
|
private func stripLiterals(fromSourceString sourceString: String,
|
|
withDelimiter delimiter: String) -> String {
|
|
var modifiedString = sourceString
|
|
|
|
// While copy of content contains literal, replace with a single character
|
|
while modifiedString.contains("\(delimiter)(") {
|
|
if let rangeStart = modifiedString.range(of: "\(delimiter)("),
|
|
let rangeEnd = modifiedString.range(of: ")",
|
|
options: .literal,
|
|
range:
|
|
rangeStart.lowerBound..<modifiedString.endIndex) {
|
|
modifiedString.replaceSubrange(rangeStart.lowerBound..<rangeEnd.upperBound,
|
|
with: "#")
|
|
|
|
} else { // Should never be the case, but break to avoid accidental infinity loop
|
|
break
|
|
}
|
|
}
|
|
|
|
return modifiedString
|
|
}
|
|
|
|
}
|