Opened 10 years ago
Closed 9 years ago
#23882 closed Bug (fixed)
django 1.7 + inotify breaks autoreload of runserver with vim
Reported by: | Thomas Capricelli | Owned by: | nobody |
---|---|---|---|
Component: | Core (Other) | Version: | 1.7 |
Severity: | Normal | Keywords: | autoreload |
Cc: | cmawebsite@…, chris.bainbridge@… | Triage Stage: | Accepted |
Has patch: | no | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description
Quick summary : since i've migrated to django-1.7, autoreload is broken on my systems. I've used django since 0.96 and i have never had any problem with this great feature.
I've turned around the problem until today by removing pyinotify : that fixed the problem for me. But of course i need pyinotify for other things...
Today I've spent time testing more thoroughly, here are the results. My system is fairly standard : linux, gentoo, python 2.7, and vim-7.4.488.
I patched django/utils/autoreload.py to add some debug to get more information.
*) when pyinotify is not installed, USE_INOTIFY is indeed False and everything is working as expected and as before.
*) when pyinotify is installed, USE_INOTIFY is True, and the file i'm testing with (main/views/gestion.py) is indeed added to the inotify watch list (around line 185 of autoreload.py). I need to load a page for this to happen, but this is expected because the autoreload stuff only consider those modules loaded by the python interpreter iiuc.
In this setup, if i use CLI and does "touch main/views/gestion.py", the autoreload is triggered (displayed in the output of runserver).
With the same setup, if i open the file with vim and just do ":w", which saves the file again, nothing is triggered. I tried doing an actual modification and save (in case vim or inotify of even django optimizes the stuff by not reloading a file whose content hasn't changed), but the behaviour is the same.
I can confirm the file was not reloaded by changing something in the file and testing from my browser.
I think to remember that vim doesn't just overwrite the file but instead writes the content to a new file and then relinks the files, which might be the reason inotify doesn't notice the change. But i can't guarantee this.
I know of so many people using django+vim, that i'm very surprised nobody was hit by this before, so maybe I'm not understanding something obvious and the reason is somewhere else ..
I'm "orzel" on irc, very often available, don't hesitate to ping me (or here) if you need some more information/test.
Change History (14)
comment:1 by , 10 years ago
comment:2 by , 10 years ago
Triage Stage: | Unreviewed → Accepted |
---|
It may be that we should also monitor the pyinotify.IN_CLOSE_WRITE
event. Could you try to add it in the inotify_code_changed
function and see if it makes a difference?
comment:3 by , 10 years ago
It doesn't work. I've read http://stackoverflow.com/questions/13312794/inotify-dont-treat-vim-editting-as-a-modification-event as well, and i agree it should. But nope :-(
I even tried putting all the flags on :
mask = (
pyinotify.IN_MODIFY |
pyinotify.IN_ATTRIB |
pyinotify.IN_CLOSE_WRITE |
pyinotify.IN_CLOSE_NOWRITE |
pyinotify.IN_MOVED_FROM |
pyinotify.IN_MOVED_TO |
pyinotify.IN_CREATE |
pyinotify.IN_DELETE |
pyinotify.IN_DELETE_SELF |
pyinotify.IN_MOVE_SELF |
pyinotify.IN_ACCESS |
pyinotify.IN_OPEN
)
that's actually 4095 !
comment:4 by , 10 years ago
I did some more test and i'm getting confused. Using only mask = ( pyinotify.IN_CREATE), i have it triggered if i actually modify the file and save. Wont trigger if i don't change the file before writting.
comment:5 by , 10 years ago
I'm using the "backup" option in ~/.vimrc, which might explain both the problem and why the problem is not as widespread as i thought it would. It seems that in this case, vim just moves the file to the backup location.
I tested without this option though, and the behaviour is the same. I'm lost :-(
comment:6 by , 10 years ago
Cc: | added |
---|---|
Keywords: | autoreload added |
comment:7 by , 10 years ago
#24875 was a dup (where adding IN_DELETE_SELF
and IN_MOVE_SELF
events was suggested).
comment:8 by , 10 years ago
The patch I provided in bug #24875 fixed this issue for me. Required flags:
pyinotify.IN_CREATE | pyinotify.IN_DELETE_SELF | pyinotify.IN_MOVE_SELF
@orzel Try the following test code, run as python test dir_to_watch
then start vim and edit/save a file in the test directory. The notifier loop will print whatever inotify events occur:
import sys import pyinotify wm = pyinotify.WatchManager() wm.add_watch(sys.argv[1], pyinotify.ALL_EVENTS) notifier = pyinotify.Notifier(wm) notifier.loop()
Django does not watch the whole directory, but instead generates a list of paths to watch based on the current python imports. This means there are some cases where files are not initially watched - basically any python file that is dynamically loaded at runtime, such as a view specified by a string literal in urls.py. These files will be watched after the corresponding web page has been loaded for the first time.
You can see the current list of paths being watched by adding a print statement to django/utils/autoreload.py:inotify_code_changed()
comment:9 by , 10 years ago
I had tried adding IN_DELETE_SELF and IN_MOVE_SELF since the beginning, but this doesn't work here. I actually still have those locally added.
comment:10 by , 10 years ago
I'll commit the patch and leave the ticket open since it appears there are still some unresolved cases.
comment:13 by , 10 years ago
Cc: | added |
---|
comment:14 by , 9 years ago
Resolution: | → fixed |
---|---|
Status: | new → closed |
I guess we can close these. If anyone still has problems, please open a new ticket with details.
For completness, inotify is compiled in my kernel, and works perfectly with other software using it. The "touch" test described above kinda confirms this.