cp: with --no-preserve=mode ensure set-group-ID bits maintained on dirs
[coreutils.git] / tests / mv / part-symlink.sh
blob9a109d301ab69e1a02f269cd575568fd917cacb2
1 #!/bin/sh
2 # make sure cp and mv can handle many combinations of local and
3 # other-partition regular/symlink'd files.
5 # Copyright (C) 2000-2024 Free Software Foundation, Inc.
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU General Public License as published by
9 # the Free Software Foundation, either version 3 of the License, or
10 # (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU General Public License for more details.
17 # You should have received a copy of the GNU General Public License
18 # along with this program. If not, see <https://www.gnu.org/licenses/>.
20 . "${srcdir=.}/tests/init.sh"; path_prepend_ ./src
21 print_ver_ cp mv
23 cleanup_() { rm -rf "$other_partition_tmpdir"; }
24 . "$abs_srcdir/tests/other-fs-tmpdir"
26 # On NFS on Linux 2.6.9 at least we get:
27 # mv: preserving permissions for 'rem_sl': Operation not supported
28 require_local_dir_
30 pwd_tmp=$(pwd)
32 # Unset CDPATH. Otherwise, output from the 'cd dir' command
33 # can make this test fail.
34 (unset CDPATH) >/dev/null 2>&1 && unset CDPATH
37 # Four cases:
38 # local regular file w/symlink on another partition
39 # (loc_reg, rem_sl)
40 # (rem_sl, loc_reg)
41 # local symlink to regular file on another partition
42 # (loc_sl, rem_reg)
43 # (rem_reg, loc_sl)
45 # Exercise those four cases for each of
46 # cp and mv, with lots of combinations of options.
48 exec 1> actual
50 # FIXME: This should be bigger: like more than 8k
51 contents=XYZ
53 loc_reg=loc_reg
54 loc_sl=loc_sl
55 rem_reg=$other_partition_tmpdir/rem_reg
56 rem_sl=$other_partition_tmpdir/rem_sl
58 for copy in cp mv; do
59 for args in \
60 'loc_reg rem_sl' \
61 'rem_sl loc_reg' \
62 'loc_sl rem_reg' \
63 'rem_reg loc_sl' \
64 ; do
65 for options in '' --rem '--rem -d' '--rem -b' -b -bd -d; do
66 case "$options" in *d*|*--rem*) test $copy = mv && continue;; esac
67 rm -rf dir || fail=1
68 rm -f "$other_partition_tmpdir"/* || fail=1
69 mkdir dir || fail=1
70 cd dir || fail=1
71 case "$args" in *loc_reg*) reg_abs="$(pwd)/$loc_reg" ;; esac
72 case "$args" in *rem_reg*) reg_abs=$rem_reg ;; esac
73 case "$args" in *loc_sl*) slink=$loc_sl ;; esac
74 case "$args" in *rem_sl*) slink=$rem_sl ;; esac
76 echo $contents > "$reg_abs" || framework_failure_
77 ln -nsf "$reg_abs" $slink || fail=1
78 actual_args=$(echo $args|sed 's,^,$,;s/ / $/')
79 actual_args=$(eval echo $actual_args)
83 # echo 1>&2 cp $options $args
84 $copy $options $actual_args 2>.err
85 copy_status=$?
86 echo $copy_status $copy $options $args
88 # Normalize the program name in the error output,
89 # remove any site-dependent part of other-partition file name,
90 # and put brackets around the output.
91 test -s .err \
92 && {
93 echo ' [' | tr -d '\n'
94 sed 's/^[^:][^:]*\(..\):/\1:/;s,'"$other_partition_tmpdir/,," .err |
95 tr -d '\n'
96 echo ']'
98 # Strip off all but the file names.
99 # Remove any site-dependent part of each file name.
100 ls=$(ls -gG --ignore=.err . \
101 | sed \
102 -e '/^total /d' \
103 -e "s,$other_partition_tmpdir/,," \
104 -e "s,$pwd_tmp/,," \
105 -e 's/^[^ ]* *[^ ]* *[^ ]* *[^ ]* *[^ ]* *[^ ]* *//')
106 ls2=$(cd "$other_partition_tmpdir" && ls -gG --ignore=.err . \
107 | sed \
108 -e '/^total /d' \
109 -e "s,$other_partition_tmpdir/,," \
110 -e "s,$pwd_tmp/,," \
111 -e 's/^[^ ]* *[^ ]* *[^ ]* *[^ ]* *[^ ]* *[^ ]* *//')
112 echo " ("$ls") ("$ls2")"
114 # If the command failed, then it must not have changed the files.
115 if test $copy_status != 0; then
116 for f in $actual_args; do
117 test -f $f ||
118 { echo " $copy FAILED but removed $f"; continue; }
119 case "$(cat $f)" in
120 "$contents") ;;
121 *) echo " $copy FAILED but modified $f";;
122 esac
123 done
126 if test $copy = cp; then
127 # Make sure the original is unchanged and that
128 # the destination is a copy.
129 for f in $actual_args; do
130 if test -f $f; then
131 if test $copy_status != 0; then
132 test
134 case "$(cat $f)" in
135 "$contents") ;;
136 *) echo " $copy FAILED";;
137 esac
138 else
139 echo " symlink-loop"
141 done
144 ) | sed 's/ *$//'
145 cd ..
146 done
147 echo
148 done
149 done
151 test $fail = 1 &&
152 { (exit 1); exit; }
154 cat <<\EOF > expected || framework_failure_
155 1 cp loc_reg rem_sl
156 [cp: 'loc_reg' and 'rem_sl' are the same file]
157 (loc_reg) (rem_sl -> dir/loc_reg)
158 0 cp --rem loc_reg rem_sl
159 (loc_reg) (rem_sl)
160 0 cp --rem -d loc_reg rem_sl
161 (loc_reg) (rem_sl)
162 0 cp --rem -b loc_reg rem_sl
163 (loc_reg) (rem_sl rem_sl~ -> dir/loc_reg)
164 0 cp -b loc_reg rem_sl
165 (loc_reg) (rem_sl rem_sl~ -> dir/loc_reg)
166 0 cp -bd loc_reg rem_sl
167 (loc_reg) (rem_sl rem_sl~ -> dir/loc_reg)
168 1 cp -d loc_reg rem_sl
169 [cp: 'loc_reg' and 'rem_sl' are the same file]
170 (loc_reg) (rem_sl -> dir/loc_reg)
172 1 cp rem_sl loc_reg
173 [cp: 'rem_sl' and 'loc_reg' are the same file]
174 (loc_reg) (rem_sl -> dir/loc_reg)
175 1 cp --rem rem_sl loc_reg
176 [cp: 'rem_sl' and 'loc_reg' are the same file]
177 (loc_reg) (rem_sl -> dir/loc_reg)
178 1 cp --rem -d rem_sl loc_reg
179 [cp: 'rem_sl' and 'loc_reg' are the same file]
180 (loc_reg) (rem_sl -> dir/loc_reg)
181 1 cp --rem -b rem_sl loc_reg
182 [cp: 'rem_sl' and 'loc_reg' are the same file]
183 (loc_reg) (rem_sl -> dir/loc_reg)
184 1 cp -b rem_sl loc_reg
185 [cp: 'rem_sl' and 'loc_reg' are the same file]
186 (loc_reg) (rem_sl -> dir/loc_reg)
187 0 cp -bd rem_sl loc_reg
188 (loc_reg -> dir/loc_reg loc_reg~) (rem_sl -> dir/loc_reg)
189 symlink-loop
190 symlink-loop
191 1 cp -d rem_sl loc_reg
192 [cp: 'rem_sl' and 'loc_reg' are the same file]
193 (loc_reg) (rem_sl -> dir/loc_reg)
195 1 cp loc_sl rem_reg
196 [cp: 'loc_sl' and 'rem_reg' are the same file]
197 (loc_sl -> rem_reg) (rem_reg)
198 1 cp --rem loc_sl rem_reg
199 [cp: 'loc_sl' and 'rem_reg' are the same file]
200 (loc_sl -> rem_reg) (rem_reg)
201 1 cp --rem -d loc_sl rem_reg
202 [cp: 'loc_sl' and 'rem_reg' are the same file]
203 (loc_sl -> rem_reg) (rem_reg)
204 1 cp --rem -b loc_sl rem_reg
205 [cp: 'loc_sl' and 'rem_reg' are the same file]
206 (loc_sl -> rem_reg) (rem_reg)
207 1 cp -b loc_sl rem_reg
208 [cp: 'loc_sl' and 'rem_reg' are the same file]
209 (loc_sl -> rem_reg) (rem_reg)
210 0 cp -bd loc_sl rem_reg
211 (loc_sl -> rem_reg) (rem_reg -> rem_reg rem_reg~)
212 symlink-loop
213 symlink-loop
214 1 cp -d loc_sl rem_reg
215 [cp: 'loc_sl' and 'rem_reg' are the same file]
216 (loc_sl -> rem_reg) (rem_reg)
218 1 cp rem_reg loc_sl
219 [cp: 'rem_reg' and 'loc_sl' are the same file]
220 (loc_sl -> rem_reg) (rem_reg)
221 0 cp --rem rem_reg loc_sl
222 (loc_sl) (rem_reg)
223 0 cp --rem -d rem_reg loc_sl
224 (loc_sl) (rem_reg)
225 0 cp --rem -b rem_reg loc_sl
226 (loc_sl loc_sl~ -> rem_reg) (rem_reg)
227 0 cp -b rem_reg loc_sl
228 (loc_sl loc_sl~ -> rem_reg) (rem_reg)
229 0 cp -bd rem_reg loc_sl
230 (loc_sl loc_sl~ -> rem_reg) (rem_reg)
231 1 cp -d rem_reg loc_sl
232 [cp: 'rem_reg' and 'loc_sl' are the same file]
233 (loc_sl -> rem_reg) (rem_reg)
235 0 mv loc_reg rem_sl
236 () (rem_sl)
237 0 mv -b loc_reg rem_sl
238 () (rem_sl rem_sl~ -> dir/loc_reg)
240 1 mv rem_sl loc_reg
241 [mv: 'rem_sl' and 'loc_reg' are the same file]
242 (loc_reg) (rem_sl -> dir/loc_reg)
243 0 mv -b rem_sl loc_reg
244 (loc_reg -> dir/loc_reg loc_reg~) ()
246 1 mv loc_sl rem_reg
247 [mv: 'loc_sl' and 'rem_reg' are the same file]
248 (loc_sl -> rem_reg) (rem_reg)
249 0 mv -b loc_sl rem_reg
250 () (rem_reg -> rem_reg rem_reg~)
252 0 mv rem_reg loc_sl
253 (loc_sl) ()
254 0 mv -b rem_reg loc_sl
255 (loc_sl loc_sl~ -> rem_reg) ()
259 # Redirect to stderr, since stdout is already taken.
260 compare expected actual 1>&2 || fail=1
262 Exit $fail