"rm -rf /etc/motd" (run by non-root) now prints a diagnostic.
authorJim Meyering <jim@meyering.net>
Wed, 20 Dec 2006 18:56:20 +0000 (19:56 +0100)
committerJim Meyering <jim@meyering.net>
Thu, 21 Dec 2006 07:58:11 +0000 (08:58 +0100)
* src/remove.c (remove_entry): Handle EACCES for a non-directory, too.
Karl Berry reported that a cross-partition "mv /etc/issue ~"
failed with the um,... suboptimal diagnostic,
"mv: cannot remove `/etc/issue': Not a directory".
* tests/rm/Makefile.am (TESTS): Add fail-eacces.
* tests/rm/fail-eacces: New file.
* NEWS: Mention that both mv and rm are affected.

ChangeLog
NEWS
src/remove.c
tests/rm/Makefile.am
tests/rm/fail-eacces [new file with mode: 0755]

index ded5770..67aff09 100644 (file)
--- a/ChangeLog
+++ b/ChangeLog
@@ -1,5 +1,17 @@
 2006-12-20  Jim Meyering  <jim@meyering.net>
 
+       "rm -rf /etc/motd" (run by non-root) now prints a diagnostic.
+       * src/remove.c (remove_entry): Handle EACCES for a non-directory, too.
+       Don't let a non-directory get by with errno == EPERM, either.
+       Check the file type directly (using cached stat value), rather
+       than trying to guess it from errno values.
+       Karl Berry reported that a cross-partition "mv /etc/issue ~"
+       failed with the um,... suboptimal diagnostic,
+       "mv: cannot remove `/etc/issue': Not a directory".
+       * tests/rm/Makefile.am (TESTS): Add fail-eacces.
+       * tests/rm/fail-eacces: New file.
+       * NEWS: Mention that both mv and rm are affected.
+
        "cut -f 2- A B" no longer triggers a double-free bug
        * src/cut.c (cut_fields): Set file-scoped global to NULL after
        freeing it.  This avoids a double-free (and core dump on some systems)
diff --git a/NEWS b/NEWS
index d4e73a8..bf1d287 100644 (file)
--- a/NEWS
+++ b/NEWS
@@ -14,6 +14,13 @@ GNU coreutils NEWS                                    -*- outline -*-
   more file arguments.  This was due to a double-free bug, introduced
   in coreutils-5.3.0.
 
+  A cross-partition "mv /etc/passwd ~" (by non-root) now prints
+  a reasonable diagnostic.  Before, it would print this:
+  "mv: cannot remove `/etc/passwd': Not a directory".
+
+  "rm -rf /etc/passwd" (run by non-root) now prints a diagnostic.
+  Before it would print nothing.
+
 * Noteworthy changes in release 6.7 (2006-12-08) [stable]
 
 ** Bug fixes
index 3dc6929..f6d3f0c 100644 (file)
@@ -1016,8 +1016,7 @@ remove_entry (int fd_cwd, Dirstack_state const *ds, char const *filename,
        errno = EISDIR;
 
       if (! x->recursive
-         || errno == ENOENT || errno == ENOTDIR
-         || errno == ELOOP || errno == ENAMETOOLONG)
+         || (cache_stat_ok (st) && !S_ISDIR (st->st_mode)))
        {
          if (ignorable_missing (x, errno))
            return RM_OK;
@@ -1028,6 +1027,7 @@ remove_entry (int fd_cwd, Dirstack_state const *ds, char const *filename,
                 quote (full_filename (filename)));
          return RM_ERROR;
        }
+      assert (!cache_stat_ok (st) || S_ISDIR (st->st_mode));
     }
   else
     {
index 6670307..f632100 100644 (file)
@@ -21,6 +21,7 @@
 AUTOMAKE_OPTIONS = 1.1 gnits
 
 TESTS = \
+  fail-eacces \
   one-file-system \
   ignorable \
   readdir-bug \
diff --git a/tests/rm/fail-eacces b/tests/rm/fail-eacces
new file mode 100755 (executable)
index 0000000..29bc911
--- /dev/null
@@ -0,0 +1,53 @@
+#!/bin/sh
+# Ensure that rm -rf unremovable-non-dir gives a diagnostic.
+
+# Copyright (C) 2006 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 2 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, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+# 02110-1301, USA.
+
+if test "$VERBOSE" = yes; then
+  set -x
+  rm --version
+fi
+
+. $srcdir/../lang-default
+
+pwd=`pwd`
+t0=`echo "$0"|sed 's,.*/,,'`.tmp; tmp=$t0/$$
+trap 'status=$?; cd "$pwd" && chmod -R u+rwx $t0 && rm -rf $t0 && exit $status' 0
+trap '(exit $?); exit $?' 1 2 13 15
+
+framework_failure=0
+mkdir -p $tmp || framework_failure=1
+cd $tmp || framework_failure=1
+mkdir d && touch d/f && chmod a-w d || framework_failure=1
+
+if test $framework_failure = 1; then
+  echo "$0: failure in testing framework" 1>&2
+  (exit 1); exit 1
+fi
+
+fail=0
+
+rm -rf d/f 2> out && fail=1
+cat <<\EOF > exp
+rm: cannot remove `d/f': Permission denied
+EOF
+
+cmp out exp || fail=1
+test $fail = 1 && diff out exp 2> /dev/null
+
+(exit $fail); exit $fail