mirror of
https://github.com/apple/sourcekit-lsp.git
synced 2026-06-24 12:21:58 +02:00
64aa354caf
When the client opts in to `workspace/tests/refresh` or `workspace/playgrounds/refresh` via experimental client capabilities, SourceKit-LSP now maintains a proactive cache of the current test and playground lists and sends the corresponding `workspace/.../refresh` notification whenever the cache changes. `workspaceTests()` / `workspacePlaygrounds()` then serve subsequent fetch requests directly from the cache. Add `EntryPointManager`: runs background scans, stores the results, fires callbacks on changes: - Start scanning when build targets are updated including initial updates, any watched files are changed, and index is updated. - Send '/refresh' server initiated requests when the cache has changed. - Coalesces rapid invalidations by cancelling any in-flight refresh task. Also: - Simplify `SourceKitIndexDelegate` from an `actor` with `AtomicInt32` to a plain `class`, since it is only called from IndexStoreDB's internal serial dispatch queue.
59 lines
1.9 KiB
Swift
59 lines
1.9 KiB
Swift
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2020 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
import Dispatch
|
|
import IndexStoreDB
|
|
@_spi(SourceKitLSP) import LanguageServerProtocolExtensions
|
|
@_spi(SourceKitLSP) import SKLogging
|
|
import SwiftExtensions
|
|
@_spi(SourceKitLSP) import ToolsProtocolsSwiftExtensions
|
|
|
|
/// `IndexDelegate` for the SourceKit workspace.
|
|
class SourceKitIndexDelegate: IndexDelegate {
|
|
let callback: @Sendable () async -> Void
|
|
|
|
/// The count of pending unit events. Whenever this transitions to 0, it represents a time where
|
|
/// the index finished processing known events. Of course, that may have already changed by the
|
|
/// time we are notified.
|
|
var pendingUnitCount = 0
|
|
|
|
package init(callback: @escaping @Sendable () async -> Void) {
|
|
self.callback = callback
|
|
}
|
|
|
|
package func processingAddedPending(_ count: Int) {
|
|
pendingUnitCount += count
|
|
}
|
|
|
|
package func processingCompleted(_ count: Int) {
|
|
pendingUnitCount -= count
|
|
if pendingUnitCount == 0 {
|
|
indexChanged()
|
|
}
|
|
|
|
if pendingUnitCount < 0 {
|
|
// Technically this is not data race safe because `pendingUnitCount` might change between the check and us setting
|
|
// it to 0. But then, this should never happen anyway, so it's fine.
|
|
logger.fault("pendingUnitCount dropped below zero: \(self.pendingUnitCount)")
|
|
pendingUnitCount = 0
|
|
indexChanged()
|
|
}
|
|
}
|
|
|
|
private func indexChanged() {
|
|
Task { [callback] in
|
|
logger.debug("IndexStoreDB changed")
|
|
await callback()
|
|
}
|
|
}
|
|
}
|