254 lines
8.1 KiB
Swift
254 lines
8.1 KiB
Swift
#if canImport(Darwin)
|
|
import Darwin
|
|
#endif
|
|
import Foundation
|
|
import IDEUtils
|
|
import SourceKittenFramework
|
|
import SwiftOperators
|
|
import SwiftParser
|
|
import SwiftParserDiagnostics
|
|
import SwiftSyntax
|
|
|
|
private typealias FileCacheKey = UUID
|
|
private let responseCache = Cache { file -> [String: SourceKitRepresentable]? in
|
|
do {
|
|
return try Request.editorOpen(file: file.file).sendIfNotDisabled()
|
|
} catch let error as Request.Error {
|
|
queuedPrintError(error.description)
|
|
return nil
|
|
} catch {
|
|
return nil
|
|
}
|
|
}
|
|
private let structureDictionaryCache = Cache { file in
|
|
return responseCache.get(file).map(Structure.init).map { SourceKittenDictionary($0.dictionary) }
|
|
}
|
|
private let syntaxTreeCache = Cache { file -> SourceFileSyntax in
|
|
return Parser.parse(source: file.contents)
|
|
}
|
|
private let foldedSyntaxTreeCache = Cache { file -> SourceFileSyntax? in
|
|
return OperatorTable.standardOperators
|
|
.foldAll(file.syntaxTree) { _ in }
|
|
.as(SourceFileSyntax.self)
|
|
}
|
|
private let locationConverterCache = Cache { file -> SourceLocationConverter in
|
|
return SourceLocationConverter(file: file.path ?? "<nopath>", tree: file.syntaxTree)
|
|
}
|
|
private let commandsCache = Cache { file -> [Command] in
|
|
guard file.contents.contains("swiftlint:") else {
|
|
return []
|
|
}
|
|
return CommandVisitor(locationConverter: file.locationConverter)
|
|
.walk(file: file, handler: \.commands)
|
|
}
|
|
private let syntaxMapCache = Cache { file in
|
|
responseCache.get(file).map { SwiftLintSyntaxMap(value: SyntaxMap(sourceKitResponse: $0)) }
|
|
}
|
|
private let syntaxClassificationsCache = Cache { $0.syntaxTree.classifications }
|
|
private let syntaxKindsByLinesCache = Cache { $0.syntaxKindsByLine() }
|
|
private let syntaxTokensByLinesCache = Cache { $0.syntaxTokensByLine() }
|
|
private let linesWithTokensCache = Cache { $0.computeLinesWithTokens() }
|
|
|
|
internal typealias AssertHandler = () -> Void
|
|
// Re-enable once all parser diagnostics in tests have been addressed.
|
|
// https://github.com/realm/SwiftLint/issues/3348
|
|
@_spi(TestHelper)
|
|
public var parserDiagnosticsDisabledForTests = false
|
|
|
|
private let assertHandlers = [FileCacheKey: AssertHandler]()
|
|
private let assertHandlerCache = Cache { file in assertHandlers[file.cacheKey] }
|
|
|
|
private class Cache<T> {
|
|
private var values = [FileCacheKey: T]()
|
|
private let factory: (SwiftLintFile) -> T
|
|
private let lock = PlatformLock()
|
|
|
|
fileprivate init(_ factory: @escaping (SwiftLintFile) -> T) {
|
|
self.factory = factory
|
|
}
|
|
|
|
fileprivate func get(_ file: SwiftLintFile) -> T {
|
|
let key = file.cacheKey
|
|
return lock.doLocked {
|
|
if let cachedValue = values[key] {
|
|
return cachedValue
|
|
}
|
|
let value = factory(file)
|
|
values[key] = value
|
|
return value
|
|
}
|
|
}
|
|
|
|
fileprivate func invalidate(_ file: SwiftLintFile) {
|
|
lock.doLocked { values.removeValue(forKey: file.cacheKey) }
|
|
}
|
|
|
|
fileprivate func clear() {
|
|
lock.doLocked { values.removeAll(keepingCapacity: false) }
|
|
}
|
|
|
|
fileprivate func set(key: FileCacheKey, value: T) {
|
|
lock.doLocked { values[key] = value }
|
|
}
|
|
|
|
fileprivate func unset(key: FileCacheKey) {
|
|
lock.doLocked { values.removeValue(forKey: key) }
|
|
}
|
|
}
|
|
|
|
extension SwiftLintFile {
|
|
fileprivate var cacheKey: FileCacheKey {
|
|
return id
|
|
}
|
|
|
|
internal var sourcekitdFailed: Bool {
|
|
get {
|
|
return responseCache.get(self) == nil
|
|
}
|
|
set {
|
|
if newValue {
|
|
responseCache.set(key: cacheKey, value: nil)
|
|
} else {
|
|
responseCache.unset(key: cacheKey)
|
|
}
|
|
}
|
|
}
|
|
|
|
internal var assertHandler: AssertHandler? {
|
|
get {
|
|
return assertHandlerCache.get(self)
|
|
}
|
|
set {
|
|
assertHandlerCache.set(key: cacheKey, value: newValue)
|
|
}
|
|
}
|
|
|
|
internal var parserDiagnostics: [String]? {
|
|
if parserDiagnosticsDisabledForTests {
|
|
return nil
|
|
}
|
|
|
|
return ParseDiagnosticsGenerator.diagnostics(for: syntaxTree)
|
|
.filter { $0.diagMessage.severity == .error }
|
|
.map(\.message)
|
|
}
|
|
|
|
internal var linesWithTokens: Set<Int> { linesWithTokensCache.get(self) }
|
|
|
|
internal var structureDictionary: SourceKittenDictionary {
|
|
guard let structureDictionary = structureDictionaryCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return SourceKittenDictionary([:])
|
|
}
|
|
queuedFatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return structureDictionary
|
|
}
|
|
|
|
internal var syntaxClassifications: SyntaxClassifications { syntaxClassificationsCache.get(self) }
|
|
|
|
internal var syntaxMap: SwiftLintSyntaxMap {
|
|
guard let syntaxMap = syntaxMapCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return SwiftLintSyntaxMap(value: SyntaxMap(data: []))
|
|
}
|
|
queuedFatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return syntaxMap
|
|
}
|
|
|
|
internal var syntaxTree: SourceFileSyntax { syntaxTreeCache.get(self) }
|
|
|
|
internal var foldedSyntaxTree: SourceFileSyntax? { foldedSyntaxTreeCache.get(self) }
|
|
|
|
internal var locationConverter: SourceLocationConverter { locationConverterCache.get(self) }
|
|
|
|
internal var commands: [Command] { commandsCache.get(self).filter { $0.isValid } }
|
|
internal var invalidCommands: [Command] { commandsCache.get(self).filter { !$0.isValid } }
|
|
|
|
internal var syntaxTokensByLines: [[SwiftLintSyntaxToken]] {
|
|
guard let syntaxTokensByLines = syntaxTokensByLinesCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return []
|
|
}
|
|
queuedFatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return syntaxTokensByLines
|
|
}
|
|
|
|
internal var syntaxKindsByLines: [[SourceKittenFramework.SyntaxKind]] {
|
|
guard let syntaxKindsByLines = syntaxKindsByLinesCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return []
|
|
}
|
|
queuedFatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return syntaxKindsByLines
|
|
}
|
|
|
|
/// Invalidates all cached data for this file.
|
|
public func invalidateCache() {
|
|
file.clearCaches()
|
|
responseCache.invalidate(self)
|
|
assertHandlerCache.invalidate(self)
|
|
structureDictionaryCache.invalidate(self)
|
|
syntaxClassificationsCache.invalidate(self)
|
|
syntaxMapCache.invalidate(self)
|
|
syntaxTokensByLinesCache.invalidate(self)
|
|
syntaxKindsByLinesCache.invalidate(self)
|
|
syntaxTreeCache.invalidate(self)
|
|
foldedSyntaxTreeCache.invalidate(self)
|
|
locationConverterCache.invalidate(self)
|
|
commandsCache.invalidate(self)
|
|
linesWithTokensCache.invalidate(self)
|
|
}
|
|
|
|
@_spi(TestHelper)
|
|
public static func clearCaches() {
|
|
responseCache.clear()
|
|
assertHandlerCache.clear()
|
|
structureDictionaryCache.clear()
|
|
syntaxClassificationsCache.clear()
|
|
syntaxMapCache.clear()
|
|
syntaxTokensByLinesCache.clear()
|
|
syntaxKindsByLinesCache.clear()
|
|
syntaxTreeCache.clear()
|
|
foldedSyntaxTreeCache.clear()
|
|
locationConverterCache.clear()
|
|
commandsCache.clear()
|
|
linesWithTokensCache.clear()
|
|
}
|
|
}
|
|
|
|
private final class PlatformLock {
|
|
#if canImport(Darwin)
|
|
private let primitiveLock: UnsafeMutablePointer<os_unfair_lock>
|
|
#else
|
|
private let primitiveLock = NSLock()
|
|
#endif
|
|
|
|
init() {
|
|
#if canImport(Darwin)
|
|
primitiveLock = UnsafeMutablePointer<os_unfair_lock>.allocate(capacity: 1)
|
|
primitiveLock.initialize(to: os_unfair_lock())
|
|
#endif
|
|
}
|
|
|
|
@discardableResult
|
|
func doLocked<U>(_ closure: () -> U) -> U {
|
|
#if canImport(Darwin)
|
|
os_unfair_lock_lock(primitiveLock)
|
|
defer { os_unfair_lock_unlock(primitiveLock) }
|
|
return closure()
|
|
#else
|
|
primitiveLock.lock()
|
|
defer { primitiveLock.unlock() }
|
|
return closure()
|
|
#endif
|
|
}
|
|
}
|