mirror of
https://github.com/realm/SwiftLint.git
synced 2026-06-06 20:18:40 +00:00
191 lines
5.7 KiB
Swift
191 lines
5.7 KiB
Swift
//
|
|
// File+Cache.swift
|
|
// SwiftLint
|
|
//
|
|
// Created by Nikolaj Schumacher on 5/26/15.
|
|
// Copyright © 2015 Realm. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
import SourceKittenFramework
|
|
|
|
private var responseCache = Cache({file -> [String: SourceKitRepresentable]? in
|
|
do {
|
|
return try Request.editorOpen(file: file).failableSend()
|
|
} catch let error as Request.Error {
|
|
queuedPrintError(error.description)
|
|
return nil
|
|
} catch {
|
|
return nil
|
|
}
|
|
})
|
|
private var structureCache = Cache({ file -> Structure? in
|
|
if let structure = responseCache.get(file).map(Structure.init) {
|
|
queueForRebuild.append(structure)
|
|
return structure
|
|
}
|
|
return nil
|
|
})
|
|
private var syntaxMapCache = Cache({ file in responseCache.get(file).map(SyntaxMap.init) })
|
|
private var syntaxKindsByLinesCache = Cache({ file in file.syntaxKindsByLine() })
|
|
private var syntaxTokensByLinesCache = Cache({ file in file.syntaxTokensByLine() })
|
|
|
|
internal typealias AssertHandler = () -> Void
|
|
private var assertHandlers = [String: AssertHandler]()
|
|
|
|
private var _allDeclarationsByType = [String: [String]]()
|
|
private var queueForRebuild = [Structure]()
|
|
|
|
private struct Cache<T> {
|
|
|
|
fileprivate var values = [String: T]()
|
|
fileprivate var factory: (File) -> T
|
|
|
|
fileprivate init(_ factory: @escaping (File) -> T) {
|
|
self.factory = factory
|
|
}
|
|
|
|
fileprivate mutating func get(_ file: File) -> T {
|
|
let key = file.cacheKey
|
|
if let value = values[key] {
|
|
return value
|
|
}
|
|
let value = factory(file)
|
|
values[key] = value
|
|
return value
|
|
}
|
|
|
|
fileprivate mutating func invalidate(_ file: File) {
|
|
if let key = file.path {
|
|
values.removeValue(forKey: key)
|
|
}
|
|
}
|
|
|
|
fileprivate mutating func clear() {
|
|
values.removeAll(keepingCapacity: false)
|
|
}
|
|
}
|
|
|
|
extension File {
|
|
|
|
fileprivate var cacheKey: String {
|
|
return path ?? contents
|
|
}
|
|
|
|
internal var sourcekitdFailed: Bool {
|
|
get {
|
|
return responseCache.get(self) == nil
|
|
}
|
|
set {
|
|
if newValue {
|
|
let value: [String: SourceKitRepresentable]? = nil
|
|
responseCache.values[cacheKey] = value
|
|
} else {
|
|
responseCache.values.removeValue(forKey: cacheKey)
|
|
}
|
|
}
|
|
}
|
|
|
|
internal var assertHandler: AssertHandler? {
|
|
get {
|
|
return assertHandlers[cacheKey]
|
|
}
|
|
set {
|
|
assertHandlers[cacheKey] = newValue
|
|
}
|
|
}
|
|
|
|
internal var structure: Structure {
|
|
guard let structure = structureCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return Structure(sourceKitResponse: [:])
|
|
}
|
|
fatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return structure
|
|
}
|
|
|
|
internal var syntaxMap: SyntaxMap {
|
|
guard let syntaxMap = syntaxMapCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return SyntaxMap(data: [])
|
|
}
|
|
fatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return syntaxMap
|
|
}
|
|
|
|
internal var syntaxTokensByLines: [[SyntaxToken]] {
|
|
guard let syntaxTokensByLines = syntaxTokensByLinesCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return []
|
|
}
|
|
fatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return syntaxTokensByLines
|
|
}
|
|
|
|
internal var syntaxKindsByLines: [[SyntaxKind]] {
|
|
guard let syntaxKindsByLines = syntaxKindsByLinesCache.get(self) else {
|
|
if let handler = assertHandler {
|
|
handler()
|
|
return []
|
|
}
|
|
fatalError("Never call this for file that sourcekitd fails.")
|
|
}
|
|
return syntaxKindsByLines
|
|
}
|
|
|
|
public func invalidateCache() {
|
|
responseCache.invalidate(self)
|
|
assertHandlers.removeValue(forKey: cacheKey)
|
|
structureCache.invalidate(self)
|
|
syntaxMapCache.invalidate(self)
|
|
syntaxTokensByLinesCache.invalidate(self)
|
|
syntaxKindsByLinesCache.invalidate(self)
|
|
}
|
|
|
|
internal static func clearCaches() {
|
|
queueForRebuild = []
|
|
_allDeclarationsByType = [:]
|
|
responseCache.clear()
|
|
assertHandlers = [:]
|
|
structureCache.clear()
|
|
syntaxMapCache.clear()
|
|
syntaxTokensByLinesCache.clear()
|
|
syntaxKindsByLinesCache.clear()
|
|
}
|
|
|
|
internal static var allDeclarationsByType: [String: [String]] {
|
|
if !queueForRebuild.isEmpty {
|
|
rebuildAllDeclarationsByType()
|
|
}
|
|
return _allDeclarationsByType
|
|
}
|
|
}
|
|
|
|
private func substructureForDict(_ dict: [String: SourceKitRepresentable]) -> [[String: SourceKitRepresentable]]? {
|
|
return (dict["key.substructure"] as? [SourceKitRepresentable])?.flatMap {
|
|
$0 as? [String: SourceKitRepresentable]
|
|
}
|
|
}
|
|
|
|
private func rebuildAllDeclarationsByType() {
|
|
let allDeclarationsByType = queueForRebuild.flatMap { structure -> (String, [String])? in
|
|
guard let firstSubstructureDict = substructureForDict(structure.dictionary)?.first,
|
|
let name = firstSubstructureDict["key.name"] as? String,
|
|
let kind = (firstSubstructureDict["key.kind"] as? String)
|
|
.flatMap(SwiftDeclarationKind.init),
|
|
kind == .protocol,
|
|
let substructure = substructureForDict(firstSubstructureDict) else {
|
|
return nil
|
|
}
|
|
return (name, substructure.flatMap({ $0["key.name"] as? String }))
|
|
}
|
|
allDeclarationsByType.forEach { _allDeclarationsByType[$0.0] = $0.1 }
|
|
queueForRebuild = []
|
|
}
|