2016-02-10 12:42:54 +01:00
#!/bin/bash
#
# Copyright (C) 2016 Colin Walters <walters@verbum.org>
#
2018-01-30 20:26:26 +01:00
# SPDX-License-Identifier: LGPL-2.0+
#
2016-02-10 12:42:54 +01:00
# This library is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License as published by the Free Software Foundation; either
# version 2 of the License, or (at your option) any later version.
#
# This library 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
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public
2021-12-06 20:20:55 -05:00
# License along with this library. If not, see <https://www.gnu.org/licenses/>.
2016-02-10 12:42:54 +01:00
set -euo pipefail
. $( dirname $0 ) /libtest.sh
2016-03-31 11:06:05 +01:00
2016-03-31 13:44:27 +01:00
skip_without_fuse
2016-03-31 11:06:05 +01:00
skip_without_user_xattrs
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
setup_test_repository "bare"
2016-02-10 12:42:54 +01:00
2021-03-31 16:41:32 -04:00
echo "1..12"
2016-02-29 14:51:03 +01:00
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
cd ${ test_tmpdir }
2016-02-10 12:42:54 +01:00
mkdir mnt
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
# The default content set amazingly doesn't have a non-broken link
ln -s firstfile files/firstfile-link
$OSTREE commit -b test2 --tree= dir = files
$OSTREE checkout -H test2 checkout-test2
2016-02-10 12:42:54 +01:00
rofiles-fuse checkout-test2 mnt
cleanup_fuse( ) {
fusermount -u ${ test_tmpdir } /mnt || true
}
2019-06-13 15:57:17 -05:00
libtest_exit_cmds += ( cleanup_fuse)
2016-02-10 12:42:54 +01:00
assert_file_has_content mnt/firstfile first
echo "ok mount"
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
# Test open(O_TRUNC) directly and via symlink
for path in firstfile{ ,-link} ; do
if cp /dev/null mnt/${ path } 2>err.txt; then
assert_not_reached " inplace mutation ${ path } "
fi
assert_file_has_content err.txt "Read-only file system"
assert_file_has_content mnt/firstfile first
assert_file_has_content checkout-test2/firstfile first
done
2017-09-06 10:39:43 -04:00
echo "ok failed inplace mutation (open O_TRUNCATE)"
2016-02-10 12:42:54 +01:00
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
# Test chmod
2017-09-06 10:39:43 -04:00
if chmod 0600 mnt/firstfile 2>err.txt; then
assert_not_reached "chmod inplace"
fi
assert_file_has_content err.txt "chmod:.*Read-only file system"
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
# Test chown with regfiles and symlinks
for path in firstfile baz/alink; do
if chown -h $( id -u) mnt/${ path } 2>err.txt; then
assert_not_reached " chown inplace ${ path } "
fi
assert_file_has_content err.txt "chown:.*Read-only file system"
done
# And test via dereferencing a symlink
if chown $( id -u) mnt/firstfile-link 2>err.txt; then
assert_not_reached "chown inplace firstfile-link"
2017-09-06 10:39:43 -04:00
fi
assert_file_has_content err.txt "chown:.*Read-only file system"
echo "ok failed mutation chmod + chown"
2021-03-31 16:41:32 -04:00
if setfattr -n user.foo -v bar mnt/firstfile-link; then
assert_not_reached "set xattr on linked file"
fi
2017-09-06 10:39:43 -04:00
# Test creating new files, using chown + chmod on them as well
2016-02-10 12:42:54 +01:00
echo anewfile-for-fuse > mnt/anewfile-for-fuse
assert_file_has_content mnt/anewfile-for-fuse anewfile-for-fuse
assert_file_has_content checkout-test2/anewfile-for-fuse anewfile-for-fuse
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
ln -s anewfile-for-fuse mnt/anewfile-for-fuse-link
# And also test modifications through a symlink
echo writevialink > mnt/anewfile-for-fuse-link
for path in anewfile-for-fuse{ ,-link} ; do
assert_file_has_content mnt/${ path } writevialink
done
chown $( id -u) mnt/anewfile-for-fuse-link
2016-02-10 12:42:54 +01:00
mkdir mnt/newfusedir
for i in $( seq 5) ; do
echo ${ i } -morenewfuse-${ i } > mnt/newfusedir/test-morenewfuse.${ i }
2017-09-06 10:39:43 -04:00
chmod 0600 mnt/newfusedir/test-morenewfuse.${ i }
chown $( id -u) mnt/newfusedir/test-morenewfuse.${ i }
2016-02-10 12:42:54 +01:00
done
assert_file_has_content checkout-test2/newfusedir/test-morenewfuse.3 3-morenewfuse-3
2021-03-31 16:41:32 -04:00
2016-02-10 12:42:54 +01:00
echo "ok new content"
2021-03-31 16:41:32 -04:00
setfattr -n user.foo -v bar mnt/anewfile-for-fuse
getfattr -d -m . mnt/anewfile-for-fuse > out.txt
assert_file_has_content_literal out.txt 'user.foo="bar"'
echo "ok new xattrs"
2016-02-10 12:42:54 +01:00
rm mnt/baz/cow
assert_not_has_file checkout-test2/baz/cow
rm mnt/baz/another -rf
assert_not_has_dir checkout-test2/baz/another
echo "ok deletion"
2016-02-29 13:23:28 +01:00
${ CMD_PREFIX } ostree --repo= repo commit -b test2 -s fromfuse --link-checkout-speedup --tree= dir = checkout-test2
2016-02-10 12:42:54 +01:00
echo "ok commit"
2016-06-26 10:25:03 -04:00
${ CMD_PREFIX } ostree --repo= repo checkout -U test2 mnt/test2-checkout-copy-fallback
rofiles-fuse: Fix lchown() and hardlink verification for symlinks
If you lchown("symlink") then we were incorrectly trying to chown the
symlink target, rather than the symlink itself. In particular, this cause
cp -a to fail for a broken symlink. Additionally, it was using the
symlink target when verifying writability, rather than the symlink
itself.
To fix this, we need pass AT_SYMLINK_NOFOLLOW in these cases.
In general, the kernel itself will always resolve any symlinks for us
before calling into the fuse backend, so we should really never do any
symlink following in the fuse fs itself. So, we pro-actively add
NOFOLLOW flags to a few other places:
truncate:
In reality this will never be hit, because
the kernel will resolve symlinks before calling us.
access:
It seems the current fuse implementation never calls this
(faccessat w/AT_SYMLINK_NOFOLLOW never reaches the fuse fs)
but if this ever is implemented this is the correct behaviour.
We would ideally do `chmod` but this is not implemented on current kernels.
Because we're not multi-threaded, this is OK anyways.
Further, our write verification wasn't correctly handling the case of hardlinked
symlinks, which can occur for `bare` checkouts but *not* `bare-user` which the
tests were using. Change to `bare` mode to verify that.
Closes: #1137
Approved by: alexlarsson
2017-09-06 09:31:16 +02:00
assert_file_has_content mnt/test2-checkout-copy-fallback/anewfile-for-fuse writevialink
2016-06-26 10:25:03 -04:00
if ${ CMD_PREFIX } ostree --repo= repo checkout -UH test2 mnt/test2-checkout-copy-hardlinked 2>err.txt; then
assert_not_reached "Checking out via hardlinks across mountpoint unexpectedly succeeded!"
fi
2017-03-06 14:51:14 -05:00
assert_file_has_content err.txt "Unable to do hardlink checkout across devices"
2016-06-26 10:25:03 -04:00
echo "ok checkout copy fallback"
2017-09-20 18:38:16 +00:00
# check that O_RDONLY|O_CREAT is handled correctly; used by flock(1) at least
flock mnt/nonexistent-file echo "ok create file in ro mode"
2017-12-14 11:05:00 -05:00
echo "ok flock"
# And now with --copyup enabled
2018-01-05 16:02:58 -05:00
copyup_reset( ) {
cd ${ test_tmpdir }
fusermount -u mnt
rm checkout-test2 -rf
$OSTREE checkout -H test2 checkout-test2
rofiles-fuse --copyup checkout-test2 mnt
}
assert_test_file( ) {
t = $1
f = $2
if ! test ${ t } " ${ f } " ; then
ls -al " ${ f } "
fatal " Failed test ${ t } ${ f } "
fi
}
copyup_reset
2017-12-14 11:05:00 -05:00
assert_file_has_content mnt/firstfile first
echo "ok copyup mount"
2018-01-05 16:02:58 -05:00
# Test O_TRUNC directly
2017-12-14 11:05:00 -05:00
firstfile_orig_inode = $( stat -c %i checkout-test2/firstfile)
2018-01-05 16:02:58 -05:00
echo -n truncating > mnt/firstfile
assert_streq " $( cat mnt/firstfile) " truncating
firstfile_new_inode = $( stat -c %i checkout-test2/firstfile)
assert_not_streq " ${ firstfile_orig_inode } " " ${ firstfile_new_inode } "
assert_test_file -f checkout-test2/firstfile
2021-03-31 16:41:32 -04:00
# Test xattr modifications
copyup_reset
firstfile_orig_inode = $( stat -c %i checkout-test2/firstfile)
setfattr -n user.foo -v bar mnt/firstfile
getfattr -d -m . mnt/firstfile > out.txt
assert_file_has_content_literal out.txt 'user.foo="bar"'
firstfile_new_inode = $( stat -c %i checkout-test2/firstfile)
assert_not_streq " ${ firstfile_orig_inode } " " ${ firstfile_new_inode } "
assert_test_file -f checkout-test2/firstfile
2018-01-05 16:02:58 -05:00
copyup_reset
firstfile_link_orig_inode = $( stat -c %i checkout-test2/firstfile-link)
firstfile_orig_inode = $( stat -c %i checkout-test2/firstfile)
# Now write via the symlink
echo -n truncating > mnt/firstfile-link
assert_streq " $( cat mnt/firstfile) " truncating
2017-12-14 11:05:00 -05:00
firstfile_new_inode = $( stat -c %i checkout-test2/firstfile)
2018-01-05 16:02:58 -05:00
firstfile_link_new_inode = $( stat -c %i checkout-test2/firstfile-link)
2017-12-14 11:05:00 -05:00
assert_not_streq " ${ firstfile_orig_inode } " " ${ firstfile_new_inode } "
2018-01-05 16:02:58 -05:00
assert_streq " ${ firstfile_link_orig_inode } " " ${ firstfile_link_new_inode } "
assert_test_file -f checkout-test2/firstfile
# Verify we didn't replace the link with a regfile somehow
assert_test_file -L checkout-test2/firstfile-link
# These both end up creating new files; in the sed case we'll then do a rename()
copyup_reset
echo "hello new file" > mnt/a-new-non-copyup-file
assert_file_has_content_literal mnt/a-new-non-copyup-file "hello new file"
sed -i -e s,first,second, mnt/firstfile
assert_file_has_content_literal mnt/firstfile "second"
2017-12-14 11:05:00 -05:00
echo "ok copyup"