2022-07-05 13:50:23 +00:00
#!/usr/bin/env bash
2023-08-01 15:39:56 +00:00
# Tags: no-parallel, no-fasttest, no-debug
2022-08-06 10:25:42 +00:00
# Tag no-parallel -- to avoid running it in parallel, this will avoid possible issues due to high pressure
2022-07-05 13:50:23 +00:00
# Test that ensures that WRITE lock failure notifies READ.
# In other words to ensure that after WRITE lock failure (DROP),
# READ lock (SELECT) available instantly.
2024-02-21 17:55:30 +00:00
# Creation of a database with Ordinary engine emits a warning.
CLICKHOUSE_CLIENT_SERVER_LOGS_LEVEL = fatal
2022-07-05 13:50:23 +00:00
CUR_DIR = $( cd " $( dirname " ${ BASH_SOURCE [0] } " ) " && pwd )
# shellcheck source=../shell_config.sh
. " $CUR_DIR " /../shell_config.sh
function wait_query_by_id_started( )
{
2023-08-01 10:09:48 +00:00
# As the query we are waiting for is running simultaneously, let's give it a little time to actually start. The
# queries are supposed to run for multiple seconds, so sleeping 0.5 seconds is not a big deal, especially when
# flushing the logs can take up to 3 to 5 seconds. Basically waiting a bit here we can increase the chance that we
# don't have spend precious time on flushing logs.
sleep 0.5
2022-07-05 13:50:23 +00:00
local query_id = $1 && shift
# wait for query to be started
while [ " $( $CLICKHOUSE_CLIENT " $@ " -q " select count() from system.processes where query_id = ' $query_id ' " ) " -ne 1 ] ; do
2022-08-06 10:25:42 +00:00
if [ " $(
2024-09-07 16:20:20 +00:00
$CLICKHOUSE_CLIENT --max_bytes_before_external_group_by 0 -m -q "
2022-08-06 10:25:42 +00:00
system flush logs;
select count( ) from system.query_log
where
event_date >= yesterday( ) and
current_database = '$CLICKHOUSE_DATABASE' and
type = 'QueryFinish' and
query_id = '$query_id'
"
) " -eq 1 ]; then
return 1
else
sleep 0.1
fi
2022-07-05 13:50:23 +00:00
done
2022-08-06 10:25:42 +00:00
return 0
2022-07-05 13:50:23 +00:00
}
# to avoid removal via separate thread
$CLICKHOUSE_CLIENT -q " CREATE DATABASE ${ CLICKHOUSE_DATABASE } _ordinary Engine=Ordinary " --allow_deprecated_database_ordinary= 1
2022-08-06 10:25:42 +00:00
# It is possible that the subsequent after INSERT query will be processed
# only after INSERT finished, it is unlikely, but it happens few times in
# debug build on CI, so if this will happen, then DROP query will be
# finished instantly, and to avoid flakiness we will retry in this case
while :; do
2024-09-07 16:20:20 +00:00
$CLICKHOUSE_CLIENT -m -q "
2022-08-06 10:25:42 +00:00
DROP TABLE IF EXISTS ${ CLICKHOUSE_DATABASE } _ordinary.data_02352;
CREATE TABLE ${ CLICKHOUSE_DATABASE } _ordinary.data_02352 ( key Int) Engine = Null( ) ;
"
insert_query_id = " insert- $( random_str 10) "
# 20 seconds sleep
2023-12-11 17:08:01 +00:00
$CLICKHOUSE_CLIENT --function_sleep_max_microseconds_per_block 20000000 --max_bytes_before_external_group_by 0 --query_id " $insert_query_id " -q " INSERT INTO ${ CLICKHOUSE_DATABASE } _ordinary.data_02352 SELECT sleepEachRow(1) FROM numbers(20) GROUP BY number " &
2022-08-06 10:25:42 +00:00
if ! wait_query_by_id_started " $insert_query_id " ; then
wait
continue
fi
drop_query_id = " drop- $( random_str 10) "
# 10 second wait
$CLICKHOUSE_CLIENT --query_id " $drop_query_id " -q " DROP TABLE ${ CLICKHOUSE_DATABASE } _ordinary.data_02352 SYNC " --lock_acquire_timeout 10 > >(
grep -m1 -o 'WRITE locking attempt on ".*" has timed out'
) 2>& 1 &
if ! wait_query_by_id_started " $drop_query_id " ; then
wait
continue
fi
# Check INSERT query again, and retry if it does not exist.
if ! wait_query_by_id_started " $insert_query_id " ; then
wait
continue
fi
# NOTE: we need to run SELECT after DROP and
# if the bug is there, then the query will wait 20 seconds (INSERT), instead of 10 (DROP) and will fail
#
# 11 seconds wait (DROP + 1 second lag)
$CLICKHOUSE_CLIENT -q " SELECT * FROM ${ CLICKHOUSE_DATABASE } _ordinary.data_02352 " --lock_acquire_timeout 11
# wait DROP and INSERT
wait
break
done | uniq
2022-07-05 13:50:23 +00:00
$CLICKHOUSE_CLIENT -q " DROP DATABASE ${ CLICKHOUSE_DATABASE } _ordinary "