2023-02-06 15:51:22 +00:00
|
|
|
import inspect
|
2023-08-03 09:45:07 +00:00
|
|
|
from contextlib import nullcontext as does_not_raise
|
2024-10-02 15:53:16 +00:00
|
|
|
from os import path as p
|
2023-02-06 15:51:22 +00:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
2023-08-03 09:45:07 +00:00
|
|
|
from helpers.client import QueryRuntimeException
|
2024-09-27 10:19:39 +00:00
|
|
|
from helpers.cluster import ClickHouseCluster
|
2024-10-02 15:53:16 +00:00
|
|
|
from helpers.keeper_utils import (
|
|
|
|
get_active_zk_connections,
|
|
|
|
replace_zookeeper_config,
|
|
|
|
reset_zookeeper_config,
|
|
|
|
)
|
2024-09-27 10:19:39 +00:00
|
|
|
from helpers.test_tools import TSV, assert_eq_with_retry
|
2023-02-06 15:51:22 +00:00
|
|
|
|
2024-10-02 15:53:16 +00:00
|
|
|
default_zk_config = p.join(p.dirname(p.realpath(__file__)), "configs/zookeeper.xml")
|
2023-02-06 15:51:22 +00:00
|
|
|
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 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")
|
|
|
|
|
|
|
|
|
2023-08-03 09:45:07 +00:00
|
|
|
@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"""
|
|
|
|
<clickhouse>
|
|
|
|
<profiles>
|
|
|
|
<default>
|
|
|
|
<ignore_on_cluster_for_replicated_udf_queries>{ignore}</ignore_on_cluster_for_replicated_udf_queries>
|
|
|
|
</default>
|
|
|
|
</profiles>
|
|
|
|
</clickhouse>
|
|
|
|
"""
|
|
|
|
),
|
|
|
|
)
|
|
|
|
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")
|
|
|
|
|
|
|
|
|
2023-02-06 15:51:22 +00:00
|
|
|
def test_create_and_replace():
|
|
|
|
node1.query("CREATE FUNCTION f1 AS (x, y) -> x + y")
|
2023-03-23 00:48:28 +00:00
|
|
|
assert node1.query("SELECT f1(12, 3)") == "15\n"
|
2023-02-06 15:51:22 +00:00
|
|
|
|
2023-12-12 14:03:49 +00:00
|
|
|
expected_error = "User-defined object 'f1' already exists"
|
2023-02-06 15:51:22 +00:00
|
|
|
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")
|
|
|
|
|
2023-12-12 14:03:49 +00:00
|
|
|
expected_error = "User-defined object 'f1' doesn't exist"
|
2023-02-06 15:51:22 +00:00
|
|
|
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")
|
2024-08-02 09:13:41 +00:00
|
|
|
node1.query(
|
|
|
|
"CREATE FUNCTION f3 AS () -> (SELECT sum(s) FROM (SELECT 1 as s UNION ALL SELECT 1 as s))"
|
|
|
|
)
|
2023-02-06 15:51:22 +00:00
|
|
|
|
|
|
|
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"
|
|
|
|
|
2024-08-02 09:13:41 +00:00
|
|
|
assert node1.query("SELECT f3()") == "2\n"
|
|
|
|
assert node2.query("SELECT f3()") == "2\n"
|
|
|
|
|
2023-02-06 15:51:22 +00:00
|
|
|
node1.query("DROP FUNCTION f2")
|
2024-08-02 09:13:41 +00:00
|
|
|
node1.query("DROP FUNCTION f3")
|
2023-02-06 15:51:22 +00:00
|
|
|
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(
|
2024-10-02 15:53:16 +00:00
|
|
|
(node1, node2),
|
2023-02-06 15:51:22 +00:00
|
|
|
inspect.cleandoc(
|
|
|
|
"""
|
|
|
|
<clickhouse>
|
|
|
|
<zookeeper>
|
|
|
|
<node index="1">
|
|
|
|
<host>zoo1</host>
|
|
|
|
<port>2181</port>
|
|
|
|
</node>
|
|
|
|
<session_timeout_ms>2000</session_timeout_ms>
|
|
|
|
</zookeeper>
|
|
|
|
</clickhouse>
|
|
|
|
"""
|
2024-10-02 15:53:16 +00:00
|
|
|
),
|
2023-02-06 15:51:22 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# config reloads, but can still work
|
2024-08-02 09:13:41 +00:00
|
|
|
node1.query(
|
|
|
|
"CREATE FUNCTION f2 AS () -> (SELECT sum(s) FROM (SELECT 1 as s UNION ALL SELECT 1 as s))"
|
|
|
|
)
|
2023-02-06 15:51:22 +00:00
|
|
|
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"])
|
2024-10-02 15:09:11 +00:00
|
|
|
cluster.wait_zookeeper_nodes_to_start(["zoo2", "zoo3"])
|
2023-02-06 15:51:22 +00:00
|
|
|
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(
|
2024-10-02 15:53:16 +00:00
|
|
|
(node1, node2),
|
2023-02-06 15:51:22 +00:00
|
|
|
inspect.cleandoc(
|
|
|
|
"""
|
|
|
|
<clickhouse>
|
|
|
|
<zookeeper>
|
|
|
|
<node index="1">
|
|
|
|
<host>zoo2</host>
|
|
|
|
<port>2181</port>
|
|
|
|
</node>
|
|
|
|
<session_timeout_ms>2000</session_timeout_ms>
|
|
|
|
</zookeeper>
|
|
|
|
</clickhouse>
|
|
|
|
"""
|
2024-10-02 15:53:16 +00:00
|
|
|
),
|
2023-02-06 15:51:22 +00:00
|
|
|
)
|
|
|
|
|
2024-10-01 22:54:11 +00:00
|
|
|
active_zk_connections = get_active_zk_connections(node1)
|
2023-02-06 15:51:22 +00:00
|
|
|
assert (
|
2024-10-01 22:54:11 +00:00
|
|
|
len(active_zk_connections) == 1
|
2023-02-06 15:51:22 +00:00
|
|
|
), "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"]),
|
|
|
|
)
|
|
|
|
|
2024-08-02 09:13:41 +00:00
|
|
|
assert node2.query("SELECT f1(12, 3), f2(), f3(12, 3)") == TSV([[15, 2, 4]])
|
2023-02-06 15:51:22 +00:00
|
|
|
|
2024-10-01 22:54:11 +00:00
|
|
|
active_zk_connections = get_active_zk_connections(node1)
|
2023-02-06 15:51:22 +00:00
|
|
|
assert (
|
2024-10-01 22:54:11 +00:00
|
|
|
len(active_zk_connections) == 1
|
2023-02-06 15:51:22 +00:00
|
|
|
), "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"])
|
2024-10-02 15:53:16 +00:00
|
|
|
reset_zookeeper_config((node1, node2), default_zk_config)
|
2023-03-23 00:48:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
# 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"])
|
2024-10-02 15:09:11 +00:00
|
|
|
cluster.wait_zookeeper_nodes_to_start(["zoo1", "zoo2", "zoo3"])
|
2023-03-23 00:48:28 +00:00
|
|
|
|
|
|
|
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")
|
2024-08-02 09:13:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_server_restart():
|
|
|
|
node1.query(
|
|
|
|
"CREATE FUNCTION f1 AS () -> (SELECT sum(s) FROM (SELECT 1 as s UNION ALL SELECT 1 as s))"
|
|
|
|
)
|
|
|
|
assert node1.query("SELECT f1()") == "2\n"
|
|
|
|
node1.restart_clickhouse()
|
|
|
|
assert node1.query("SELECT f1()") == "2\n"
|
|
|
|
node1.query("DROP FUNCTION f1")
|