2021-03-19 13:20:19 +00:00
|
|
|
import os
|
2021-01-11 17:37:08 +00:00
|
|
|
import logging
|
|
|
|
import random
|
|
|
|
import string
|
|
|
|
import time
|
|
|
|
|
|
|
|
import pytest
|
2022-07-07 20:19:15 +00:00
|
|
|
from helpers.cluster import ClickHouseCluster
|
2022-09-18 21:47:43 +00:00
|
|
|
from helpers.wait_for_helpers import wait_for_delete_empty_parts
|
2022-10-04 16:15:50 +00:00
|
|
|
from helpers.wait_for_helpers import wait_for_delete_inactive_parts
|
2021-01-11 17:37:08 +00:00
|
|
|
|
2021-12-17 09:00:22 +00:00
|
|
|
|
2021-03-19 13:20:19 +00:00
|
|
|
SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__))
|
2022-03-22 16:39:58 +00:00
|
|
|
COMMON_CONFIGS = [
|
|
|
|
"configs/config.d/bg_processing_pool_conf.xml",
|
|
|
|
"configs/config.d/clusters.xml",
|
|
|
|
]
|
2021-03-19 13:20:19 +00:00
|
|
|
|
|
|
|
|
2022-07-07 20:19:15 +00:00
|
|
|
def replace_config(path, old, new):
|
|
|
|
config = open(path, "r")
|
2021-03-19 13:20:19 +00:00
|
|
|
config_lines = config.readlines()
|
|
|
|
config.close()
|
|
|
|
config_lines = [line.replace(old, new) for line in config_lines]
|
2022-07-07 20:19:15 +00:00
|
|
|
config = open(path, "w")
|
2021-03-19 13:20:19 +00:00
|
|
|
config.writelines(config_lines)
|
|
|
|
config.close()
|
|
|
|
|
|
|
|
|
2021-01-11 17:37:08 +00:00
|
|
|
@pytest.fixture(scope="module")
|
|
|
|
def cluster():
|
|
|
|
try:
|
|
|
|
cluster = ClickHouseCluster(__file__)
|
2021-04-15 08:26:44 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
cluster.add_instance(
|
|
|
|
"node",
|
|
|
|
main_configs=COMMON_CONFIGS + ["configs/config.d/storage_conf.xml"],
|
|
|
|
macros={"cluster": "node", "replica": "0"},
|
|
|
|
with_minio=True,
|
|
|
|
with_zookeeper=True,
|
|
|
|
stay_alive=True,
|
|
|
|
)
|
|
|
|
cluster.add_instance(
|
|
|
|
"node_another_bucket",
|
|
|
|
main_configs=COMMON_CONFIGS
|
|
|
|
+ ["configs/config.d/storage_conf_another_bucket.xml"],
|
|
|
|
macros={"cluster": "node_another_bucket", "replica": "0"},
|
|
|
|
with_zookeeper=True,
|
|
|
|
stay_alive=True,
|
|
|
|
)
|
|
|
|
cluster.add_instance(
|
|
|
|
"node_another_bucket_path",
|
|
|
|
main_configs=COMMON_CONFIGS
|
|
|
|
+ ["configs/config.d/storage_conf_another_bucket_path.xml"],
|
|
|
|
stay_alive=True,
|
|
|
|
)
|
|
|
|
cluster.add_instance(
|
|
|
|
"node_not_restorable",
|
|
|
|
main_configs=COMMON_CONFIGS
|
|
|
|
+ ["configs/config.d/storage_conf_not_restorable.xml"],
|
|
|
|
stay_alive=True,
|
|
|
|
)
|
2021-04-15 08:26:44 +00:00
|
|
|
|
2021-01-11 17:37:08 +00:00
|
|
|
logging.info("Starting cluster...")
|
|
|
|
cluster.start()
|
|
|
|
logging.info("Cluster started")
|
|
|
|
|
|
|
|
yield cluster
|
|
|
|
finally:
|
|
|
|
cluster.shutdown()
|
|
|
|
|
|
|
|
|
|
|
|
def random_string(length):
|
|
|
|
letters = string.ascii_letters
|
2022-03-22 16:39:58 +00:00
|
|
|
return "".join(random.choice(letters) for i in range(length))
|
2021-01-11 17:37:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
def generate_values(date_str, count, sign=1):
|
|
|
|
data = [[date_str, sign * (i + 1), random_string(10)] for i in range(count)]
|
|
|
|
data.sort(key=lambda tup: tup[1])
|
2021-01-12 17:18:40 +00:00
|
|
|
return ",".join(["('{}',{},'{}',{})".format(x, y, z, 0) for x, y, z in data])
|
2021-01-11 17:37:08 +00:00
|
|
|
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
def create_table(
|
|
|
|
node, table_name, attach=False, replicated=False, db_atomic=False, uuid=""
|
|
|
|
):
|
2022-05-14 16:41:54 +00:00
|
|
|
node.query("DROP DATABASE IF EXISTS s3")
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"CREATE DATABASE IF NOT EXISTS s3 ENGINE = {engine}".format(
|
|
|
|
engine="Atomic" if db_atomic else "Ordinary"
|
2022-06-23 19:40:05 +00:00
|
|
|
),
|
|
|
|
settings={"allow_deprecated_database_ordinary": 1},
|
2022-03-22 16:39:58 +00:00
|
|
|
)
|
2021-01-11 17:37:08 +00:00
|
|
|
|
|
|
|
create_table_statement = """
|
2022-01-18 09:31:31 +00:00
|
|
|
{create} TABLE s3.{table_name} {uuid} {on_cluster} (
|
2021-01-11 17:37:08 +00:00
|
|
|
dt Date,
|
|
|
|
id Int64,
|
|
|
|
data String,
|
2021-01-12 17:18:40 +00:00
|
|
|
counter Int64,
|
2021-01-11 17:37:08 +00:00
|
|
|
INDEX min_max (id) TYPE minmax GRANULARITY 3
|
2021-04-15 08:26:44 +00:00
|
|
|
) ENGINE={engine}
|
2021-01-11 17:37:08 +00:00
|
|
|
PARTITION BY dt
|
|
|
|
ORDER BY (dt, id)
|
|
|
|
SETTINGS
|
|
|
|
storage_policy='s3',
|
2022-09-22 22:51:13 +00:00
|
|
|
index_granularity=512,
|
|
|
|
old_parts_lifetime=1
|
2022-03-22 16:39:58 +00:00
|
|
|
""".format(
|
|
|
|
create="ATTACH" if attach else "CREATE",
|
|
|
|
table_name=table_name,
|
|
|
|
uuid="UUID '{uuid}'".format(uuid=uuid) if db_atomic and uuid else "",
|
|
|
|
on_cluster="ON CLUSTER '{}'".format(node.name) if replicated else "",
|
|
|
|
engine="ReplicatedMergeTree('/clickhouse/tables/{cluster}/test', '{replica}')"
|
|
|
|
if replicated
|
|
|
|
else "MergeTree()",
|
|
|
|
)
|
2021-01-11 17:37:08 +00:00
|
|
|
|
|
|
|
node.query(create_table_statement)
|
|
|
|
|
|
|
|
|
2021-01-12 17:18:40 +00:00
|
|
|
def purge_s3(cluster, bucket):
|
|
|
|
minio = cluster.minio_client
|
|
|
|
for obj in list(minio.list_objects(bucket, recursive=True)):
|
2021-03-24 10:05:43 +00:00
|
|
|
if str(obj.object_name).find(".SCHEMA_VERSION") != -1:
|
|
|
|
continue
|
2021-01-12 17:18:40 +00:00
|
|
|
minio.remove_object(bucket, obj.object_name)
|
|
|
|
|
|
|
|
|
|
|
|
def drop_s3_metadata(node):
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", "rm -rf /var/lib/clickhouse/disks/s3/*"], user="root"
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
def drop_shadow_information(node):
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", "rm -rf /var/lib/clickhouse/shadow/*"], user="root"
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
|
2021-04-16 14:43:16 +00:00
|
|
|
def create_restore_file(node, revision=None, bucket=None, path=None, detached=None):
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", "mkdir -p /var/lib/clickhouse/disks/s3/"], user="root"
|
|
|
|
)
|
2022-10-04 16:15:50 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", "touch /var/lib/clickhouse/disks/s3/restore"], user="root"
|
|
|
|
)
|
2021-04-16 14:43:16 +00:00
|
|
|
|
|
|
|
add_restore_option = 'echo -en "{}={}\n" >> /var/lib/clickhouse/disks/s3/restore'
|
|
|
|
if revision:
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", add_restore_option.format("revision", revision)], user="root"
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
if bucket:
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", add_restore_option.format("source_bucket", bucket)],
|
|
|
|
user="root",
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
if path:
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", add_restore_option.format("source_path", path)], user="root"
|
|
|
|
)
|
2021-04-14 21:55:48 +00:00
|
|
|
if detached:
|
2022-03-22 16:39:58 +00:00
|
|
|
node.exec_in_container(
|
|
|
|
["bash", "-c", add_restore_option.format("detached", "true")], user="root"
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
def get_revision_counter(node, backup_number):
|
2022-03-22 16:39:58 +00:00
|
|
|
return int(
|
|
|
|
node.exec_in_container(
|
|
|
|
[
|
|
|
|
"bash",
|
|
|
|
"-c",
|
|
|
|
"cat /var/lib/clickhouse/disks/s3/shadow/{}/revision.txt".format(
|
|
|
|
backup_number
|
|
|
|
),
|
|
|
|
],
|
|
|
|
user="root",
|
|
|
|
)
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
def get_table_uuid(node, db_atomic, table):
|
|
|
|
uuid = ""
|
|
|
|
if db_atomic:
|
2022-03-22 16:39:58 +00:00
|
|
|
uuid = node.query(
|
|
|
|
"SELECT uuid FROM system.tables WHERE database='s3' AND table='{}' FORMAT TabSeparated".format(
|
|
|
|
table
|
|
|
|
)
|
|
|
|
).strip()
|
2022-01-18 09:31:31 +00:00
|
|
|
return uuid
|
|
|
|
|
|
|
|
|
2021-01-11 17:37:08 +00:00
|
|
|
@pytest.fixture(autouse=True)
|
|
|
|
def drop_table(cluster):
|
|
|
|
yield
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node_names = [
|
|
|
|
"node",
|
|
|
|
"node_another_bucket",
|
|
|
|
"node_another_bucket_path",
|
|
|
|
"node_not_restorable",
|
|
|
|
]
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
for node_name in node_names:
|
|
|
|
node = cluster.instances[node_name]
|
2021-04-15 08:26:44 +00:00
|
|
|
node.query("DROP TABLE IF EXISTS s3.test SYNC")
|
|
|
|
node.query("DROP DATABASE IF EXISTS s3 SYNC")
|
2021-01-11 17:37:08 +00:00
|
|
|
|
2021-01-12 17:18:40 +00:00
|
|
|
drop_s3_metadata(node)
|
|
|
|
drop_shadow_information(node)
|
2021-01-11 17:37:08 +00:00
|
|
|
|
2021-01-12 17:18:40 +00:00
|
|
|
buckets = [cluster.minio_bucket, cluster.minio_bucket_2]
|
|
|
|
for bucket in buckets:
|
|
|
|
purge_s3(cluster, bucket)
|
2021-01-11 17:37:08 +00:00
|
|
|
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
@pytest.mark.parametrize("replicated", [False, True])
|
|
|
|
@pytest.mark.parametrize("db_atomic", [False, True])
|
2022-01-18 09:31:31 +00:00
|
|
|
def test_full_restore(cluster, replicated, db_atomic):
|
2021-01-11 17:37:08 +00:00
|
|
|
node = cluster.instances["node"]
|
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
create_table(node, "test", attach=False, replicated=replicated, db_atomic=db_atomic)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-04", 4096, -1))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096, -1))
|
|
|
|
)
|
2021-01-11 17:37:08 +00:00
|
|
|
|
2021-04-21 11:31:09 +00:00
|
|
|
node.query("DETACH TABLE s3.test")
|
2021-01-12 17:18:40 +00:00
|
|
|
drop_s3_metadata(node)
|
|
|
|
create_restore_file(node)
|
2022-12-30 13:47:30 +00:00
|
|
|
node.restart_clickhouse()
|
2021-01-11 17:37:08 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node.query("SELECT count(*) FROM s3.test FORMAT Values") == "({})".format(
|
|
|
|
4096 * 4
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
assert node.query("SELECT sum(id) FROM s3.test FORMAT Values") == "({})".format(0)
|
|
|
|
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
@pytest.mark.parametrize("db_atomic", [False, True])
|
2022-01-18 09:31:31 +00:00
|
|
|
def test_restore_another_bucket_path(cluster, db_atomic):
|
2021-01-12 17:18:40 +00:00
|
|
|
node = cluster.instances["node"]
|
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
create_table(node, "test", db_atomic=db_atomic)
|
|
|
|
uuid = get_table_uuid(node, db_atomic, "test")
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-04", 4096, -1))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096, -1))
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
# To ensure parts have merged
|
|
|
|
node.query("OPTIMIZE TABLE s3.test")
|
2022-10-13 16:07:25 +00:00
|
|
|
wait_for_delete_inactive_parts(node, "s3.test", retry_count=120)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node.query("SELECT count(*) FROM s3.test FORMAT Values") == "({})".format(
|
|
|
|
4096 * 4
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
assert node.query("SELECT sum(id) FROM s3.test FORMAT Values") == "({})".format(0)
|
|
|
|
|
|
|
|
node_another_bucket = cluster.instances["node_another_bucket"]
|
|
|
|
|
|
|
|
create_restore_file(node_another_bucket, bucket="root")
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket, "test", attach=True, db_atomic=db_atomic, uuid=uuid
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 4)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node_another_bucket_path = cluster.instances["node_another_bucket_path"]
|
|
|
|
|
|
|
|
create_restore_file(node_another_bucket_path, bucket="root2", path="data")
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket_path.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket_path, "test", attach=True, db_atomic=db_atomic, uuid=uuid
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket_path.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 4)
|
|
|
|
assert node_another_bucket_path.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
@pytest.mark.parametrize("db_atomic", [False, True])
|
2022-01-18 09:31:31 +00:00
|
|
|
def test_restore_different_revisions(cluster, db_atomic):
|
2021-01-12 17:18:40 +00:00
|
|
|
node = cluster.instances["node"]
|
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
create_table(node, "test", db_atomic=db_atomic)
|
|
|
|
uuid = get_table_uuid(node, db_atomic, "test")
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-04", 4096, -1))
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision1 = get_revision_counter(node, 1)
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096, -1))
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision2 = get_revision_counter(node, 2)
|
|
|
|
|
|
|
|
# To ensure parts have merged
|
|
|
|
node.query("OPTIMIZE TABLE s3.test")
|
2022-10-13 16:07:25 +00:00
|
|
|
wait_for_delete_inactive_parts(node, "s3.test", retry_count=120)
|
|
|
|
|
|
|
|
assert node.query("SELECT count(*) from system.parts where table = 'test'") == "3\n"
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision3 = get_revision_counter(node, 3)
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node.query("SELECT count(*) FROM s3.test FORMAT Values") == "({})".format(
|
|
|
|
4096 * 4
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
assert node.query("SELECT sum(id) FROM s3.test FORMAT Values") == "({})".format(0)
|
2022-10-13 16:07:25 +00:00
|
|
|
assert node.query("SELECT count(*) from system.parts where table = 'test'") == "3\n"
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node_another_bucket = cluster.instances["node_another_bucket"]
|
|
|
|
|
|
|
|
# Restore to revision 1 (2 parts).
|
|
|
|
create_restore_file(node_another_bucket, revision=revision1, bucket="root")
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket, "test", attach=True, db_atomic=db_atomic, uuid=uuid
|
|
|
|
)
|
|
|
|
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 2)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert (
|
|
|
|
node_another_bucket.query(
|
|
|
|
"SELECT count(*) from system.parts where table = 'test'"
|
|
|
|
)
|
|
|
|
== "2\n"
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
# Restore to revision 2 (4 parts).
|
2021-04-21 11:31:09 +00:00
|
|
|
node_another_bucket.query("DETACH TABLE s3.test")
|
2021-01-12 17:18:40 +00:00
|
|
|
create_restore_file(node_another_bucket, revision=revision2, bucket="root")
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 4)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert (
|
|
|
|
node_another_bucket.query(
|
|
|
|
"SELECT count(*) from system.parts where table = 'test'"
|
|
|
|
)
|
|
|
|
== "4\n"
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
# Restore to revision 3 (4 parts + 1 merged).
|
2021-04-21 11:31:09 +00:00
|
|
|
node_another_bucket.query("DETACH TABLE s3.test")
|
2021-01-12 17:18:40 +00:00
|
|
|
create_restore_file(node_another_bucket, revision=revision3, bucket="root")
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 4)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert (
|
|
|
|
node_another_bucket.query(
|
|
|
|
"SELECT count(*) from system.parts where table = 'test'"
|
|
|
|
)
|
2022-10-13 16:07:25 +00:00
|
|
|
== "3\n"
|
2022-03-22 16:39:58 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@pytest.mark.parametrize("db_atomic", [False, True])
|
2022-01-18 09:31:31 +00:00
|
|
|
def test_restore_mutations(cluster, db_atomic):
|
2021-01-12 17:18:40 +00:00
|
|
|
node = cluster.instances["node"]
|
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
create_table(node, "test", db_atomic=db_atomic)
|
|
|
|
uuid = get_table_uuid(node, db_atomic, "test")
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096, -1))
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision_before_mutation = get_revision_counter(node, 1)
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"ALTER TABLE s3.test UPDATE counter = 1 WHERE 1", settings={"mutations_sync": 2}
|
|
|
|
)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision_after_mutation = get_revision_counter(node, 2)
|
|
|
|
|
|
|
|
node_another_bucket = cluster.instances["node_another_bucket"]
|
|
|
|
|
|
|
|
# Restore to revision before mutation.
|
2022-03-22 16:39:58 +00:00
|
|
|
create_restore_file(
|
|
|
|
node_another_bucket, revision=revision_before_mutation, bucket="root"
|
|
|
|
)
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket, "test", attach=True, db_atomic=db_atomic, uuid=uuid
|
|
|
|
)
|
|
|
|
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 2)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
# Restore to revision after mutation.
|
2021-04-21 11:31:09 +00:00
|
|
|
node_another_bucket.query("DETACH TABLE s3.test")
|
2022-03-22 16:39:58 +00:00
|
|
|
create_restore_file(
|
|
|
|
node_another_bucket, revision=revision_after_mutation, bucket="root"
|
|
|
|
)
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 2)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 2)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test WHERE id > 0 FORMAT Values"
|
|
|
|
) == "({})".format(4096)
|
2021-01-12 17:18:40 +00:00
|
|
|
|
|
|
|
# Restore to revision in the middle of mutation.
|
|
|
|
# Unfinished mutation should be completed after table startup.
|
2021-04-21 11:31:09 +00:00
|
|
|
node_another_bucket.query("DETACH TABLE s3.test")
|
2021-01-13 12:05:32 +00:00
|
|
|
revision = (revision_before_mutation + revision_after_mutation) // 2
|
2021-01-12 17:18:40 +00:00
|
|
|
create_restore_file(node_another_bucket, revision=revision, bucket="root")
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2021-01-12 17:18:40 +00:00
|
|
|
|
2021-01-13 12:05:32 +00:00
|
|
|
# Wait for unfinished mutation completion.
|
|
|
|
time.sleep(3)
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 2)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 2)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test WHERE id > 0 FORMAT Values"
|
|
|
|
) == "({})".format(4096)
|
|
|
|
|
|
|
|
|
2022-05-14 16:41:54 +00:00
|
|
|
def test_migrate_to_restorable_schema(cluster):
|
|
|
|
db_atomic = True
|
2021-03-19 13:20:19 +00:00
|
|
|
node = cluster.instances["node_not_restorable"]
|
2022-07-07 20:19:15 +00:00
|
|
|
config_path = os.path.join(
|
|
|
|
SCRIPT_DIR,
|
|
|
|
"./{}/node_not_restorable/configs/config.d/storage_conf_not_restorable.xml".format(
|
|
|
|
cluster.instances_dir_name
|
|
|
|
),
|
|
|
|
)
|
2021-03-19 13:20:19 +00:00
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
create_table(node, "test", db_atomic=db_atomic)
|
|
|
|
uuid = get_table_uuid(node, db_atomic, "test")
|
2021-03-19 13:20:19 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-04", 4096, -1))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096, -1))
|
|
|
|
)
|
|
|
|
|
|
|
|
replace_config(
|
2022-07-07 20:19:15 +00:00
|
|
|
config_path,
|
|
|
|
"<send_metadata>false</send_metadata>",
|
|
|
|
"<send_metadata>true</send_metadata>",
|
2022-03-22 16:39:58 +00:00
|
|
|
)
|
2021-03-19 13:20:19 +00:00
|
|
|
node.restart_clickhouse()
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-06", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-06", 4096, -1))
|
|
|
|
)
|
2021-03-19 13:20:19 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision = get_revision_counter(node, 1)
|
|
|
|
|
|
|
|
assert revision != 0
|
|
|
|
|
|
|
|
node_another_bucket = cluster.instances["node_another_bucket"]
|
|
|
|
|
|
|
|
# Restore to revision before mutation.
|
2022-03-22 16:39:58 +00:00
|
|
|
create_restore_file(
|
|
|
|
node_another_bucket, revision=revision, bucket="root", path="another_data"
|
|
|
|
)
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket, "test", attach=True, db_atomic=db_atomic, uuid=uuid
|
|
|
|
)
|
2021-03-19 13:20:19 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 6)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
2021-04-14 21:55:48 +00:00
|
|
|
|
2022-05-14 16:53:30 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
@pytest.mark.parametrize("replicated", [False, True])
|
|
|
|
@pytest.mark.parametrize("db_atomic", [False, True])
|
2022-01-18 09:31:31 +00:00
|
|
|
def test_restore_to_detached(cluster, replicated, db_atomic):
|
2021-04-14 21:55:48 +00:00
|
|
|
node = cluster.instances["node"]
|
|
|
|
|
2022-01-18 09:31:31 +00:00
|
|
|
create_table(node, "test", attach=False, replicated=replicated, db_atomic=db_atomic)
|
|
|
|
uuid = get_table_uuid(node, db_atomic, "test")
|
2021-04-14 21:55:48 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-04", 4096, -1))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-05", 4096))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-06", 4096, -1))
|
|
|
|
)
|
|
|
|
node.query(
|
|
|
|
"INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-07", 4096, 0))
|
|
|
|
)
|
2021-04-14 21:55:48 +00:00
|
|
|
|
|
|
|
# Add some mutation.
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query(
|
|
|
|
"ALTER TABLE s3.test UPDATE counter = 1 WHERE 1", settings={"mutations_sync": 2}
|
|
|
|
)
|
2021-04-14 21:55:48 +00:00
|
|
|
|
|
|
|
# Detach some partition.
|
|
|
|
node.query("ALTER TABLE s3.test DETACH PARTITION '2020-01-07'")
|
2022-10-04 16:15:50 +00:00
|
|
|
wait_for_delete_empty_parts(node, "s3.test", retry_count=120)
|
|
|
|
wait_for_delete_inactive_parts(node, "s3.test", retry_count=120)
|
2021-04-14 21:55:48 +00:00
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision = get_revision_counter(node, 1)
|
|
|
|
|
|
|
|
node_another_bucket = cluster.instances["node_another_bucket"]
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
create_restore_file(
|
|
|
|
node_another_bucket,
|
|
|
|
revision=revision,
|
|
|
|
bucket="root",
|
|
|
|
path="data",
|
|
|
|
detached=True,
|
|
|
|
)
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket,
|
|
|
|
"test",
|
|
|
|
replicated=replicated,
|
|
|
|
db_atomic=db_atomic,
|
|
|
|
uuid=uuid,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
2021-04-14 21:55:48 +00:00
|
|
|
|
|
|
|
node_another_bucket.query("ALTER TABLE s3.test ATTACH PARTITION '2020-01-03'")
|
|
|
|
node_another_bucket.query("ALTER TABLE s3.test ATTACH PARTITION '2020-01-04'")
|
|
|
|
node_another_bucket.query("ALTER TABLE s3.test ATTACH PARTITION '2020-01-05'")
|
|
|
|
node_another_bucket.query("ALTER TABLE s3.test ATTACH PARTITION '2020-01-06'")
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 4)
|
2022-09-18 21:25:26 +00:00
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 4)
|
2021-04-15 08:26:44 +00:00
|
|
|
|
|
|
|
# Attach partition that was already detached before backup-restore.
|
|
|
|
node_another_bucket.query("ALTER TABLE s3.test ATTACH PARTITION '2020-01-07'")
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 5)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(id) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT sum(counter) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(4096 * 5)
|
2022-01-18 09:31:31 +00:00
|
|
|
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
@pytest.mark.parametrize("replicated", [False, True])
|
|
|
|
@pytest.mark.parametrize("db_atomic", [False, True])
|
2022-01-18 09:31:31 +00:00
|
|
|
def test_restore_without_detached(cluster, replicated, db_atomic):
|
|
|
|
node = cluster.instances["node"]
|
|
|
|
|
|
|
|
create_table(node, "test", attach=False, replicated=replicated, db_atomic=db_atomic)
|
|
|
|
uuid = get_table_uuid(node, db_atomic, "test")
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
node.query("INSERT INTO s3.test VALUES {}".format(generate_values("2020-01-03", 1)))
|
2022-01-18 09:31:31 +00:00
|
|
|
|
|
|
|
assert node.query("SELECT count(*) FROM s3.test FORMAT Values") == "({})".format(1)
|
|
|
|
|
|
|
|
node.query("ALTER TABLE s3.test FREEZE")
|
|
|
|
revision = get_revision_counter(node, 1)
|
|
|
|
|
|
|
|
node_another_bucket = cluster.instances["node_another_bucket"]
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
create_restore_file(
|
|
|
|
node_another_bucket,
|
|
|
|
revision=revision,
|
|
|
|
bucket="root",
|
|
|
|
path="data",
|
|
|
|
detached=True,
|
|
|
|
)
|
2022-12-30 13:47:30 +00:00
|
|
|
node_another_bucket.restart_clickhouse()
|
2022-03-22 16:39:58 +00:00
|
|
|
create_table(
|
|
|
|
node_another_bucket,
|
|
|
|
"test",
|
|
|
|
replicated=replicated,
|
|
|
|
db_atomic=db_atomic,
|
|
|
|
uuid=uuid,
|
|
|
|
)
|
|
|
|
|
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(0)
|
2022-01-18 09:31:31 +00:00
|
|
|
|
|
|
|
node_another_bucket.query("ALTER TABLE s3.test ATTACH PARTITION '2020-01-03'")
|
|
|
|
|
2022-03-22 16:39:58 +00:00
|
|
|
assert node_another_bucket.query(
|
|
|
|
"SELECT count(*) FROM s3.test FORMAT Values"
|
|
|
|
) == "({})".format(1)
|