import inspect
from contextlib import nullcontext as does_not_raise
import pytest
import time
import os.path
from helpers.cluster import ClickHouseCluster
from helpers.client import QueryRuntimeException
from helpers.test_tools import assert_eq_with_retry, TSV
SCRIPT_DIR = os.path.dirname(os.path.realpath(__file__))
cluster = ClickHouseCluster(__file__, zookeeper_config_path="configs/zookeeper.xml")
node1 = cluster.add_instance(
"node1",
main_configs=["configs/config.xml"],
with_zookeeper=True,
stay_alive=True,
)
node2 = cluster.add_instance(
"node2",
main_configs=["configs/config.xml"],
with_zookeeper=True,
stay_alive=True,
)
all_nodes = [node1, node2]
@pytest.fixture(scope="module", autouse=True)
def started_cluster():
try:
cluster.start()
yield cluster
finally:
cluster.shutdown()
def wait_zookeeper_node_to_start(zk_nodes, timeout=60):
start = time.time()
while time.time() - start < timeout:
try:
for instance in zk_nodes:
conn = cluster.get_kazoo_client(instance)
conn.get_children("/")
print("All instances of ZooKeeper started")
return
except Exception as ex:
print(("Can't connect to ZooKeeper " + str(ex)))
time.sleep(0.5)
def replace_zookeeper_config(new_config):
node1.replace_config("/etc/clickhouse-server/conf.d/zookeeper.xml", new_config)
node2.replace_config("/etc/clickhouse-server/conf.d/zookeeper.xml", new_config)
node1.query("SYSTEM RELOAD CONFIG")
node2.query("SYSTEM RELOAD CONFIG")
def revert_zookeeper_config():
with open(os.path.join(SCRIPT_DIR, "configs/zookeeper.xml"), "r") as f:
replace_zookeeper_config(f.read())
def get_active_zk_connections():
return str(
node1.exec_in_container(
[
"bash",
"-c",
"lsof -a -i4 -i6 -itcp -w | grep 2181 | grep ESTABLISHED | wc -l",
],
privileged=True,
user="root",
)
).strip()
def test_create_and_drop():
node1.query("CREATE FUNCTION f1 AS (x, y) -> x + y")
assert node1.query("SELECT f1(12, 3)") == "15\n"
node1.query("DROP FUNCTION f1")
@pytest.mark.parametrize(
"ignore, expected_raise",
[("true", does_not_raise()), ("false", pytest.raises(QueryRuntimeException))],
)
def test_create_and_drop_udf_on_cluster(ignore, expected_raise):
node1.replace_config(
"/etc/clickhouse-server/users.d/users.xml",
inspect.cleandoc(
f"""
{ignore}
"""
),
)
node1.query("SYSTEM RELOAD CONFIG")
with expected_raise:
node1.query("CREATE FUNCTION f1 ON CLUSTER default AS (x, y) -> x + y")
assert node1.query("SELECT f1(12, 3)") == "15\n"
node1.query("DROP FUNCTION f1 ON CLUSTER default")
def test_create_and_replace():
node1.query("CREATE FUNCTION f1 AS (x, y) -> x + y")
assert node1.query("SELECT f1(12, 3)") == "15\n"
expected_error = "User-defined object 'f1' already exists"
assert expected_error in node1.query_and_get_error(
"CREATE FUNCTION f1 AS (x, y) -> x + 2 * y"
)
node1.query("CREATE FUNCTION IF NOT EXISTS f1 AS (x, y) -> x + 3 * y")
assert node1.query("SELECT f1(12, 3)") == "15\n"
node1.query("CREATE OR REPLACE FUNCTION f1 AS (x, y) -> x + 4 * y")
assert node1.query("SELECT f1(12, 3)") == "24\n"
node1.query("DROP FUNCTION f1")
def test_drop_if_exists():
node1.query("CREATE FUNCTION f1 AS (x, y) -> x + y")
node1.query("DROP FUNCTION IF EXISTS f1")
node1.query("DROP FUNCTION IF EXISTS f1")
expected_error = "User-defined object 'f1' doesn't exist"
assert expected_error in node1.query_and_get_error("DROP FUNCTION f1")
def test_replication():
node1.query("CREATE FUNCTION f2 AS (x, y) -> x - y")
assert (
node1.query("SELECT create_query FROM system.functions WHERE name='f2'")
== "CREATE FUNCTION f2 AS (x, y) -> (x - y)\n"
)
assert_eq_with_retry(
node2,
"SELECT create_query FROM system.functions WHERE name='f2'",
"CREATE FUNCTION f2 AS (x, y) -> (x - y)\n",
)
assert node1.query("SELECT f2(12,3)") == "9\n"
assert node2.query("SELECT f2(12,3)") == "9\n"
node1.query("DROP FUNCTION f2")
assert (
node1.query("SELECT create_query FROM system.functions WHERE name='f2'") == ""
)
assert_eq_with_retry(
node2, "SELECT create_query FROM system.functions WHERE name='f2'", ""
)
def test_replication_replace_by_another_node_after_creation():
node1.query("CREATE FUNCTION f2 AS (x, y) -> x - y")
assert_eq_with_retry(
node2,
"SELECT create_query FROM system.functions WHERE name='f2'",
"CREATE FUNCTION f2 AS (x, y) -> (x - y)\n",
)
node2.query("CREATE OR REPLACE FUNCTION f2 AS (x, y) -> x + y")
assert_eq_with_retry(
node1,
"SELECT create_query FROM system.functions WHERE name='f2'",
"CREATE FUNCTION f2 AS (x, y) -> (x + y)\n",
)
node1.query("DROP FUNCTION f2")
assert_eq_with_retry(
node1, "SELECT create_query FROM system.functions WHERE name='f2'", ""
)
assert_eq_with_retry(
node2, "SELECT create_query FROM system.functions WHERE name='f2'", ""
)
# UserDefinedSQLObjectsLoaderFromZooKeeper must be able to continue working after reloading ZooKeeper.
def test_reload_zookeeper():
node1.query("CREATE FUNCTION f1 AS (x, y) -> x + y")
assert_eq_with_retry(
node2, "SELECT name FROM system.functions WHERE name ='f1'", "f1\n"
)
# remove zoo2, zoo3 from configs
replace_zookeeper_config(
inspect.cleandoc(
"""
zoo1
2181
2000
"""
)
)
# config reloads, but can still work
node1.query("CREATE FUNCTION f2 AS (x, y) -> x - y")
assert_eq_with_retry(
node2,
"SELECT name FROM system.functions WHERE name IN ['f1', 'f2'] ORDER BY name",
TSV(["f1", "f2"]),
)
# stop all zookeepers, user-defined functions will be readonly
cluster.stop_zookeeper_nodes(["zoo1", "zoo2", "zoo3"])
assert node2.query(
"SELECT name FROM system.functions WHERE name IN ['f1', 'f2'] ORDER BY name"
) == TSV(["f1", "f2"])
assert "ZooKeeper" in node1.query_and_get_error(
"CREATE FUNCTION f3 AS (x, y) -> x * y"
)
# start zoo2, zoo3, user-defined functions will be readonly too, because it only connect to zoo1
cluster.start_zookeeper_nodes(["zoo2", "zoo3"])
wait_zookeeper_node_to_start(["zoo2", "zoo3"])
assert node2.query(
"SELECT name FROM system.functions WHERE name IN ['f1', 'f2', 'f3'] ORDER BY name"
) == TSV(["f1", "f2"])
assert "ZooKeeper" in node1.query_and_get_error(
"CREATE FUNCTION f3 AS (x, y) -> x * y"
)
# set config to zoo2, server will be normal
replace_zookeeper_config(
inspect.cleandoc(
"""
zoo2
2181
2000
"""
)
)
active_zk_connections = get_active_zk_connections()
assert (
active_zk_connections == "1"
), "Total connections to ZooKeeper not equal to 1, {}".format(active_zk_connections)
node1.query("CREATE FUNCTION f3 AS (x, y) -> x / y")
assert_eq_with_retry(
node2,
"SELECT name FROM system.functions WHERE name IN ['f1', 'f2', 'f3'] ORDER BY name",
TSV(["f1", "f2", "f3"]),
)
assert node2.query("SELECT f1(12, 3), f2(12, 3), f3(12, 3)") == TSV([[15, 9, 4]])
active_zk_connections = get_active_zk_connections()
assert (
active_zk_connections == "1"
), "Total connections to ZooKeeper not equal to 1, {}".format(active_zk_connections)
node1.query("DROP FUNCTION f1")
node1.query("DROP FUNCTION f2")
node1.query("DROP FUNCTION f3")
# switch to the original version of zookeeper config
cluster.start_zookeeper_nodes(["zoo1", "zoo2", "zoo3"])
revert_zookeeper_config()
# Start without ZooKeeper must be possible, user-defined functions will be loaded after connecting to ZooKeeper.
def test_start_without_zookeeper():
node2.stop_clickhouse()
node1.query("CREATE FUNCTION f1 AS (x, y) -> x + y")
cluster.stop_zookeeper_nodes(["zoo1", "zoo2", "zoo3"])
node2.start_clickhouse()
assert (
node2.query("SELECT create_query FROM system.functions WHERE name='f1'") == ""
)
cluster.start_zookeeper_nodes(["zoo1", "zoo2", "zoo3"])
wait_zookeeper_node_to_start(["zoo1", "zoo2", "zoo3"])
assert_eq_with_retry(
node2,
"SELECT create_query FROM system.functions WHERE name='f1'",
"CREATE FUNCTION f1 AS (x, y) -> (x + y)\n",
)
node1.query("DROP FUNCTION f1")