1
0
mirror of https://github.com/django/django.git synced 2025-01-05 07:55:47 +00:00

Fixed #30361 -- Increased the default timeout of watchman client to 5 seconds and made it customizable.

Made the default timeout of watchman client customizable via
DJANGO_WATCHMAN_TIMEOUT environment variable.
This commit is contained in:
Jacob Green 2019-04-23 09:08:05 -07:00 committed by Mariusz Felisiak
parent efeceba589
commit ed3c59097a
No known key found for this signature in database
GPG Key ID: 2EF56372BA48CD1B
5 changed files with 22 additions and 2 deletions

View File

@ -364,6 +364,7 @@ answer newbie questions, and generally made Django that much better:
Jaap Roes <jaap.roes@gmail.com> Jaap Roes <jaap.roes@gmail.com>
Jack Moffitt <https://metajack.im/> Jack Moffitt <https://metajack.im/>
Jacob Burch <jacobburch@gmail.com> Jacob Burch <jacobburch@gmail.com>
Jacob Green
Jacob Kaplan-Moss <jacob@jacobian.org> Jacob Kaplan-Moss <jacob@jacobian.org>
Jakub Paczkowski <jakub@paczkowski.eu> Jakub Paczkowski <jakub@paczkowski.eu>
Jakub Wilk <jwilk@jwilk.net> Jakub Wilk <jwilk@jwilk.net>

View File

@ -366,11 +366,12 @@ class WatchmanReloader(BaseReloader):
def __init__(self): def __init__(self):
self.roots = defaultdict(set) self.roots = defaultdict(set)
self.processed_request = threading.Event() self.processed_request = threading.Event()
self.client_timeout = int(os.environ.get('DJANGO_WATCHMAN_TIMEOUT', 5))
super().__init__() super().__init__()
@cached_property @cached_property
def client(self): def client(self):
return pywatchman.client() return pywatchman.client(timeout=self.client_timeout)
def _watch_root(self, root): def _watch_root(self, root):
# In practice this shouldn't occur, however, it's possible that a # In practice this shouldn't occur, however, it's possible that a
@ -528,7 +529,7 @@ class WatchmanReloader(BaseReloader):
def check_availability(cls): def check_availability(cls):
if not pywatchman: if not pywatchman:
raise WatchmanUnavailable('pywatchman not installed.') raise WatchmanUnavailable('pywatchman not installed.')
client = pywatchman.client(timeout=0.01) client = pywatchman.client(timeout=0.1)
try: try:
result = client.capabilityCheck() result = client.capabilityCheck()
except Exception: except Exception:

View File

@ -897,6 +897,11 @@ more robust change detection, and a reduction in power usage.
for optimal performance. See the `watchman documentation`_ for information for optimal performance. See the `watchman documentation`_ for information
on how to do this. on how to do this.
.. admonition:: Watchman timeout
The default timeout of ``Watchman`` client is 5 seconds. You can change it
by setting the ``DJANGO_WATCHMAN_TIMEOUT`` environment variable.
.. _Watchman: https://facebook.github.io/watchman/ .. _Watchman: https://facebook.github.io/watchman/
.. _pywatchman: https://pypi.org/project/pywatchman/ .. _pywatchman: https://pypi.org/project/pywatchman/
.. _watchman documentation: https://facebook.github.io/watchman/docs/config.html#ignore_dirs .. _watchman documentation: https://facebook.github.io/watchman/docs/config.html#ignore_dirs

View File

@ -55,3 +55,7 @@ Bugfixes
:class:`~django.contrib.sessions.middleware.SessionMiddleware` subclasses, :class:`~django.contrib.sessions.middleware.SessionMiddleware` subclasses,
rather than requiring :mod:`django.contrib.sessions` to be in rather than requiring :mod:`django.contrib.sessions` to be in
:setting:`INSTALLED_APPS` (:ticket:`30312`). :setting:`INSTALLED_APPS` (:ticket:`30312`).
* Increased the default timeout when using ``Watchman`` to 5 seconds to prevent
falling back to ``StatReloader`` on larger projects and made it customizable
via the ``DJANGO_WATCHMAN_TIMEOUT`` environment variable (:ticket:`30361`).

View File

@ -558,6 +558,11 @@ def skip_unless_watchman_available():
class WatchmanReloaderTests(ReloaderTests, IntegrationTests): class WatchmanReloaderTests(ReloaderTests, IntegrationTests):
RELOADER_CLS = autoreload.WatchmanReloader RELOADER_CLS = autoreload.WatchmanReloader
def setUp(self):
super().setUp()
# Shorten the timeout to speed up tests.
self.reloader.client_timeout = 0.1
def test_watch_glob_ignores_non_existing_directories_two_levels(self): def test_watch_glob_ignores_non_existing_directories_two_levels(self):
with mock.patch.object(self.reloader, '_subscribe') as mocked_subscribe: with mock.patch.object(self.reloader, '_subscribe') as mocked_subscribe:
self.reloader._watch_glob(self.tempdir / 'does_not_exist' / 'more', ['*']) self.reloader._watch_glob(self.tempdir / 'does_not_exist' / 'more', ['*'])
@ -638,6 +643,10 @@ class WatchmanReloaderTests(ReloaderTests, IntegrationTests):
self.reloader.update_watches() self.reloader.update_watches()
self.assertIsInstance(mocked_server_status.call_args[0][0], TestException) self.assertIsInstance(mocked_server_status.call_args[0][0], TestException)
@mock.patch.dict(os.environ, {'DJANGO_WATCHMAN_TIMEOUT': '10'})
def test_setting_timeout_from_environment_variable(self):
self.assertEqual(self.RELOADER_CLS.client_timeout, 10)
@skipIf(on_macos_with_hfs(), "These tests do not work with HFS+ as a filesystem") @skipIf(on_macos_with_hfs(), "These tests do not work with HFS+ as a filesystem")
class StatReloaderTests(ReloaderTests, IntegrationTests): class StatReloaderTests(ReloaderTests, IntegrationTests):