source: trunk/src/allmydata/test/test_inotify.py @ 2cb710a

Last change on this file since 2cb710a was 2cb710a, checked in by Jean-Paul Calderone <exarkun@…>, at 2019-03-14T15:26:06Z

Just touch the path once

  • Property mode set to 100644
File size: 5.5 KB
Line 
1# Copyright (c) Twisted Matrix Laboratories.
2# See LICENSE for details.
3
4"""
5Tests for the inotify-alike implementation L{allmydata.watchdog}.
6"""
7
8# Note: See https://twistedmatrix.com/trac/ticket/8915 for a proposal
9# to avoid all of this duplicated code from Twisted.
10
11from twisted.internet import defer, reactor
12from twisted.python import filepath, runtime
13
14from allmydata.frontends.magic_folder import get_inotify_module
15from .common import (
16    AsyncTestCase,
17    skipIf,
18)
19inotify = get_inotify_module()
20
21
22@skipIf(runtime.platformType == "win32", "inotify does not yet work on windows")
23class INotifyTests(AsyncTestCase):
24    """
25    Define all the tests for the basic functionality exposed by
26    L{inotify.INotify}.
27    """
28    def setUp(self):
29        self.dirname = filepath.FilePath(self.mktemp())
30        self.dirname.createDirectory()
31        self.inotify = inotify.INotify()
32        self.inotify.startReading()
33        self.addCleanup(self.inotify.stopReading)
34        return super(INotifyTests, self).setUp()
35
36
37    def _notificationTest(self, mask, operation, expectedPath=None):
38        """
39        Test notification from some filesystem operation.
40
41        @param mask: The event mask to use when setting up the watch.
42
43        @param operation: A function which will be called with the
44            name of a file in the watched directory and which should
45            trigger the event.
46
47        @param expectedPath: Optionally, the name of the path which is
48            expected to come back in the notification event; this will
49            also be passed to C{operation} (primarily useful when the
50            operation is being done to the directory itself, not a
51            file in it).
52
53        @return: A L{Deferred} which fires successfully when the
54            expected event has been received or fails otherwise.
55        """
56        if expectedPath is None:
57            expectedPath = self.dirname.child("foo.bar")
58        notified = defer.Deferred()
59        def cbNotified(result):
60            (watch, filename, events) = result
61            self.assertEqual(filename.asBytesMode(), expectedPath.asBytesMode())
62            self.assertTrue(events & mask)
63            self.inotify.ignore(self.dirname)
64        notified.addCallback(cbNotified)
65
66        def notify_event(*args):
67            notified.callback(args)
68        self.inotify.watch(
69            self.dirname, mask=mask,
70            callbacks=[notify_event])
71        operation(expectedPath)
72        return notified
73
74
75    def test_modify(self):
76        """
77        Writing to a file in a monitored directory sends an
78        C{inotify.IN_MODIFY} event to the callback.
79        """
80        def operation(path):
81            with path.open("w") as fObj:
82                fObj.write(b'foo')
83
84        return self._notificationTest(inotify.IN_MODIFY, operation)
85
86
87    def test_attrib(self):
88        """
89        Changing the metadata of a file in a monitored directory
90        sends an C{inotify.IN_ATTRIB} event to the callback.
91        """
92        def operation(path):
93            path.touch()
94
95        return self._notificationTest(inotify.IN_ATTRIB, operation)
96
97
98    def test_closeWrite(self):
99        """
100        Closing a file which was open for writing in a monitored
101        directory sends an C{inotify.IN_CLOSE_WRITE} event to the
102        callback.
103        """
104        def operation(path):
105            path.open("w").close()
106
107        return self._notificationTest(inotify.IN_CLOSE_WRITE, operation)
108
109
110    def test_delete(self):
111        """
112        Deleting a file in a monitored directory sends an
113        C{inotify.IN_DELETE} event to the callback.
114        """
115        expectedPath = self.dirname.child("foo.bar")
116        expectedPath.touch()
117        notified = defer.Deferred()
118        def cbNotified(result):
119            (watch, filename, events) = result
120            self.assertEqual(filename.asBytesMode(), expectedPath.asBytesMode())
121            self.assertTrue(events & inotify.IN_DELETE)
122        notified.addCallback(cbNotified)
123        self.inotify.watch(
124            self.dirname, mask=inotify.IN_DELETE,
125            callbacks=[lambda *args: notified.callback(args)])
126        expectedPath.remove()
127        return notified
128
129
130    def test_humanReadableMask(self):
131        """
132        L{inotify.humaReadableMask} translates all the possible event
133        masks to a human readable string.
134        """
135        for mask, value in inotify._FLAG_TO_HUMAN:
136            self.assertEqual(inotify.humanReadableMask(mask)[0], value)
137
138        checkMask = (
139            inotify.IN_CLOSE_WRITE | inotify.IN_ACCESS | inotify.IN_OPEN)
140        self.assertEqual(
141            set(inotify.humanReadableMask(checkMask)),
142            set(['close_write', 'access', 'open']))
143
144
145    def test_noAutoAddSubdirectory(self):
146        """
147        L{inotify.INotify.watch} with autoAdd==False will stop inotify
148        from watching subdirectories created under the watched one.
149        """
150        def _callback(wp, fp, mask):
151            # We are notified before we actually process new
152            # directories, so we need to defer this check.
153            def _():
154                try:
155                    self.assertFalse(self.inotify._isWatched(subdir))
156                    d.callback(None)
157                except Exception:
158                    d.errback()
159            reactor.callLater(0, _)
160
161        checkMask = inotify.IN_ISDIR | inotify.IN_CREATE
162        self.inotify.watch(
163            self.dirname, mask=checkMask, autoAdd=False,
164            callbacks=[_callback])
165        subdir = self.dirname.child('test')
166        d = defer.Deferred()
167        subdir.createDirectory()
168        return d
Note: See TracBrowser for help on using the repository browser.