#! /bin/bash
# SPDX-License-Identifier: GPL-2.0
# Copyright (c) 2020 Red Hat, Inc.  All Rights Reserved.
#
# FS QA Test No. 603
#
# Test per-type(user, group and project) filesystem quota timers, make sure
# enforcement
#
. ./common/preamble
_begin_fstest auto quick quota

projid=$seq

# Override the default cleanup function.
_cleanup()
{
	_restore_project_quota $projid
	cd /
	rm -f $tmp.*
}

# Import common functions.
. ./common/filter
. ./common/quota

init_files()
{
	local dir=$1

	echo "### Initialize files, and their mode and ownership"
	touch $dir/file{1,2} 2>/dev/null
	chown $qa_user $dir/file{1,2} 2>/dev/null
	chgrp $qa_user $dir/file{1,2} 2>/dev/null
	chmod 777 $dir 2>/dev/null
}

cleanup_files()
{
	echo "### Remove all files"
	rm -f ${1}/file{1,2,3,4,5,6}
}

# When project quota is exceeded, some filesystems return ENOSPC (e.g. XFS),
# some filsystems return EDQUOT(e.g. ext4). The behavior isn't definitized.
# So filter the ENOSPC and EDQUOT output.
filter_enospc_edquot()
{
	# The filter is only for project quota
	if [ "$1" = "P" ];then
		sed -e "s,Disk quota exceeded,EDQUOT|ENOSPC,g" \
		    -e "s,No space left on device,EDQUOT|ENOSPC,g"
	else
		cat -
	fi
}

test_grace()
{
	local type=$1
	local dir=$2
	local bgrace=$3
	local igrace=$4

	init_files $dir
	echo "--- Test block quota ---"
	# Firstly fit below block soft limit
	echo "Write 225 blocks..."
	_su $qa_user -c "$XFS_IO_PROG -c 'pwrite 0 $((225 * $BLOCK_SIZE))' \
		-c fsync $dir/file1" 2>&1 >>$seqres.full | \
		_filter_xfs_io_error | tee -a $seqres.full
	repquota -v -$type $SCRATCH_MNT | grep -v "^root" >>$seqres.full 2>&1
	# Secondly overcome block soft limit
	echo "Rewrite 250 blocks plus 1 byte, over the block softlimit..."
	_su $qa_user -c "$XFS_IO_PROG -c 'pwrite 0 $((250 * $BLOCK_SIZE + 1))' \
		-c fsync $dir/file1" 2>&1 >>$seqres.full | \
		_filter_xfs_io_error | tee -a $seqres.full
	repquota -v -$type $SCRATCH_MNT | grep -v "^root" >>$seqres.full 2>&1
	# Reset grace time here, make below grace time test more accurate
	setquota -$type $qa_user -T $bgrace $igrace $SCRATCH_MNT >>$seqres.full 2>&1
	# Now sleep enough grace time and check that softlimit got enforced
	sleep $((bgrace + 1))
	echo "Try to write 1 one more block after grace..."
	_su $qa_user -c "$XFS_IO_PROG -c 'truncate 0' -c 'pwrite 0 $BLOCK_SIZE' \
		$dir/file2" 2>&1 >>$seqres.full | _filter_xfs_io_error | \
		filter_enospc_edquot $type | tee -a $seqres.full
	repquota -v -$type $SCRATCH_MNT | grep -v "^root" >>$seqres.full 2>&1
	echo "--- Test inode quota ---"
	# And now the softlimit test for inodes
	# First reset space limits so that we don't have problems with
	# space reservations on XFS
	setquota -$type $qa_user 0 0 3 100 $SCRATCH_MNT 
	echo "Create 2 more files, over the inode softlimit..."
	_su $qa_user -c "touch $dir/file3 $dir/file4" 2>&1 >>$seqres.full | \
		_filter_scratch | tee -a $seqres.full
	repquota -v -$type $SCRATCH_MNT  | grep -v "^root" >>$seqres.full 2>&1
	# Reset grace time here, make below grace time test more accurate
	setquota -$type $qa_user -T $bgrace $igrace $SCRATCH_MNT >>$seqres.full 2>&1
	# Wait and check grace time enforcement
	sleep $((igrace+1))
	echo "Try to create one more inode after grace..."
	_su $qa_user -c "touch $dir/file5" 2>&1 >>$seqres.full | \
		filter_enospc_edquot $type | _filter_scratch | \
		tee -a $seqres.full
	repquota -v -$type $SCRATCH_MNT  | grep -v "^root" >>$seqres.full 2>&1
	cleanup_files $dir
}

_require_scratch
# xfs requires v5 format to support all three quota types at the same time
if [ "$FSTYP" = "xfs" ]; then
	_require_scratch_xfs_crc
fi
_require_setquota_project
_require_quota
_require_user
_require_group

_scratch_mkfs >$seqres.full 2>&1
_scratch_enable_pquota
_qmount_option "usrquota,grpquota,prjquota"
_qmount
_force_vfs_quota_testing $SCRATCH_MNT
_require_prjquota $SCRATCH_DEV
BLOCK_SIZE=$(_get_file_block_size $SCRATCH_MNT)
rm -rf $SCRATCH_MNT/t
mkdir $SCRATCH_MNT/t
$XFS_IO_PROG -r -c "chproj $projid" -c "chattr +P" $SCRATCH_MNT/t
_create_project_quota $SCRATCH_MNT/t $projid $qa_user

echo "### Set up different grace timers to each type of quota"
UBGRACE=12
UIGRACE=10
GBGRACE=4
GIGRACE=2
PBGRACE=8
PIGRACE=6

setquota -u $qa_user $((250 * $BLOCK_SIZE / 1024)) \
	$((1000 * $BLOCK_SIZE / 1024)) 3 100 $SCRATCH_MNT
setquota -u -t $UBGRACE $UIGRACE $SCRATCH_MNT
echo; echo "### Test user quota softlimit and grace time"
test_grace u $SCRATCH_MNT $UBGRACE $UIGRACE
# Reset the user quota space & inode limits, avoid it affect later test
setquota -u $qa_user 0 0 0 0 $SCRATCH_MNT

setquota -g $qa_user $((250 * $BLOCK_SIZE / 1024)) \
	$((1000 * $BLOCK_SIZE / 1024)) 3 100 $SCRATCH_MNT
setquota -g -t $GBGRACE $GIGRACE $SCRATCH_MNT
echo; echo "### Test group quota softlimit and grace time"
test_grace g $SCRATCH_MNT $GBGRACE $GIGRACE
# Reset the group quota space & inode limits, avoid it affect later test
setquota -g $qa_user 0 0 0 0 $SCRATCH_MNT

setquota -P $qa_user $((250 * $BLOCK_SIZE / 1024)) \
	$((1000 * $BLOCK_SIZE / 1024)) 3 100 $SCRATCH_MNT
setquota -P -t $PBGRACE $PIGRACE $SCRATCH_MNT
echo; echo "### Test project quota softlimit and grace time"
test_grace P $SCRATCH_MNT/t $PBGRACE $PIGRACE
# Reset the project quota space & inode limits
setquota -P $qa_user 0 0 0 0 $SCRATCH_MNT

# success, all done
status=0
exit
