-
Notifications
You must be signed in to change notification settings - Fork 339
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
improvement: deduplicate compile requests #7006
base: main
Are you sure you want to change the base?
Conversation
event.eventType match { | ||
case EventType.CreateOrModify => | ||
try { | ||
val md5 = MD5.bytesToHex(FileIO.readAllBytes(path)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Later in onChange
the file is read again (if the event isn't ignored). Should I try to deduplicate these reads?
event.eventType match { | ||
case EventType.CreateOrModify => | ||
try { | ||
val md5 = MD5.bytesToHex(FileIO.readAllBytes(path)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe we should do it before compile? So that we wouldn't do noop compilations as well?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I put it before compile. I also added a draft of a solution to deduplicate reading in file content. (I feel like that might be costly.) WDYT?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I realised we also compile the file when it did not change but its dependencies did (on didFocus
). I wonder if it's safe to always discard compile on didSave
if the file did not change. Can we rely only on didFocus
? Otherwise, there may be some compilation that finished and user may want to force compile (on this build target) using save. Maybe the first option to only deduplicate on file watcher events was better?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I realised we also compile the file when it did not change but its dependencies did (on didFocus). I wonder if it's safe to always discard compile on didSave if the file did not change.
But if they want to force a save with didSave, they need to change the file, right? Did save will not be sent on a unchanged file anyway.
If any upstream projects change, we should always maybe mark anything that depends on them as "dirty" and should always be recompiled next time a compilation is invoked?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
But if they want to force a save with didSave, they need to change the file, right? Did save will not be sent on a unchanged file anyway
At least in VSCode Save
will always send didSave
even if the file did not change.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't see didSave being sent if I save an unchanged file 🤔 Or do you mean in a different case?
focusedDocumentBuildTarget: Option[BuildTargetIdentifier], | ||
): Future[Unit] = { | ||
val paths = |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe we could just find first that changed in a given target?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, though I doubt it ever too many files at once. Only file watcher passes more than one file.
import scala.meta.io.AbsolutePath | ||
|
||
class SavedFileSignatures { | ||
private val previousCreateOrModify = TrieMap[AbsolutePath, String]() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could we reuse MutableMd5Fingerprints somehow? Maybe just save them in didSave with a flag to MutableMd5Fingerprints and then we can get the last saved one instead of calculating again?
We can still have a separate collection in Compilations so that it's possible to clear it and force new compilations?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For didSave
we put them anyway there but for file watcher event like CreateOrModify
we don't, and we don't want to since it stores content, which we don't need. Also there is no way to check that the caller of compileFile/s
saved the fingerprint (there may be an old one there), so it becomes some hidden knowledge that you should.
Edit: we can potentially use flags and mark "used ones"
metals/src/main/scala/scala/meta/internal/metals/MetalsLspService.scala
Outdated
Show resolved
Hide resolved
metals/src/main/scala/scala/meta/internal/metals/MetalsLspService.scala
Outdated
Show resolved
Hide resolved
.getOrElse(current) | ||
} | ||
focusedDocumentBuildTarget.set( | ||
buildTargets.inverseSources(path).getOrElse(null) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we pack it into an abtraction that always returns Option?
toCompile <- paths.foldLeft( | ||
Future.successful(Seq.empty[BuildTargetIdentifier]) | ||
) { case (toCompile, (path, fingerprint)) => | ||
toCompile.flatMap { acc => |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why not just map it and do Future.sequence
on it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To do it sequentially, so we deduplicate if there is the same build target.
} yield () | ||
} | ||
|
||
def cascadeCompile(targets: Seq[BuildTargetIdentifier]): Future[Unit] = { | ||
val inverseDependencyLeaves = | ||
targets.flatMap(buildTargets.inverseDependencyLeaves).distinct | ||
fileChanges.willCompile(inverseDependencyLeaves.toList) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Should we not remove these only once it compiled?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think so. If we remove after it is compiled we can fail to notice some changes. Let's say we do:
- compile
- bloop compiles
- some changes are made, before metals processes finish compilation
- we remove the build target from dirty
The other way around, we can't miss anything. The compilation for that build target was added in the queue anyway and will happen at a point in time.
private def findBuildTargetIfShouldCompile( | ||
path: AbsolutePath, | ||
fingerprint: Option[Fingerprint], | ||
assumeDidNotChange: Boolean = false, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could we add some docs on why we need the boolean?
fixes: #6896
Problem
Code generation leads to infinite compilation loop.
Solution
Ignore create or modify file watcher event when content of the file is the same as with the previously when this event was triggered.Ignore compile for file if it did not change from last compile.