-
Notifications
You must be signed in to change notification settings - Fork 12.6k
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
Fix dropped watches #47482
Fix dropped watches #47482
Conversation
Just to comment on the quote below (and not the PR, which itself seems fine; someone else on the team may be better to review it than me):
FWIW it's unfortunately more complicated than that; Also of interest may be #43790 (and its linked issues), and that VS Code is actually removing all of its watchers in favor of |
Have you compared the time between the |
I should have been more clear...Re-adding the watch and processing files we are supposed to process comes with cost. We should not expect fixing this bug comes at zero cost. I think what you are more interested in is what is the overhead of resetting the watch. Setting a watch on every file in our mono repo (over 10k files) takes less than a second, so I don't think we are talking about anything that will have serious performance implications here. That said, this solution can potentially be optimized by stating the file and checking the inode rather than blindly resetting the watch on every rename event. That's a bit more of a complicated change and I doubt it's worth it given the low cost of setting a watch and that stating the file will introduce new cost 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.
This is incorrect change as this is switching the watcher to "Polling mode" watch instead of relying on file system events which is what it is supposed to do. Polling watch is always going to be accurate but it is CPU intensive and hence less desirable for directories and is only used when there is error..
Note that rename event happens whenever a file is added or removed to the folder
If you share your logs, i will be able to investigate the issue and propose the next steps
Eg when watching temp folder, adding file temp/something.js will invoke rename event
@sheetalkamat I'm confused by your comment. When I think of poll, I imagine periodically rechecking the state of the file system rather than waiting for a callback from the OS. Here, we are waiting on a callback from the OS, but we are resetting the watch when the callback fires because it is very likely to be invalid on Unix OS. When I use polling on our large monorepo, there is a noticeable overhead to it. I see no measurable overhead to resetting the watch when a file changes. |
I'm not sure what else I can provide here to help you debug. I detailed two very reliable repros on MacOS in the issue description and comments. The one in the comments involves only using VSCode and vim in a separate terminal. But happy to go over some logs over VC if that will move this forward. We are having to maintain this patch across TS upgrades which is a pain. |
Also, one final clarification - the rename events occur (at least on macOS) for all sorts of reasons, including when the file changes. It does not only fire for adding/removing a file from the directory. |
I'm relatively new to the watch code, but I wanted to write out my understanding of the original code and this PR in case it clears things up (so do correct me if I'm wrong). It seems like The original big long condition in The PR changes the condition to, on a rename, if it doesn't exist, do the same thing as above, but if it does exist, the watcher is recreated assuming that a rename event is likely to mean that the folder points somewhere else on disk via a different inode (as documented in the PR description), but is done so by calling that polling watcher creation function. That latter point I think is the contentious one; I'd think that if we know (or very likely know) that the watcher is now bad, that it would need to be recreated via But, it also feels like it should be expensive to be recreating directory watches (which are potentially recursive) over and over again. Maybe that's why the PR uses polling in the first place, because those are effectively "free" given they defer work until some later poll as opposed to asking the system to do some bookkeeping on FS events? Maybe ideally, it'd be nice if there was a cheap way to just get the inode number of a directory and compare it without doing the work to recreate a watcher. But, this feels like something that node's watcher should really be handling, as the API is based on paths. It sure would be a surprise for watchers to just totally stop working depending on the details of what some program does to rename or replace a directory/file with other contents or something. |
I think issue here is the number of times the watch will be recreated (if not using polling). Using polling watch on any change in directory is not ideal either. We already have option to allow I will need to debug and experiment a little to see what better approach will be here. |
ahh ok I think I understand the problem. It's this line:
Should that just be
? I can test this to make sure that the repro steps I outlined in the attached Issue are fixed and update this PR if we think that would resolve the concern here. |
I think that addresses my commentary on the original "it's now going to poll which is wrong" comment, but as mentioned above, recreating a non-polling watcher may get expensive. It'd be worth testing (as it's not a huge change), but I can't claim that's "the thing" that would make the PR correct, compared to some other mechanism to address this problem. |
Well, given that inode changes require a new watch, I see three possible paths:
I'm happy to do either 2 or 3 here. |
Approach 3 is definitely not a generic solution we are looking for.. Just like how we have heuristics for checking directory is still present or not, we would use some kind of heuristics but that needs some experimenting and looking at the events to come up with a plan.. I am going to spend some time on that early next week and will get back to you. |
I have completed testing the change to use |
Closing in favor of #48997 |
Fixes #47466
On unix systems, fs.watch places a watch on an inode, not a file path. If a given file is assigned a new inode, the watch is no longer valid and any changes to that file will not generate a watch for TypeScript to consume. This is quite common when using standard tools like vim and git.
This changes the watch handling code to always recreate the watch any time a rename event occurs and the file still exists. In this case, it is very likely that the inode changed (at least in my experience), so its a simple solution to the problem and I would guess it has low overhead. This patch eliminates the most common cause of this bug. There are still race conditions after this PR if someone is editing a file in the background at the exact same time that tsc is setting the watch for the first time. To fix that, setting a watch should be done as follows:
IMO the TypeScript team should seriously consider just using chokidar. In addition to avoiding bugs like these (both the one fixed here and the one not fixed and described above), it will also provide much better performance on MacOS because it uses the native fsevents system calls.