tail: revert to polling if a followed directory is replaced (#1283760)
This commit is contained in:
parent
e0567d54a7
commit
e00cb1843f
168
coreutils-8.27-tail-inotify-recreate.patch
Normal file
168
coreutils-8.27-tail-inotify-recreate.patch
Normal file
@ -0,0 +1,168 @@
|
|||||||
|
From ba5fe2d4b8b6a8366f48b1ad1f97fe26c9089b53 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Sebastian Kisela <skisela@redhat.com>
|
||||||
|
Date: Wed, 5 Apr 2017 09:40:41 +0200
|
||||||
|
Subject: [PATCH] tail: revert to polling if a followed directory is replaced
|
||||||
|
|
||||||
|
* src/tail.c (tail_forever_inotify): Add the IN_DELETE_SELF flag when
|
||||||
|
creating watch for the parent directory. After the parent directory
|
||||||
|
is removed, an event is caught and then we switch from inotify to
|
||||||
|
polling mode. Till now, inotify has always frozen because it waited for
|
||||||
|
an event from a watched dir, which has been already deleted and was not
|
||||||
|
added again.
|
||||||
|
* tests/tail-2/inotify-dir-recreate.sh: Add a test case.
|
||||||
|
* tests/local.mk: Reference the new test.
|
||||||
|
Fixes http://bugs.gnu.org/26363
|
||||||
|
Reported at https://bugzilla.redhat.com/1283760
|
||||||
|
|
||||||
|
Upstream-commit: ba5fe2d4b8b6a8366f48b1ad1f97fe26c9089b53
|
||||||
|
|
||||||
|
---
|
||||||
|
src/tail.c | 22 +++++++++-
|
||||||
|
tests/local.mk | 1 +
|
||||||
|
tests/tail-2/inotify-dir-recreate.sh | 82 ++++++++++++++++++++++++++++++++++++
|
||||||
|
4 files changed, 107 insertions(+), 1 deletion(-)
|
||||||
|
create mode 100755 tests/tail-2/inotify-dir-recreate.sh
|
||||||
|
|
||||||
|
diff --git a/src/tail.c b/src/tail.c
|
||||||
|
index d1552d4..6328fe0 100644
|
||||||
|
--- a/src/tail.c
|
||||||
|
+++ b/src/tail.c
|
||||||
|
@@ -1457,7 +1457,8 @@ tail_forever_inotify (int wd, struct File_spec *f, size_t n_files,
|
||||||
|
In that case the same watch descriptor is returned. */
|
||||||
|
f[i].parent_wd = inotify_add_watch (wd, dirlen ? f[i].name : ".",
|
||||||
|
(IN_CREATE | IN_DELETE
|
||||||
|
- | IN_MOVED_TO | IN_ATTRIB));
|
||||||
|
+ | IN_MOVED_TO | IN_ATTRIB
|
||||||
|
+ | IN_DELETE_SELF));
|
||||||
|
|
||||||
|
f[i].name[dirlen] = prev;
|
||||||
|
|
||||||
|
@@ -1628,6 +1629,25 @@ tail_forever_inotify (int wd, struct File_spec *f, size_t n_files,
|
||||||
|
ev = void_ev;
|
||||||
|
evbuf_off += sizeof (*ev) + ev->len;
|
||||||
|
|
||||||
|
+ /* If a directory is deleted, IN_DELETE_SELF is emitted
|
||||||
|
+ with ev->name of length 0.
|
||||||
|
+ We need to catch it, otherwise it would wait forever,
|
||||||
|
+ as wd for directory becomes inactive. Revert to polling now. */
|
||||||
|
+ if ((ev->mask & IN_DELETE_SELF) && ! ev->len)
|
||||||
|
+ {
|
||||||
|
+ for (i = 0; i < n_files; i++)
|
||||||
|
+ {
|
||||||
|
+ if (ev->wd == f[i].parent_wd)
|
||||||
|
+ {
|
||||||
|
+ hash_free (wd_to_name);
|
||||||
|
+ error (0, 0,
|
||||||
|
+ _("directory containing watched file was removed"));
|
||||||
|
+ errno = 0; /* we've already diagnosed enough errno detail. */
|
||||||
|
+ return true;
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+
|
||||||
|
if (ev->len) /* event on ev->name in watched directory. */
|
||||||
|
{
|
||||||
|
size_t j;
|
||||||
|
diff --git a/tests/local.mk b/tests/local.mk
|
||||||
|
index 3fe9ba8..e890c9a 100644
|
||||||
|
--- a/tests/local.mk
|
||||||
|
+++ b/tests/local.mk
|
||||||
|
@@ -176,6 +176,7 @@ all_tests = \
|
||||||
|
tests/tail-2/descriptor-vs-rename.sh \
|
||||||
|
tests/tail-2/inotify-rotate.sh \
|
||||||
|
tests/tail-2/inotify-rotate-resources.sh \
|
||||||
|
+ tests/tail-2/inotify-dir-recreate.sh \
|
||||||
|
tests/chmod/no-x.sh \
|
||||||
|
tests/chgrp/basic.sh \
|
||||||
|
tests/rm/dangling-symlink.sh \
|
||||||
|
diff --git a/tests/tail-2/inotify-dir-recreate.sh b/tests/tail-2/inotify-dir-recreate.sh
|
||||||
|
new file mode 100755
|
||||||
|
index 0000000..eaa8422
|
||||||
|
--- /dev/null
|
||||||
|
+++ b/tests/tail-2/inotify-dir-recreate.sh
|
||||||
|
@@ -0,0 +1,82 @@
|
||||||
|
+#!/bin/sh
|
||||||
|
+# Makes sure, inotify will switch to polling mode if directory
|
||||||
|
+# of the watched file was removed and recreated.
|
||||||
|
+# (...instead of getting stuck forever)
|
||||||
|
+
|
||||||
|
+# Copyright (C) 2006-2017 Free Software Foundation, Inc.
|
||||||
|
+
|
||||||
|
+# This program is free software: you can redistribute it and/or modify
|
||||||
|
+# it under the terms of the GNU General Public License as published by
|
||||||
|
+# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
+# (at your option) any later version.
|
||||||
|
+
|
||||||
|
+# This program is distributed in the hope that it will be useful,
|
||||||
|
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
+# GNU General Public License for more details.
|
||||||
|
+
|
||||||
|
+# You should have received a copy of the GNU General Public License
|
||||||
|
+# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
+
|
||||||
|
+. "${srcdir=.}/tests/init.sh"; path_prepend_ ./src
|
||||||
|
+print_ver_ tail
|
||||||
|
+
|
||||||
|
+
|
||||||
|
+# Terminate any background tail process
|
||||||
|
+cleanup_() { kill $pid 2>/dev/null && wait $pid; }
|
||||||
|
+
|
||||||
|
+cleanup_fail_ ()
|
||||||
|
+{
|
||||||
|
+ warn_ $1
|
||||||
|
+ cleanup_
|
||||||
|
+ fail=1
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+# $check_re - string to be found
|
||||||
|
+# $check_f - file to be searched
|
||||||
|
+check_tail_output_ ()
|
||||||
|
+{
|
||||||
|
+ local delay="$1"
|
||||||
|
+ grep $check_re $check_f > /dev/null ||
|
||||||
|
+ { sleep $delay ; return 1; }
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+grep_timeout_ ()
|
||||||
|
+{
|
||||||
|
+ check_re="$1"
|
||||||
|
+ check_f="$2"
|
||||||
|
+ retry_delay_ check_tail_output_ .1 5
|
||||||
|
+}
|
||||||
|
+
|
||||||
|
+# Prepare the file to be watched
|
||||||
|
+mkdir dir && echo 'inotify' > dir/file || framework_failure_
|
||||||
|
+
|
||||||
|
+#tail must print content of the file to stdout, verify
|
||||||
|
+timeout 60 tail -F dir/file &>out & pid=$!
|
||||||
|
+grep_timeout_ 'inotify' 'out' ||
|
||||||
|
+{ cleanup_fail_ 'file to be tailed does not exist'; }
|
||||||
|
+
|
||||||
|
+# Remove the directory, should get the message about the deletion
|
||||||
|
+rm -r dir || framework_failure_
|
||||||
|
+grep_timeout_ 'polling' 'out' ||
|
||||||
|
+{ cleanup_fail_ 'tail did not switch to polling mode'; }
|
||||||
|
+
|
||||||
|
+# Recreate the dir, must get a message about recreation
|
||||||
|
+mkdir dir && touch dir/file || framework_failure_
|
||||||
|
+grep_timeout_ 'appeared' 'out' ||
|
||||||
|
+{ cleanup_fail_ 'previously removed file did not appear'; }
|
||||||
|
+
|
||||||
|
+cleanup_
|
||||||
|
+
|
||||||
|
+# Expected result for the whole process
|
||||||
|
+cat <<\EOF > exp || framework_failure_
|
||||||
|
+inotify
|
||||||
|
+tail: 'dir/file' has become inaccessible: No such file or directory
|
||||||
|
+tail: directory containing watched file was removed
|
||||||
|
+tail: inotify cannot be used, reverting to polling
|
||||||
|
+tail: 'dir/file' has appeared; following new file
|
||||||
|
+EOF
|
||||||
|
+
|
||||||
|
+compare exp out || fail=1
|
||||||
|
+
|
||||||
|
+Exit $fail
|
||||||
|
--
|
||||||
|
2.9.3
|
||||||
|
|
@ -1,7 +1,7 @@
|
|||||||
Summary: A set of basic GNU tools commonly used in shell scripts
|
Summary: A set of basic GNU tools commonly used in shell scripts
|
||||||
Name: coreutils
|
Name: coreutils
|
||||||
Version: 8.27
|
Version: 8.27
|
||||||
Release: 4%{?dist}
|
Release: 5%{?dist}
|
||||||
License: GPLv3+
|
License: GPLv3+
|
||||||
Group: System Environment/Base
|
Group: System Environment/Base
|
||||||
Url: https://www.gnu.org/software/coreutils/
|
Url: https://www.gnu.org/software/coreutils/
|
||||||
@ -22,6 +22,9 @@ Patch1: coreutils-8.27-date-debug-test.patch
|
|||||||
# date, touch: fix out-of-bounds write via large TZ variable (CVE-2017-7476)
|
# date, touch: fix out-of-bounds write via large TZ variable (CVE-2017-7476)
|
||||||
Patch2: coreutils-8.27-CVE-2017-7476.patch
|
Patch2: coreutils-8.27-CVE-2017-7476.patch
|
||||||
|
|
||||||
|
# tail: revert to polling if a followed directory is replaced
|
||||||
|
Patch3: coreutils-8.27-tail-inotify-recreate.patch
|
||||||
|
|
||||||
# disable the test-lock gnulib test prone to deadlock
|
# disable the test-lock gnulib test prone to deadlock
|
||||||
Patch100: coreutils-8.26-test-lock.patch
|
Patch100: coreutils-8.26-test-lock.patch
|
||||||
|
|
||||||
@ -304,6 +307,9 @@ fi
|
|||||||
%license COPYING
|
%license COPYING
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
|
* Fri Apr 28 2017 Sebastian Kisela <skisela@redhat.com> - 8.27-5
|
||||||
|
- tail: revert to polling if a followed directory is replaced
|
||||||
|
|
||||||
* Thu Apr 27 2017 Kamil Dudka <kdudka@redhat.com> - 8.27-4
|
* Thu Apr 27 2017 Kamil Dudka <kdudka@redhat.com> - 8.27-4
|
||||||
- date, touch: fix out-of-bounds write via large TZ variable (CVE-2017-7476)
|
- date, touch: fix out-of-bounds write via large TZ variable (CVE-2017-7476)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user