Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fill package inclusion cache in background #3816

Merged
merged 5 commits into from
Dec 21, 2024
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

### Added
* Add sections to breadcrumbs
* Improve performance when starting a run configuration and when using autocompletion directly after starting the IDE
* Change order in structure view to match source file and sectioning level
* Add command redefinitions to command definition filter in structure view
* Add support for automatic language injection on the minted environment
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
package nl.hannahsten.texifyidea.index.file

import com.intellij.openapi.application.ApplicationManager

Check warning on line 3 in src/nl/hannahsten/texifyidea/index/file/LatexExternalPackageInclusionCache.kt

View workflow job for this annotation

GitHub Actions / Qodana Community for JVM

Unused import directive

Unused import directive
Fixed Show fixed Hide fixed
import com.intellij.openapi.application.runInEdt
import com.intellij.openapi.application.runReadAction
import com.intellij.openapi.project.DumbService
import com.intellij.openapi.project.Project
import com.intellij.psi.search.GlobalSearchScope
Expand All @@ -8,6 +11,8 @@
import nl.hannahsten.texifyidea.algorithm.DFS
import nl.hannahsten.texifyidea.lang.LatexPackage
import nl.hannahsten.texifyidea.util.files.removeFileExtension
import nl.hannahsten.texifyidea.util.runInBackground
import java.util.concurrent.atomic.AtomicBoolean

/**
* Compute and cache for each package style file all the other style files it includes (directly or indirectly).
Expand All @@ -18,11 +23,13 @@

private val cache = concurrentMapOf<LatexPackage, Set<LatexPackage>>()

private var isFillingCache = AtomicBoolean(false)

/**
* Map every LaTeX package style file to all the style files it includes, directly or indirectly.
*/
@Synchronized
fun getAllPackageInclusions(project: Project): Map<LatexPackage, Set<LatexPackage>> {
fun updateOrGetCache(project: Project): Map<LatexPackage, Set<LatexPackage>> {
if (cache.isNotEmpty() || DumbService.isDumb(project)) return cache

// Make sure the index is ready (#3754)
Expand All @@ -31,20 +38,34 @@
val directChildren = mutableMapOf<LatexPackage, MutableSet<LatexPackage>>()

// Get direct children from the index
FileBasedIndex.getInstance().getAllKeys(LatexExternalPackageInclusionIndex.Cache.id, project).forEach { indexKey ->
FileBasedIndex.getInstance().processValues(
LatexExternalPackageInclusionIndex.Cache.id, indexKey, null, { file, _ ->
val key = LatexPackage(file.name.removeFileExtension())
directChildren[key] = directChildren.getOrDefault(key, mutableSetOf()).also { it.add(LatexPackage((indexKey))) }
true
},
GlobalSearchScope.everythingScope(project)
)
}
if (isFillingCache.getAndSet(true)) return cache
// ???
runInEdt {
runInBackground(project, "Retrieving LaTeX package inclusions...") { indicator ->
try {
runReadAction { FileBasedIndex.getInstance().getAllKeys(LatexExternalPackageInclusionIndex.Cache.id, project) }.forEach { indexKey ->
runReadAction {
FileBasedIndex.getInstance().processValues(
LatexExternalPackageInclusionIndex.Cache.id, indexKey, null, { file, _ ->
indicator.checkCanceled()
val key = LatexPackage(file.name.removeFileExtension())
directChildren[key] = directChildren.getOrDefault(key, mutableSetOf()).also { it.add(LatexPackage((indexKey))) }
true
},
GlobalSearchScope.everythingScope(project)
)
}
}

// Do some DFS for indirect inclusions
for (latexPackage in directChildren.keys) {
cache[latexPackage] = DFS(latexPackage) { parent -> directChildren[parent] ?: emptySet() }.execute()
// Do some DFS for indirect inclusions
for (latexPackage in directChildren.keys) {
cache[latexPackage] = DFS(latexPackage) { parent -> directChildren[parent] ?: emptySet() }.execute()
}
}
finally {
isFillingCache.set(false)
}
}
}

return cache
Expand All @@ -55,7 +76,7 @@
*/
fun getAllIndirectlyIncludedPackages(packages: Collection<LatexPackage>, project: Project): Set<LatexPackage> {
val result = packages.toMutableSet()
val allInclusions = getAllPackageInclusions(project)
val allInclusions = updateOrGetCache(project)
for (latexPackage in packages) {
result.addAll(allInclusions[latexPackage] ?: emptySet())
}
Expand Down
4 changes: 2 additions & 2 deletions src/nl/hannahsten/texifyidea/run/compiler/LatexCompiler.kt
Original file line number Diff line number Diff line change
Expand Up @@ -287,9 +287,9 @@ enum class LatexCompiler(private val displayName: String, val executableName: St
*/
fun getCommand(runConfig: LatexRunConfiguration, project: Project): List<String>? {
val rootManager = ProjectRootManager.getInstance(project)
val fileIndex = rootManager.fileIndex
val mainFile = runConfig.mainFile ?: return null
val moduleRoot = fileIndex.getContentRootForFile(mainFile)
// Getting the content root is an expensive operation (See WorkspaceFileIndexDataImpl#ensureIsUpToDate), and since it probably won't change often we reuse a cached value
val moduleRoot = runConfig.outputPath.contentRoot
// For now we disable module roots with Docker
// Could be improved by mounting them to the right directory
val moduleRoots = if (runConfig.getLatexDistributionType().isDocker()) {
Expand Down
6 changes: 6 additions & 0 deletions src/nl/hannahsten/texifyidea/util/General.kt
Original file line number Diff line number Diff line change
Expand Up @@ -88,3 +88,9 @@ fun runInBackground(project: Project?, description: String, function: (indicator
}
})
}

fun runInBackgroundWithoutProgress(function: () -> Unit) {
ApplicationManager.getApplication().invokeLater {
function()
}
}
Loading