diff --git a/.github/ISSUE_TEMPLATE/10_question.md b/.github/ISSUE_TEMPLATE/10_question.md
deleted file mode 100644
index 08a05a844e0..00000000000
--- a/.github/ISSUE_TEMPLATE/10_question.md
+++ /dev/null
@@ -1,20 +0,0 @@
----
-name: Question
-about: Ask a question about ClickHouse
-title: ''
-labels: question
-assignees: ''
-
----
-
-> Make sure to check documentation https://clickhouse.com/docs/en/ first. If the question is concise and probably has a short answer, asking it in [community Slack](https://join.slack.com/t/clickhousedb/shared_invite/zt-1gh9ds7f4-PgDhJAaF8ad5RbWBAAjzFg) is probably the fastest way to find the answer. For more complicated questions, consider asking them on StackOverflow with "clickhouse" tag https://stackoverflow.com/questions/tagged/clickhouse
-
-> If you still prefer GitHub issues, remove all this text and ask your question here.
-
-**Company or project name**
-
-Put your company name or project description here
-
-**Question**
-
-Your question
diff --git a/.github/ISSUE_TEMPLATE/10_question.yaml b/.github/ISSUE_TEMPLATE/10_question.yaml
new file mode 100644
index 00000000000..39d4c27807a
--- /dev/null
+++ b/.github/ISSUE_TEMPLATE/10_question.yaml
@@ -0,0 +1,20 @@
+name: Question
+description: Ask a question about ClickHouse
+labels: ["question"]
+body:
+ - type: markdown
+ attributes:
+ value: |
+ > Make sure to check documentation https://clickhouse.com/docs/en/ first. If the question is concise and probably has a short answer, asking it in [community Slack](https://join.slack.com/t/clickhousedb/shared_invite/zt-1gh9ds7f4-PgDhJAaF8ad5RbWBAAjzFg) is probably the fastest way to find the answer. For more complicated questions, consider asking them on StackOverflow with "clickhouse" tag https://stackoverflow.com/questions/tagged/clickhouse
+ - type: textarea
+ attributes:
+ label: Company or project name
+ description: Put your company name or project description here.
+ validations:
+ required: false
+ - type: textarea
+ attributes:
+ label: Question
+ description: Please put your question here.
+ validations:
+ required: true
diff --git a/docker/test/stateless/run.sh b/docker/test/stateless/run.sh
index 4d2c2e6f466..f94621ba092 100755
--- a/docker/test/stateless/run.sh
+++ b/docker/test/stateless/run.sh
@@ -285,7 +285,7 @@ stop_logs_replication
# Try to get logs while server is running
failed_to_save_logs=0
-for table in query_log zookeeper_log trace_log transactions_info_log metric_log
+for table in query_log zookeeper_log trace_log transactions_info_log metric_log blob_storage_log
do
err=$(clickhouse-client -q "select * from system.$table into outfile '/test_output/$table.tsv.gz' format TSVWithNamesAndTypes")
echo "$err"
@@ -339,7 +339,7 @@ if [ $failed_to_save_logs -ne 0 ]; then
# directly
# - even though ci auto-compress some files (but not *.tsv) it does this only
# for files >64MB, we want this files to be compressed explicitly
- for table in query_log zookeeper_log trace_log transactions_info_log metric_log
+ for table in query_log zookeeper_log trace_log transactions_info_log metric_log blob_storage_log
do
clickhouse-local "$data_path_config" --only-system-tables --stacktrace -q "select * from system.$table format TSVWithNamesAndTypes" | zstd --threads=0 > /test_output/$table.tsv.zst ||:
if [[ -n "$USE_DATABASE_REPLICATED" ]] && [[ "$USE_DATABASE_REPLICATED" -eq 1 ]]; then
diff --git a/docker/test/upgrade/run.sh b/docker/test/upgrade/run.sh
index 1f2cc9903b2..a4c4c75e5b3 100644
--- a/docker/test/upgrade/run.sh
+++ b/docker/test/upgrade/run.sh
@@ -25,7 +25,8 @@ azurite-blob --blobHost 0.0.0.0 --blobPort 10000 --debug /azurite_log &
./setup_minio.sh stateless # to have a proper environment
echo "Get previous release tag"
-previous_release_tag=$(dpkg --info package_folder/clickhouse-client*.deb | grep "Version: " | awk '{print $2}' | cut -f1 -d'+' | get_previous_release_tag)
+# shellcheck disable=SC2016
+previous_release_tag=$(dpkg-deb --showformat='${Version}' --show package_folder/clickhouse-client*.deb | get_previous_release_tag)
echo $previous_release_tag
echo "Clone previous release repository"
diff --git a/docs/en/engines/table-engines/integrations/mongodb.md b/docs/en/engines/table-engines/integrations/mongodb.md
index f87e8da8b5b..5bb3bc752f5 100644
--- a/docs/en/engines/table-engines/integrations/mongodb.md
+++ b/docs/en/engines/table-engines/integrations/mongodb.md
@@ -34,10 +34,11 @@ CREATE TABLE [IF NOT EXISTS] [db.]table_name
- `options` — MongoDB connection string options (optional parameter).
:::tip
-If you are using the MongoDB Atlas cloud offering please add these options:
+If you are using the MongoDB Atlas cloud offering:
```
-'connectTimeoutMS=10000&ssl=true&authSource=admin'
+- connection url can be obtained from 'Atlas SQL' option
+- use options: 'connectTimeoutMS=10000&ssl=true&authSource=admin'
```
:::
diff --git a/docs/en/engines/table-engines/mergetree-family/mergetree.md b/docs/en/engines/table-engines/mergetree-family/mergetree.md
index 689c05a24af..803b753fe0a 100644
--- a/docs/en/engines/table-engines/mergetree-family/mergetree.md
+++ b/docs/en/engines/table-engines/mergetree-family/mergetree.md
@@ -39,8 +39,8 @@ If you need to update rows frequently, we recommend using the [`ReplacingMergeTr
``` sql
CREATE TABLE [IF NOT EXISTS] [db.]table_name [ON CLUSTER cluster]
(
- name1 [type1] [[NOT] NULL] [DEFAULT|MATERIALIZED|ALIAS|EPHEMERAL expr1] [COMMENT ...] [CODEC(codec1)] [STATISTIC(stat1)] [TTL expr1] [PRIMARY KEY] [SETTINGS (name = value, ...)],
- name2 [type2] [[NOT] NULL] [DEFAULT|MATERIALIZED|ALIAS|EPHEMERAL expr2] [COMMENT ...] [CODEC(codec2)] [STATISTIC(stat2)] [TTL expr2] [PRIMARY KEY] [SETTINGS (name = value, ...)],
+ name1 [type1] [[NOT] NULL] [DEFAULT|MATERIALIZED|ALIAS|EPHEMERAL expr1] [COMMENT ...] [CODEC(codec1)] [STATISTICS(stat1)] [TTL expr1] [PRIMARY KEY] [SETTINGS (name = value, ...)],
+ name2 [type2] [[NOT] NULL] [DEFAULT|MATERIALIZED|ALIAS|EPHEMERAL expr2] [COMMENT ...] [CODEC(codec2)] [STATISTICS(stat2)] [TTL expr2] [PRIMARY KEY] [SETTINGS (name = value, ...)],
...
INDEX index_name1 expr1 TYPE type1(...) [GRANULARITY value1],
INDEX index_name2 expr2 TYPE type2(...) [GRANULARITY value2],
@@ -1043,12 +1043,12 @@ ClickHouse versions 22.3 through 22.7 use a different cache configuration, see [
## Column Statistics (Experimental) {#column-statistics}
-The statistic declaration is in the columns section of the `CREATE` query for tables from the `*MergeTree*` Family when we enable `set allow_experimental_statistic = 1`.
+The statistics declaration is in the columns section of the `CREATE` query for tables from the `*MergeTree*` Family when we enable `set allow_experimental_statistics = 1`.
``` sql
CREATE TABLE tab
(
- a Int64 STATISTIC(tdigest),
+ a Int64 STATISTICS(TDigest, Uniq),
b Float64
)
ENGINE = MergeTree
@@ -1058,19 +1058,23 @@ ORDER BY a
We can also manipulate statistics with `ALTER` statements.
```sql
-ALTER TABLE tab ADD STATISTIC b TYPE tdigest;
-ALTER TABLE tab DROP STATISTIC a TYPE tdigest;
+ALTER TABLE tab ADD STATISTICS b TYPE TDigest, Uniq;
+ALTER TABLE tab DROP STATISTICS a;
```
-These lightweight statistics aggregate information about distribution of values in columns.
-They can be used for query optimization when we enable `set allow_statistic_optimize = 1`.
+These lightweight statistics aggregate information about distribution of values in columns. Statistics are stored in every part and updated when every insert comes.
+They can be used for prewhere optimization only if we enable `set allow_statistics_optimize = 1`.
#### Available Types of Column Statistics {#available-types-of-column-statistics}
-- `tdigest`
+- `TDigest`
Stores distribution of values from numeric columns in [TDigest](https://github.com/tdunning/t-digest) sketch.
+- `Uniq`
+
+ Estimate the number of distinct values of a column by HyperLogLog.
+
## Column-level Settings {#column-level-settings}
Certain MergeTree settings can be override at column level:
diff --git a/docs/en/operations/server-configuration-parameters/settings.md b/docs/en/operations/server-configuration-parameters/settings.md
index a5fe74fd0c6..ee3ec5126a2 100644
--- a/docs/en/operations/server-configuration-parameters/settings.md
+++ b/docs/en/operations/server-configuration-parameters/settings.md
@@ -1206,6 +1206,16 @@ Expired time for HSTS in seconds. The default value is 0 means clickhouse disabl
600000
```
+## mlock_executable {#mlock_executable}
+
+Perform mlockall after startup to lower first queries latency and to prevent clickhouse executable from being paged out under high IO load. Enabling this option is recommended but will lead to increased startup time for up to a few seconds.
+Keep in mind that this parameter would not work without "CAP_IPC_LOCK" capability.
+**Example**
+
+``` xml
+false
+```
+
## include_from {#include_from}
The path to the file with substitutions. Both XML and YAML formats are supported.
@@ -1353,6 +1363,26 @@ Examples:
127.0.0.1
```
+## listen_try {#listen_try}
+
+The server will not exit if IPv6 or IPv4 networks are unavailable while trying to listen.
+
+Examples:
+
+``` xml
+0
+```
+
+## listen_reuse_port {#listen_reuse_port}
+
+Allow multiple servers to listen on the same address:port. Requests will be routed to a random server by the operating system. Enabling this setting is not recommended.
+
+Examples:
+
+``` xml
+0
+```
+
## listen_backlog {#listen_backlog}
Backlog (queue size of pending connections) of the listen socket.
diff --git a/docs/en/operations/settings/settings.md b/docs/en/operations/settings/settings.md
index 0b905df21d4..ffaf53085c4 100644
--- a/docs/en/operations/settings/settings.md
+++ b/docs/en/operations/settings/settings.md
@@ -3170,6 +3170,18 @@ Possible values:
Default value: `0`.
+## lightweight_deletes_sync {#lightweight_deletes_sync}
+
+The same as 'mutation_sync', but controls only execution of lightweight deletes.
+
+Possible values:
+
+- 0 - Mutations execute asynchronously.
+- 1 - The query waits for the lightweight deletes to complete on the current server.
+- 2 - The query waits for the lightweight deletes to complete on all replicas (if they exist).
+
+Default value: `2`.
+
**See Also**
- [Synchronicity of ALTER Queries](../../sql-reference/statements/alter/index.md#synchronicity-of-alter-queries)
@@ -5108,7 +5120,7 @@ a Tuple(
)
```
-## allow_experimental_statistic {#allow_experimental_statistic}
+## allow_experimental_statistics {#allow_experimental_statistics}
Allows defining columns with [statistics](../../engines/table-engines/mergetree-family/mergetree.md#table_engine-mergetree-creating-a-table) and [manipulate statistics](../../engines/table-engines/mergetree-family/mergetree.md#column-statistics).
diff --git a/docs/en/sql-reference/aggregate-functions/reference/largestTriangleThreeBuckets.md b/docs/en/sql-reference/aggregate-functions/reference/largestTriangleThreeBuckets.md
index 06443994dd9..4f73aadb8da 100644
--- a/docs/en/sql-reference/aggregate-functions/reference/largestTriangleThreeBuckets.md
+++ b/docs/en/sql-reference/aggregate-functions/reference/largestTriangleThreeBuckets.md
@@ -24,6 +24,8 @@ Alias: `lttb`.
- `x` — x coordinate. [Integer](../../../sql-reference/data-types/int-uint.md) , [Float](../../../sql-reference/data-types/float.md) , [Decimal](../../../sql-reference/data-types/decimal.md) , [Date](../../../sql-reference/data-types/date.md), [Date32](../../../sql-reference/data-types/date32.md), [DateTime](../../../sql-reference/data-types/datetime.md), [DateTime64](../../../sql-reference/data-types/datetime64.md).
- `y` — y coordinate. [Integer](../../../sql-reference/data-types/int-uint.md) , [Float](../../../sql-reference/data-types/float.md) , [Decimal](../../../sql-reference/data-types/decimal.md) , [Date](../../../sql-reference/data-types/date.md), [Date32](../../../sql-reference/data-types/date32.md), [DateTime](../../../sql-reference/data-types/datetime.md), [DateTime64](../../../sql-reference/data-types/datetime64.md).
+NaNs are ignored in the provided series, meaning that any NaN values will be excluded from the analysis. This ensures that the function operates only on valid numerical data.
+
**Parameters**
- `n` — number of points in the resulting series. [UInt64](../../../sql-reference/data-types/int-uint.md).
@@ -61,7 +63,7 @@ Result:
``` text
┌────────largestTriangleThreeBuckets(4)(x, y)───────────┐
-│ [(1,10),(3,15),(5,40),(10,70)] │
+│ [(1,10),(3,15),(9,55),(10,70)] │
└───────────────────────────────────────────────────────┘
```
diff --git a/docs/en/sql-reference/functions/type-conversion-functions.md b/docs/en/sql-reference/functions/type-conversion-functions.md
index 5dd1d5ceebe..2ec51d43c59 100644
--- a/docs/en/sql-reference/functions/type-conversion-functions.md
+++ b/docs/en/sql-reference/functions/type-conversion-functions.md
@@ -2423,11 +2423,7 @@ Result:
## toUnixTimestamp64Milli
-## toUnixTimestamp64Micro
-
-## toUnixTimestamp64Nano
-
-Converts a `DateTime64` to a `Int64` value with fixed sub-second precision. Input value is scaled up or down appropriately depending on it precision.
+Converts a `DateTime64` to a `Int64` value with fixed millisecond precision. The input value is scaled up or down appropriately depending on its precision.
:::note
The output value is a timestamp in UTC, not in the timezone of `DateTime64`.
@@ -2437,24 +2433,22 @@ The output value is a timestamp in UTC, not in the timezone of `DateTime64`.
```sql
toUnixTimestamp64Milli(value)
-toUnixTimestamp64Micro(value)
-toUnixTimestamp64Nano(value)
```
**Arguments**
-- `value` — DateTime64 value with any precision.
+- `value` — DateTime64 value with any precision. [DateTime64](../data-types/datetime64.md).
**Returned value**
-- `value` converted to the `Int64` data type.
+- `value` converted to the `Int64` data type. [Int64](../data-types/int-uint.md).
-**Examples**
+**Example**
Query:
```sql
-WITH toDateTime64('2019-09-16 19:20:12.345678910', 6) AS dt64
+WITH toDateTime64('2009-02-13 23:31:31.011', 3, 'UTC') AS dt64
SELECT toUnixTimestamp64Milli(dt64);
```
@@ -2462,14 +2456,77 @@ Result:
```response
┌─toUnixTimestamp64Milli(dt64)─┐
-│ 1568650812345 │
+│ 1234567891011 │
└──────────────────────────────┘
```
+## toUnixTimestamp64Micro
+
+Converts a `DateTime64` to a `Int64` value with fixed microsecond precision. The input value is scaled up or down appropriately depending on its precision.
+
+:::note
+The output value is a timestamp in UTC, not in the timezone of `DateTime64`.
+:::
+
+**Syntax**
+
+```sql
+toUnixTimestamp64Micro(value)
+```
+
+**Arguments**
+
+- `value` — DateTime64 value with any precision. [DateTime64](../data-types/datetime64.md).
+
+**Returned value**
+
+- `value` converted to the `Int64` data type. [Int64](../data-types/int-uint.md).
+
+**Example**
+
Query:
-``` sql
-WITH toDateTime64('2019-09-16 19:20:12.345678910', 6) AS dt64
+```sql
+WITH toDateTime64('1970-01-15 06:56:07.891011', 6, 'UTC') AS dt64
+SELECT toUnixTimestamp64Micro(dt64);
+```
+
+Result:
+
+```response
+┌─toUnixTimestamp64Micro(dt64)─┐
+│ 1234567891011 │
+└──────────────────────────────┘
+```
+
+## toUnixTimestamp64Nano
+
+Converts a `DateTime64` to a `Int64` value with fixed nanosecond precision. The input value is scaled up or down appropriately depending on its precision.
+
+:::note
+The output value is a timestamp in UTC, not in the timezone of `DateTime64`.
+:::
+
+**Syntax**
+
+```sql
+toUnixTimestamp64Nano(value)
+```
+
+**Arguments**
+
+- `value` — DateTime64 value with any precision. [DateTime64](../data-types/datetime64.md).
+
+**Returned value**
+
+- `value` converted to the `Int64` data type. [Int64](../data-types/int-uint.md).
+
+**Example**
+
+Query:
+
+```sql
+WITH toDateTime64('1970-01-01 00:20:34.567891011', 9, 'UTC') AS dt64
SELECT toUnixTimestamp64Nano(dt64);
```
@@ -2477,34 +2534,32 @@ Result:
```response
┌─toUnixTimestamp64Nano(dt64)─┐
-│ 1568650812345678000 │
+│ 1234567891011 │
└─────────────────────────────┘
```
## fromUnixTimestamp64Milli
-## fromUnixTimestamp64Micro
+Converts an `Int64` to a `DateTime64` value with fixed millisecond precision and optional timezone. The input value is scaled up or down appropriately depending on its precision.
-## fromUnixTimestamp64Nano
-
-Converts an `Int64` to a `DateTime64` value with fixed sub-second precision and optional timezone. Input value is scaled up or down appropriately depending on it’s precision. Please note that input value is treated as UTC timestamp, not timestamp at given (or implicit) timezone.
+:::note
+Please note that input value is treated as a UTC timestamp, not timestamp at the given (or implicit) timezone.
+:::
**Syntax**
``` sql
fromUnixTimestamp64Milli(value[, timezone])
-fromUnixTimestamp64Micro(value[, timezone])
-fromUnixTimestamp64Nano(value[, timezone])
```
**Arguments**
-- `value` — `Int64` value with any precision.
-- `timezone` — `String` (optional) timezone name of the result.
+- `value` — value with any precision. [Int64](../data-types/int-uint.md).
+- `timezone` — (optional) timezone name of the result. [String](../data-types/string.md).
**Returned value**
-- `value` converted to the `DateTime64` data type.
+- `value` converted to DateTime64 with precision `3`. [DateTime64](../data-types/datetime64.md).
**Example**
@@ -2512,15 +2567,101 @@ Query:
``` sql
WITH CAST(1234567891011, 'Int64') AS i64
-SELECT fromUnixTimestamp64Milli(i64, 'UTC');
+SELECT
+ fromUnixTimestamp64Milli(i64, 'UTC') AS x,
+ toTypeName(x);
```
Result:
```response
-┌─fromUnixTimestamp64Milli(i64, 'UTC')─┐
-│ 2009-02-13 23:31:31.011 │
-└──────────────────────────────────────┘
+┌───────────────────────x─┬─toTypeName(x)────────┐
+│ 2009-02-13 23:31:31.011 │ DateTime64(3, 'UTC') │
+└─────────────────────────┴──────────────────────┘
+```
+
+## fromUnixTimestamp64Micro
+
+Converts an `Int64` to a `DateTime64` value with fixed microsecond precision and optional timezone. The input value is scaled up or down appropriately depending on its precision.
+
+:::note
+Please note that input value is treated as a UTC timestamp, not timestamp at the given (or implicit) timezone.
+:::
+
+**Syntax**
+
+``` sql
+fromUnixTimestamp64Micro(value[, timezone])
+```
+
+**Arguments**
+
+- `value` — value with any precision. [Int64](../data-types/int-uint.md).
+- `timezone` — (optional) timezone name of the result. [String](../data-types/string.md).
+
+**Returned value**
+
+- `value` converted to DateTime64 with precision `6`. [DateTime64](../data-types/datetime64.md).
+
+**Example**
+
+Query:
+
+``` sql
+WITH CAST(1234567891011, 'Int64') AS i64
+SELECT
+ fromUnixTimestamp64Micro(i64, 'UTC') AS x,
+ toTypeName(x);
+```
+
+Result:
+
+```response
+┌──────────────────────────x─┬─toTypeName(x)────────┐
+│ 1970-01-15 06:56:07.891011 │ DateTime64(6, 'UTC') │
+└────────────────────────────┴──────────────────────┘
+```
+
+## fromUnixTimestamp64Nano
+
+Converts an `Int64` to a `DateTime64` value with fixed nanosecond precision and optional timezone. The input value is scaled up or down appropriately depending on its precision.
+
+:::note
+Please note that input value is treated as a UTC timestamp, not timestamp at the given (or implicit) timezone.
+:::
+
+**Syntax**
+
+``` sql
+fromUnixTimestamp64Nano(value[, timezone])
+```
+
+**Arguments**
+
+- `value` — value with any precision. [Int64](../data-types/int-uint.md).
+- `timezone` — (optional) timezone name of the result. [String](../data-types/string.md).
+
+**Returned value**
+
+- `value` converted to DateTime64 with precision `9`. [DateTime64](../data-types/datetime64.md).
+
+**Example**
+
+Query:
+
+``` sql
+WITH CAST(1234567891011, 'Int64') AS i64
+SELECT
+ fromUnixTimestamp64Nano(i64, 'UTC') AS x,
+ toTypeName(x);
+```
+
+Result:
+
+```response
+┌─────────────────────────────x─┬─toTypeName(x)────────┐
+│ 1970-01-01 00:20:34.567891011 │ DateTime64(9, 'UTC') │
+└───────────────────────────────┴──────────────────────┘
```
## formatRow
diff --git a/docs/en/sql-reference/statements/alter/index.md b/docs/en/sql-reference/statements/alter/index.md
index 3cfb99cff83..f81d4f02e0c 100644
--- a/docs/en/sql-reference/statements/alter/index.md
+++ b/docs/en/sql-reference/statements/alter/index.md
@@ -16,7 +16,7 @@ Most `ALTER TABLE` queries modify table settings or data:
- [INDEX](/docs/en/sql-reference/statements/alter/skipping-index.md)
- [CONSTRAINT](/docs/en/sql-reference/statements/alter/constraint.md)
- [TTL](/docs/en/sql-reference/statements/alter/ttl.md)
-- [STATISTIC](/docs/en/sql-reference/statements/alter/statistic.md)
+- [STATISTICS](/docs/en/sql-reference/statements/alter/statistics.md)
- [APPLY DELETED MASK](/docs/en/sql-reference/statements/alter/apply-deleted-mask.md)
:::note
diff --git a/docs/en/sql-reference/statements/alter/partition.md b/docs/en/sql-reference/statements/alter/partition.md
index ce5cecf6fd6..0ed1e523669 100644
--- a/docs/en/sql-reference/statements/alter/partition.md
+++ b/docs/en/sql-reference/statements/alter/partition.md
@@ -139,7 +139,7 @@ For the query to run successfully, the following conditions must be met:
ALTER TABLE table2 [ON CLUSTER cluster] REPLACE PARTITION partition_expr FROM table1
```
-This query copies the data partition from the `table1` to `table2` and replaces existing partition in the `table2`.
+This query copies the data partition from `table1` to `table2` and replaces the existing partition in `table2`. The operation is atomic.
Note that:
diff --git a/docs/en/sql-reference/statements/alter/statistic.md b/docs/en/sql-reference/statements/alter/statistic.md
deleted file mode 100644
index 1c2e45b23fd..00000000000
--- a/docs/en/sql-reference/statements/alter/statistic.md
+++ /dev/null
@@ -1,25 +0,0 @@
----
-slug: /en/sql-reference/statements/alter/statistic
-sidebar_position: 45
-sidebar_label: STATISTIC
----
-
-# Manipulating Column Statistics
-
-The following operations are available:
-
-- `ALTER TABLE [db].table ADD STATISTIC (columns list) TYPE type` - Adds statistic description to tables metadata.
-
-- `ALTER TABLE [db].table DROP STATISTIC (columns list) TYPE type` - Removes statistic description from tables metadata and deletes statistic files from disk.
-
-- `ALTER TABLE [db].table CLEAR STATISTIC (columns list) TYPE type` - Deletes statistic files from disk.
-
-- `ALTER TABLE [db.]table MATERIALIZE STATISTIC (columns list) TYPE type` - Rebuilds the statistic for columns. Implemented as a [mutation](../../../sql-reference/statements/alter/index.md#mutations).
-
-The first two commands are lightweight in a sense that they only change metadata or remove files.
-
-Also, they are replicated, syncing statistics metadata via ZooKeeper.
-
-:::note
-Statistic manipulation is supported only for tables with [`*MergeTree`](../../../engines/table-engines/mergetree-family/mergetree.md) engine (including [replicated](../../../engines/table-engines/mergetree-family/replication.md) variants).
-:::
diff --git a/docs/en/sql-reference/statements/alter/statistics.md b/docs/en/sql-reference/statements/alter/statistics.md
new file mode 100644
index 00000000000..80024781f88
--- /dev/null
+++ b/docs/en/sql-reference/statements/alter/statistics.md
@@ -0,0 +1,33 @@
+---
+slug: /en/sql-reference/statements/alter/statistics
+sidebar_position: 45
+sidebar_label: STATISTICS
+---
+
+# Manipulating Column Statistics
+
+The following operations are available:
+
+- `ALTER TABLE [db].table ADD STATISTICS (columns list) TYPE (type list)` - Adds statistic description to tables metadata.
+
+- `ALTER TABLE [db].table MODIFY STATISTICS (columns list) TYPE (type list)` - Modifies statistic description to tables metadata.
+
+- `ALTER TABLE [db].table DROP STATISTICS (columns list)` - Removes statistics from the metadata of the specified columns and deletes all statistics objects in all parts for the specified columns.
+
+- `ALTER TABLE [db].table CLEAR STATISTICS (columns list)` - Deletes all statistics objects in all parts for the specified columns. Statistics objects can be rebuild using `ALTER TABLE MATERIALIZE STATISTICS`.
+
+- `ALTER TABLE [db.]table MATERIALIZE STATISTICS (columns list)` - Rebuilds the statistic for columns. Implemented as a [mutation](../../../sql-reference/statements/alter/index.md#mutations).
+
+The first two commands are lightweight in a sense that they only change metadata or remove files.
+
+Also, they are replicated, syncing statistics metadata via ZooKeeper.
+
+There is an example adding two statistics types to two columns:
+
+```
+ALTER TABLE t1 MODIFY STATISTICS c, d TYPE TDigest, Uniq;
+```
+
+:::note
+Statistic manipulation is supported only for tables with [`*MergeTree`](../../../engines/table-engines/mergetree-family/mergetree.md) engine (including [replicated](../../../engines/table-engines/mergetree-family/replication.md) variants).
+:::
diff --git a/src/Access/Common/AccessType.h b/src/Access/Common/AccessType.h
index 7f0eff2184b..e9f24a8c685 100644
--- a/src/Access/Common/AccessType.h
+++ b/src/Access/Common/AccessType.h
@@ -51,10 +51,11 @@ enum class AccessType : uint8_t
M(ALTER_CLEAR_INDEX, "CLEAR INDEX", TABLE, ALTER_INDEX) \
M(ALTER_INDEX, "INDEX", GROUP, ALTER_TABLE) /* allows to execute ALTER ORDER BY or ALTER {ADD|DROP...} INDEX */\
\
- M(ALTER_ADD_STATISTIC, "ALTER ADD STATISTIC", TABLE, ALTER_STATISTIC) \
- M(ALTER_DROP_STATISTIC, "ALTER DROP STATISTIC", TABLE, ALTER_STATISTIC) \
- M(ALTER_MATERIALIZE_STATISTIC, "ALTER MATERIALIZE STATISTIC", TABLE, ALTER_STATISTIC) \
- M(ALTER_STATISTIC, "STATISTIC", GROUP, ALTER_TABLE) /* allows to execute ALTER STATISTIC */\
+ M(ALTER_ADD_STATISTICS, "ALTER ADD STATISTIC", TABLE, ALTER_STATISTICS) \
+ M(ALTER_DROP_STATISTICS, "ALTER DROP STATISTIC", TABLE, ALTER_STATISTICS) \
+ M(ALTER_MODIFY_STATISTICS, "ALTER MODIFY STATISTIC", TABLE, ALTER_STATISTICS) \
+ M(ALTER_MATERIALIZE_STATISTICS, "ALTER MATERIALIZE STATISTIC", TABLE, ALTER_STATISTICS) \
+ M(ALTER_STATISTICS, "STATISTIC", GROUP, ALTER_TABLE) /* allows to execute ALTER STATISTIC */\
\
M(ALTER_ADD_PROJECTION, "ADD PROJECTION", TABLE, ALTER_PROJECTION) \
M(ALTER_DROP_PROJECTION, "DROP PROJECTION", TABLE, ALTER_PROJECTION) \
diff --git a/src/AggregateFunctions/QuantileTDigest.h b/src/AggregateFunctions/QuantileTDigest.h
index d5a4f6b576a..4207ea587b1 100644
--- a/src/AggregateFunctions/QuantileTDigest.h
+++ b/src/AggregateFunctions/QuantileTDigest.h
@@ -334,6 +334,18 @@ public:
compress(); // Allows reading/writing TDigests with different epsilon/max_centroids params
}
+ Float64 getCountEqual(Float64 value) const
+ {
+ Float64 result = 0;
+ for (const auto & c : centroids)
+ {
+ /// std::cerr << "c "<< c.mean << " "<< c.count << std::endl;
+ if (value == c.mean)
+ result += c.count;
+ }
+ return result;
+ }
+
Float64 getCountLessThan(Float64 value) const
{
bool first = true;
diff --git a/src/Common/ErrorCodes.cpp b/src/Common/ErrorCodes.cpp
index ea6f9510927..29993ed33e4 100644
--- a/src/Common/ErrorCodes.cpp
+++ b/src/Common/ErrorCodes.cpp
@@ -586,7 +586,7 @@
M(705, TABLE_NOT_EMPTY) \
M(706, LIBSSH_ERROR) \
M(707, GCP_ERROR) \
- M(708, ILLEGAL_STATISTIC) \
+ M(708, ILLEGAL_STATISTICS) \
M(709, CANNOT_GET_REPLICATED_DATABASE_SNAPSHOT) \
M(710, FAULT_INJECTED) \
M(711, FILECACHE_ACCESS_DENIED) \
diff --git a/src/Common/MemoryTrackerSwitcher.h b/src/Common/MemoryTrackerSwitcher.h
index 3c99fd12353..796b5295a83 100644
--- a/src/Common/MemoryTrackerSwitcher.h
+++ b/src/Common/MemoryTrackerSwitcher.h
@@ -15,6 +15,7 @@ struct MemoryTrackerSwitcher
return;
auto * thread_tracker = CurrentThread::getMemoryTracker();
+
prev_untracked_memory = current_thread->untracked_memory;
prev_memory_tracker_parent = thread_tracker->getParent();
@@ -31,8 +32,10 @@ struct MemoryTrackerSwitcher
CurrentThread::flushUntrackedMemory();
auto * thread_tracker = CurrentThread::getMemoryTracker();
- current_thread->untracked_memory = prev_untracked_memory;
+ /// It is important to set untracked memory after the call of
+ /// 'setParent' because it may flush untracked memory to the wrong parent.
thread_tracker->setParent(prev_memory_tracker_parent);
+ current_thread->untracked_memory = prev_untracked_memory;
}
private:
diff --git a/src/Core/Settings.h b/src/Core/Settings.h
index f4de7ee86fb..27ce54c03a7 100644
--- a/src/Core/Settings.h
+++ b/src/Core/Settings.h
@@ -160,8 +160,8 @@ class IColumn;
M(Bool, enable_multiple_prewhere_read_steps, true, "Move more conditions from WHERE to PREWHERE and do reads from disk and filtering in multiple steps if there are multiple conditions combined with AND", 0) \
M(Bool, move_primary_key_columns_to_end_of_prewhere, true, "Move PREWHERE conditions containing primary key columns to the end of AND chain. It is likely that these conditions are taken into account during primary key analysis and thus will not contribute a lot to PREWHERE filtering.", 0) \
\
- M(Bool, allow_statistic_optimize, false, "Allows using statistic to optimize queries", 0) \
- M(Bool, allow_experimental_statistic, false, "Allows using statistic", 0) \
+ M(Bool, allow_statistics_optimize, false, "Allows using statistics to optimize queries", 0) ALIAS(allow_statistic_optimize) \
+ M(Bool, allow_experimental_statistics, false, "Allows using statistics", 0) ALIAS(allow_experimental_statistic) \
\
M(UInt64, alter_sync, 1, "Wait for actions to manipulate the partitions. 0 - do not wait, 1 - wait for execution only of itself, 2 - wait for everyone.", 0) ALIAS(replication_alter_partitions_sync) \
M(Int64, replication_wait_for_inactive_replica_timeout, 120, "Wait for inactive replica to execute ALTER/OPTIMIZE. Time in seconds, 0 - do not wait, negative - wait for unlimited time.", 0) \
@@ -891,6 +891,7 @@ class IColumn;
M(Bool, geo_distance_returns_float64_on_float64_arguments, true, "If all four arguments to `geoDistance`, `greatCircleDistance`, `greatCircleAngle` functions are Float64, return Float64 and use double precision for internal calculations. In previous ClickHouse versions, the functions always returned Float32.", 0) \
M(Bool, allow_get_client_http_header, false, "Allow to use the function `getClientHTTPHeader` which lets to obtain a value of an the current HTTP request's header. It is not enabled by default for security reasons, because some headers, such as `Cookie`, could contain sensitive info. Note that the `X-ClickHouse-*` and `Authentication` headers are always restricted and cannot be obtained with this function.", 0) \
M(Bool, cast_string_to_dynamic_use_inference, false, "Use types inference during String to Dynamic conversion", 0) \
+ M(Bool, enable_blob_storage_log, true, "Write information about blob storage operations to system.blob_storage_log table", 0) \
\
/** Experimental functions */ \
M(Bool, allow_experimental_materialized_postgresql_table, false, "Allows to use the MaterializedPostgreSQL table engine. Disabled by default, because this feature is experimental", 0) \
diff --git a/src/Core/SettingsChangesHistory.h b/src/Core/SettingsChangesHistory.h
index 3ddacd06a06..814c08c5705 100644
--- a/src/Core/SettingsChangesHistory.h
+++ b/src/Core/SettingsChangesHistory.h
@@ -83,7 +83,7 @@ namespace SettingsChangesHistory
/// For newly added setting choose the most appropriate previous_value (for example, if new setting
/// controls new feature and it's 'true' by default, use 'false' as previous_value).
/// It's used to implement `compatibility` setting (see https://github.com/ClickHouse/ClickHouse/issues/35972)
-static std::map settings_changes_history =
+static const std::map settings_changes_history =
{
{"24.6", {{"materialize_skip_indexes_on_insert", true, true, "Added new setting to allow to disable materialization of skip indexes on insert"},
{"materialize_statistics_on_insert", true, true, "Added new setting to allow to disable materialization of statistics on insert"},
@@ -96,6 +96,11 @@ static std::map sett
{"hdfs_ignore_file_doesnt_exist", false, false, "Allow to return 0 rows when the requested files don't exist instead of throwing an exception in HDFS table engine"},
{"azure_ignore_file_doesnt_exist", false, false, "Allow to return 0 rows when the requested files don't exist instead of throwing an exception in AzureBlobStorage table engine"},
{"s3_ignore_file_doesnt_exist", false, false, "Allow to return 0 rows when the requested files don't exist instead of throwing an exception in S3 table engine"},
+ {"enable_blob_storage_log", true, true, "Write information about blob storage operations to system.blob_storage_log table"},
+ {"allow_statistic_optimize", false, false, "Old setting which popped up here being renamed."},
+ {"allow_experimental_statistic", false, false, "Old setting which popped up here being renamed."},
+ {"allow_statistics_optimize", false, false, "The setting was renamed. The previous name is `allow_statistic_optimize`."},
+ {"allow_experimental_statistics", false, false, "The setting was renamed. The previous name is `allow_experimental_statistic`."}
}},
{"24.5", {{"allow_deprecated_error_prone_window_functions", true, false, "Allow usage of deprecated error prone window functions (neighbor, runningAccumulate, runningDifferenceStartingWithFirstValue, runningDifference)"},
{"allow_experimental_join_condition", false, false, "Support join with inequal conditions which involve columns from both left and right table. e.g. t1.y < t2.y."},
diff --git a/src/DataTypes/IDataType.h b/src/DataTypes/IDataType.h
index 46c30240ef8..85fce671cbb 100644
--- a/src/DataTypes/IDataType.h
+++ b/src/DataTypes/IDataType.h
@@ -543,6 +543,7 @@ template constexpr bool IsDataTypeNumber = false;
template constexpr bool IsDataTypeDateOrDateTime = false;
template constexpr bool IsDataTypeDate = false;
template constexpr bool IsDataTypeEnum = false;
+template constexpr bool IsDataTypeStringOrFixedString = false;
template constexpr bool IsDataTypeDecimalOrNumber = IsDataTypeDecimal || IsDataTypeNumber;
@@ -556,6 +557,8 @@ class DataTypeDate;
class DataTypeDate32;
class DataTypeDateTime;
class DataTypeDateTime64;
+class DataTypeString;
+class DataTypeFixedString;
template constexpr bool IsDataTypeDecimal> = true;
@@ -572,6 +575,9 @@ template <> inline constexpr bool IsDataTypeDateOrDateTime = tru
template <> inline constexpr bool IsDataTypeDateOrDateTime = true;
template <> inline constexpr bool IsDataTypeDateOrDateTime = true;
+template <> inline constexpr bool IsDataTypeStringOrFixedString = true;
+template <> inline constexpr bool IsDataTypeStringOrFixedString = true;
+
template
class DataTypeEnum;
diff --git a/src/Databases/DatabaseOnDisk.cpp b/src/Databases/DatabaseOnDisk.cpp
index 5cb4198e1a2..025ac585593 100644
--- a/src/Databases/DatabaseOnDisk.cpp
+++ b/src/Databases/DatabaseOnDisk.cpp
@@ -670,7 +670,7 @@ void DatabaseOnDisk::iterateMetadataFiles(ContextPtr local_context, const Iterat
for (auto it = metadata_files.begin(); it < metadata_files.end(); std::advance(it, batch_size))
{
std::span batch{it, std::min(std::next(it, batch_size), metadata_files.end())};
- pool.scheduleOrThrowOnError(
+ pool.scheduleOrThrow(
[batch, &process_metadata_file, &process_tmp_drop_metadata_file]() mutable
{
setThreadName("DatabaseOnDisk");
@@ -679,7 +679,7 @@ void DatabaseOnDisk::iterateMetadataFiles(ContextPtr local_context, const Iterat
process_metadata_file(file.first);
else
process_tmp_drop_metadata_file(file.first);
- });
+ }, Priority{}, getContext()->getSettingsRef().lock_acquire_timeout.totalMicroseconds());
}
pool.wait();
}
diff --git a/src/Databases/DatabaseOrdinary.cpp b/src/Databases/DatabaseOrdinary.cpp
index 5d36f1cc3d6..10a8e06e8f0 100644
--- a/src/Databases/DatabaseOrdinary.cpp
+++ b/src/Databases/DatabaseOrdinary.cpp
@@ -44,6 +44,7 @@ namespace ErrorCodes
extern const int LOGICAL_ERROR;
extern const int UNKNOWN_DATABASE_ENGINE;
extern const int NOT_IMPLEMENTED;
+ extern const int UNEXPECTED_NODE_IN_ZOOKEEPER;
}
static constexpr size_t METADATA_FILE_BUFFER_SIZE = 32768;
@@ -76,6 +77,20 @@ static void setReplicatedEngine(ASTCreateQuery * create_query, ContextPtr contex
String replica_path = server_settings.default_replica_path;
String replica_name = server_settings.default_replica_name;
+ /// Check that replica path doesn't exist
+ Macros::MacroExpansionInfo info;
+ StorageID table_id = StorageID(create_query->getDatabase(), create_query->getTable(), create_query->uuid);
+ info.table_id = table_id;
+ info.expand_special_macros_only = false;
+
+ String zookeeper_path = context->getMacros()->expand(replica_path, info);
+ if (context->getZooKeeper()->exists(zookeeper_path))
+ throw Exception(
+ ErrorCodes::UNEXPECTED_NODE_IN_ZOOKEEPER,
+ "Found existing ZooKeeper path {} while trying to convert table {} to replicated. Table will not be converted.",
+ zookeeper_path, backQuote(table_id.getFullTableName())
+ );
+
auto args = std::make_shared();
args->children.push_back(std::make_shared(replica_path));
args->children.push_back(std::make_shared(replica_name));
diff --git a/src/Disks/ObjectStorages/ObjectStorageIteratorAsync.cpp b/src/Disks/ObjectStorages/ObjectStorageIteratorAsync.cpp
index 0420de0f8dd..a249789df4b 100644
--- a/src/Disks/ObjectStorages/ObjectStorageIteratorAsync.cpp
+++ b/src/Disks/ObjectStorages/ObjectStorageIteratorAsync.cpp
@@ -36,30 +36,24 @@ void IObjectStorageIteratorAsync::deactivate()
void IObjectStorageIteratorAsync::nextBatch()
{
std::lock_guard lock(mutex);
+
if (is_finished)
{
current_batch.clear();
current_batch_iterator = current_batch.begin();
+ return;
}
- else
- {
- if (!is_initialized)
- {
- outcome_future = scheduleBatch();
- is_initialized = true;
- }
+ if (!is_initialized)
+ {
+ outcome_future = scheduleBatch();
+ is_initialized = true;
+ }
+
+ try
+ {
chassert(outcome_future.valid());
- BatchAndHasNext result;
- try
- {
- result = outcome_future.get();
- }
- catch (...)
- {
- is_finished = true;
- throw;
- }
+ BatchAndHasNext result = outcome_future.get();
current_batch = std::move(result.batch);
current_batch_iterator = current_batch.begin();
@@ -71,6 +65,11 @@ void IObjectStorageIteratorAsync::nextBatch()
else
is_finished = true;
}
+ catch (...)
+ {
+ is_finished = true;
+ throw;
+ }
}
void IObjectStorageIteratorAsync::next()
@@ -95,35 +94,39 @@ std::future IObjectStorageIterator
bool IObjectStorageIteratorAsync::isValid()
{
+ std::lock_guard lock(mutex);
+
if (!is_initialized)
nextBatch();
- std::lock_guard lock(mutex);
return current_batch_iterator != current_batch.end();
}
RelativePathWithMetadataPtr IObjectStorageIteratorAsync::current()
{
+ std::lock_guard lock(mutex);
+
if (!isValid())
throw Exception(ErrorCodes::LOGICAL_ERROR, "Trying to access invalid iterator");
- std::lock_guard lock(mutex);
return *current_batch_iterator;
}
RelativePathsWithMetadata IObjectStorageIteratorAsync::currentBatch()
{
+ std::lock_guard lock(mutex);
+
if (!isValid())
throw Exception(ErrorCodes::LOGICAL_ERROR, "Trying to access invalid iterator");
- std::lock_guard lock(mutex);
return current_batch;
}
std::optional IObjectStorageIteratorAsync::getCurrentBatchAndScheduleNext()
{
std::lock_guard lock(mutex);
+
if (!is_initialized)
nextBatch();
diff --git a/src/Functions/FunctionsConversion.cpp b/src/Functions/FunctionsConversion.cpp
index 44d0b750af9..2a0b2f1d075 100644
--- a/src/Functions/FunctionsConversion.cpp
+++ b/src/Functions/FunctionsConversion.cpp
@@ -709,7 +709,7 @@ bool tryParseImpl(typename DataType::FieldType & x, ReadBuffer & rb, const DateL
else
return tryReadFloatTextFast(x, rb);
}
- else /*if constexpr (is_integer_v)*/
+ else /*if constexpr (is_integral_v)*/
return tryReadIntText(x, rb);
}
@@ -814,6 +814,16 @@ enum class ConvertFromStringParsingMode : uint8_t
BestEffortUS
};
+struct AccurateConvertStrategyAdditions
+{
+ UInt32 scale { 0 };
+};
+
+struct AccurateOrNullConvertStrategyAdditions
+{
+ UInt32 scale { 0 };
+};
+
template
struct ConvertThroughParsing
@@ -1020,7 +1030,13 @@ struct ConvertThroughParsing
break;
}
}
- parseImpl(vec_to[i], read_buffer, local_time_zone, precise_float_parsing);
+ if constexpr (std::is_same_v)
+ {
+ if (!tryParseImpl(vec_to[i], read_buffer, local_time_zone, precise_float_parsing))
+ throw Exception(ErrorCodes::CANNOT_PARSE_TEXT, "Cannot parse string to type {}", TypeName);
+ }
+ else
+ parseImpl(vec_to[i], read_buffer, local_time_zone, precise_float_parsing);
} while (false);
}
}
@@ -1120,16 +1136,6 @@ struct ConvertThroughParsing
/// Function toUnixTimestamp has exactly the same implementation as toDateTime of String type.
struct NameToUnixTimestamp { static constexpr auto name = "toUnixTimestamp"; };
-struct AccurateConvertStrategyAdditions
-{
- UInt32 scale { 0 };
-};
-
-struct AccurateOrNullConvertStrategyAdditions
-{
- UInt32 scale { 0 };
-};
-
enum class BehaviourOnErrorFromString : uint8_t
{
ConvertDefaultBehaviorTag,
@@ -3174,8 +3180,11 @@ private:
{
TypeIndex from_type_index = from_type->getTypeId();
WhichDataType which(from_type_index);
+ TypeIndex to_type_index = to_type->getTypeId();
+ WhichDataType to(to_type_index);
bool can_apply_accurate_cast = (cast_type == CastType::accurate || cast_type == CastType::accurateOrNull)
&& (which.isInt() || which.isUInt() || which.isFloat());
+ can_apply_accurate_cast |= cast_type == CastType::accurate && which.isStringOrFixedString() && to.isNativeInteger();
FormatSettings::DateTimeOverflowBehavior date_time_overflow_behavior = default_date_time_overflow_behavior;
if (context)
@@ -3260,6 +3269,20 @@ private:
return true;
}
}
+ else if constexpr (IsDataTypeStringOrFixedString)
+ {
+ if constexpr (IsDataTypeNumber)
+ {
+ chassert(wrapper_cast_type == CastType::accurate);
+ result_column = ConvertImpl::execute(
+ arguments,
+ result_type,
+ input_rows_count,
+ BehaviourOnErrorFromString::ConvertDefaultBehaviorTag,
+ AccurateConvertStrategyAdditions());
+ }
+ return true;
+ }
return false;
});
diff --git a/src/IO/S3/BlobStorageLogWriter.cpp b/src/IO/S3/BlobStorageLogWriter.cpp
index aaf4aea5a8e..c2f0cb86928 100644
--- a/src/IO/S3/BlobStorageLogWriter.cpp
+++ b/src/IO/S3/BlobStorageLogWriter.cpp
@@ -23,6 +23,9 @@ void BlobStorageLogWriter::addEvent(
if (!log)
return;
+ if (log->shouldIgnorePath(local_path_.empty() ? local_path : local_path_))
+ return;
+
if (!time_now.time_since_epoch().count())
time_now = std::chrono::system_clock::now();
diff --git a/src/Interpreters/Access/InterpreterGrantQuery.cpp b/src/Interpreters/Access/InterpreterGrantQuery.cpp
index a137404a669..6a46ac9c330 100644
--- a/src/Interpreters/Access/InterpreterGrantQuery.cpp
+++ b/src/Interpreters/Access/InterpreterGrantQuery.cpp
@@ -438,6 +438,12 @@ BlockIO InterpreterGrantQuery::execute()
RolesOrUsersSet roles_to_revoke;
collectRolesToGrantOrRevoke(access_control, query, roles_to_grant, roles_to_revoke);
+ /// Replacing empty database with the default. This step must be done before replication to avoid privilege escalation.
+ String current_database = getContext()->getCurrentDatabase();
+ elements_to_grant.replaceEmptyDatabase(current_database);
+ elements_to_revoke.replaceEmptyDatabase(current_database);
+ query.access_rights_elements.replaceEmptyDatabase(current_database);
+
/// Executing on cluster.
if (!query.cluster.empty())
{
@@ -453,9 +459,6 @@ BlockIO InterpreterGrantQuery::execute()
}
/// Check if the current user has corresponding access rights granted with grant option.
- String current_database = getContext()->getCurrentDatabase();
- elements_to_grant.replaceEmptyDatabase(current_database);
- elements_to_revoke.replaceEmptyDatabase(current_database);
bool need_check_grantees_are_allowed = true;
if (!query.current_grants)
checkGrantOption(access_control, *current_user_access, grantees, need_check_grantees_are_allowed, elements_to_grant, elements_to_revoke);
diff --git a/src/Interpreters/BlobStorageLog.cpp b/src/Interpreters/BlobStorageLog.cpp
index 0324ef8713c..f20ac9165ac 100644
--- a/src/Interpreters/BlobStorageLog.cpp
+++ b/src/Interpreters/BlobStorageLog.cpp
@@ -9,6 +9,8 @@
#include
#include
+#include
+#include
namespace DB
{
@@ -69,4 +71,32 @@ void BlobStorageLogElement::appendToBlock(MutableColumns & columns) const
columns[i++]->insert(error_message);
}
+void BlobStorageLog::addSettingsForQuery(ContextMutablePtr & mutable_context, IAST::QueryKind query_kind) const
+{
+ SystemLog::addSettingsForQuery(mutable_context, query_kind);
+
+ if (query_kind == IAST::QueryKind::Insert)
+ mutable_context->setSetting("enable_blob_storage_log", false);
+}
+
+static std::string_view normalizePath(std::string_view path)
+{
+ if (path.starts_with("./"))
+ path.remove_prefix(2);
+ if (path.ends_with("/"))
+ path.remove_suffix(1);
+ return path;
+}
+
+void BlobStorageLog::prepareTable()
+{
+ SystemLog::prepareTable();
+ if (auto merge_tree_table = std::dynamic_pointer_cast(getStorage()))
+ {
+ std::unique_lock lock{prepare_mutex};
+ const auto & relative_data_path = merge_tree_table->getRelativeDataPath();
+ prefix_to_ignore = normalizePath(relative_data_path);
+ }
+}
+
}
diff --git a/src/Interpreters/BlobStorageLog.h b/src/Interpreters/BlobStorageLog.h
index 15e15be4f87..cf8f37299f7 100644
--- a/src/Interpreters/BlobStorageLog.h
+++ b/src/Interpreters/BlobStorageLog.h
@@ -1,11 +1,14 @@
#pragma once
-#include
-#include
-#include
-#include
-#include
#include
+#include
+
+#include
+
+#include
+#include
+#include
+#include
namespace DB
{
@@ -51,7 +54,23 @@ struct BlobStorageLogElement
class BlobStorageLog : public SystemLog
{
+public:
using SystemLog::SystemLog;
+
+ /// We should not log events for table itself to avoid infinite recursion
+ bool shouldIgnorePath(const String & path) const
+ {
+ std::shared_lock lock{prepare_mutex};
+ return !prefix_to_ignore.empty() && path.starts_with(prefix_to_ignore);
+ }
+
+protected:
+ void prepareTable() override;
+ void addSettingsForQuery(ContextMutablePtr & mutable_context, IAST::QueryKind query_kind) const override;
+
+private:
+ mutable std::shared_mutex prepare_mutex;
+ String prefix_to_ignore;
};
}
diff --git a/src/Interpreters/Context.cpp b/src/Interpreters/Context.cpp
index 5c9ae4716b9..06b3adb328d 100644
--- a/src/Interpreters/Context.cpp
+++ b/src/Interpreters/Context.cpp
@@ -4103,6 +4103,13 @@ std::shared_ptr Context::getBackupLog() const
std::shared_ptr Context::getBlobStorageLog() const
{
+ bool enable_blob_storage_log = settings.enable_blob_storage_log;
+ if (hasQueryContext())
+ enable_blob_storage_log = getQueryContext()->getSettingsRef().enable_blob_storage_log;
+
+ if (!enable_blob_storage_log)
+ return {};
+
SharedLockGuard lock(shared->mutex);
if (!shared->system_logs)
diff --git a/src/Interpreters/ITokenExtractor.cpp b/src/Interpreters/ITokenExtractor.cpp
index 1c5d0d4b6d4..f0bf90fcb5c 100644
--- a/src/Interpreters/ITokenExtractor.cpp
+++ b/src/Interpreters/ITokenExtractor.cpp
@@ -240,4 +240,34 @@ bool SplitTokenExtractor::nextInStringLike(const char * data, size_t length, siz
return !bad_token && !token.empty();
}
+void SplitTokenExtractor::substringToBloomFilter(const char * data, size_t length, BloomFilter & bloom_filter, bool is_prefix, bool is_suffix) const
+{
+ size_t cur = 0;
+ size_t token_start = 0;
+ size_t token_len = 0;
+
+ while (cur < length && nextInString(data, length, &cur, &token_start, &token_len))
+ // In order to avoid filter updates with incomplete tokens,
+ // first token is ignored, unless substring is prefix and
+ // last token is ignored, unless substring is suffix
+ if ((token_start > 0 || is_prefix) && (token_start + token_len < length || is_suffix))
+ bloom_filter.add(data + token_start, token_len);
+}
+
+void SplitTokenExtractor::substringToGinFilter(const char * data, size_t length, GinFilter & gin_filter, bool is_prefix, bool is_suffix) const
+{
+ gin_filter.setQueryString(data, length);
+
+ size_t cur = 0;
+ size_t token_start = 0;
+ size_t token_len = 0;
+
+ while (cur < length && nextInString(data, length, &cur, &token_start, &token_len))
+ // In order to avoid filter updates with incomplete tokens,
+ // first token is ignored, unless substring is prefix and
+ // last token is ignored, unless substring is suffix
+ if ((token_start > 0 || is_prefix) && (token_start + token_len < length || is_suffix))
+ gin_filter.addTerm(data + token_start, token_len);
+}
+
}
diff --git a/src/Interpreters/ITokenExtractor.h b/src/Interpreters/ITokenExtractor.h
index 2423ef12311..76711606d09 100644
--- a/src/Interpreters/ITokenExtractor.h
+++ b/src/Interpreters/ITokenExtractor.h
@@ -28,8 +28,22 @@ struct ITokenExtractor
/// It skips unescaped `%` and `_` and supports escaping symbols, but it is less lightweight.
virtual bool nextInStringLike(const char * data, size_t length, size_t * pos, String & out) const = 0;
+ /// Updates Bloom filter from exact-match string filter value
virtual void stringToBloomFilter(const char * data, size_t length, BloomFilter & bloom_filter) const = 0;
+ /// Updates Bloom filter from substring-match string filter value.
+ /// An `ITokenExtractor` implementation may decide to skip certain
+ /// tokens depending on whether the substring is a prefix or a suffix.
+ virtual void substringToBloomFilter(
+ const char * data,
+ size_t length,
+ BloomFilter & bloom_filter,
+ bool is_prefix [[maybe_unused]],
+ bool is_suffix [[maybe_unused]]) const
+ {
+ stringToBloomFilter(data, length, bloom_filter);
+ }
+
virtual void stringPaddedToBloomFilter(const char * data, size_t length, BloomFilter & bloom_filter) const
{
stringToBloomFilter(data, length, bloom_filter);
@@ -37,8 +51,22 @@ struct ITokenExtractor
virtual void stringLikeToBloomFilter(const char * data, size_t length, BloomFilter & bloom_filter) const = 0;
+ /// Updates GIN filter from exact-match string filter value
virtual void stringToGinFilter(const char * data, size_t length, GinFilter & gin_filter) const = 0;
+ /// Updates GIN filter from substring-match string filter value.
+ /// An `ITokenExtractor` implementation may decide to skip certain
+ /// tokens depending on whether the substring is a prefix or a suffix.
+ virtual void substringToGinFilter(
+ const char * data,
+ size_t length,
+ GinFilter & gin_filter,
+ bool is_prefix [[maybe_unused]],
+ bool is_suffix [[maybe_unused]]) const
+ {
+ stringToGinFilter(data, length, gin_filter);
+ }
+
virtual void stringPaddedToGinFilter(const char * data, size_t length, GinFilter & gin_filter) const
{
stringToGinFilter(data, length, gin_filter);
@@ -148,6 +176,11 @@ struct SplitTokenExtractor final : public ITokenExtractorHelpergetSettings().allow_experimental_statistic && (
- command_ast->type == ASTAlterCommand::ADD_STATISTIC ||
- command_ast->type == ASTAlterCommand::DROP_STATISTIC ||
- command_ast->type == ASTAlterCommand::MATERIALIZE_STATISTIC))
- throw Exception(ErrorCodes::INCORRECT_QUERY, "Alter table with statistic is now disabled. Turn on allow_experimental_statistic");
+ if (!getContext()->getSettings().allow_experimental_statistics && (
+ command_ast->type == ASTAlterCommand::ADD_STATISTICS ||
+ command_ast->type == ASTAlterCommand::DROP_STATISTICS ||
+ command_ast->type == ASTAlterCommand::MATERIALIZE_STATISTICS))
+ throw Exception(ErrorCodes::INCORRECT_QUERY, "Alter table with statistics is now disabled. Turn on allow_experimental_statistics");
}
if (typeid_cast(database.get()))
@@ -343,19 +343,24 @@ AccessRightsElements InterpreterAlterQuery::getRequiredAccessForCommand(const AS
required_access.emplace_back(AccessType::ALTER_SAMPLE_BY, database, table);
break;
}
- case ASTAlterCommand::ADD_STATISTIC:
+ case ASTAlterCommand::ADD_STATISTICS:
{
- required_access.emplace_back(AccessType::ALTER_ADD_STATISTIC, database, table);
+ required_access.emplace_back(AccessType::ALTER_ADD_STATISTICS, database, table);
break;
}
- case ASTAlterCommand::DROP_STATISTIC:
+ case ASTAlterCommand::MODIFY_STATISTICS:
{
- required_access.emplace_back(AccessType::ALTER_DROP_STATISTIC, database, table);
+ required_access.emplace_back(AccessType::ALTER_MODIFY_STATISTICS, database, table);
break;
}
- case ASTAlterCommand::MATERIALIZE_STATISTIC:
+ case ASTAlterCommand::DROP_STATISTICS:
{
- required_access.emplace_back(AccessType::ALTER_MATERIALIZE_STATISTIC, database, table);
+ required_access.emplace_back(AccessType::ALTER_DROP_STATISTICS, database, table);
+ break;
+ }
+ case ASTAlterCommand::MATERIALIZE_STATISTICS:
+ {
+ required_access.emplace_back(AccessType::ALTER_MATERIALIZE_STATISTICS, database, table);
break;
}
case ASTAlterCommand::ADD_INDEX:
diff --git a/src/Interpreters/InterpreterCreateQuery.cpp b/src/Interpreters/InterpreterCreateQuery.cpp
index f2e03ca41bd..66936dc25d7 100644
--- a/src/Interpreters/InterpreterCreateQuery.cpp
+++ b/src/Interpreters/InterpreterCreateQuery.cpp
@@ -448,9 +448,9 @@ ASTPtr InterpreterCreateQuery::formatColumns(const ColumnsDescription & columns)
column_declaration->children.push_back(column_declaration->codec);
}
- if (column.stat)
+ if (!column.statistics.empty())
{
- column_declaration->stat_type = column.stat->ast;
+ column_declaration->stat_type = column.statistics.getAST();
column_declaration->children.push_back(column_declaration->stat_type);
}
@@ -675,11 +675,13 @@ ColumnsDescription InterpreterCreateQuery::getColumnsDescription(
col_decl.codec, column.type, sanity_check_compression_codecs, allow_experimental_codecs, enable_deflate_qpl_codec, enable_zstd_qat_codec);
}
+ column.statistics.column_name = column.name; /// We assign column name here for better exception error message.
if (col_decl.stat_type)
{
- if (!skip_checks && !context_->getSettingsRef().allow_experimental_statistic)
- throw Exception(ErrorCodes::INCORRECT_QUERY, "Create table with statistic is now disabled. Turn on allow_experimental_statistic");
- column.stat = StatisticDescription::getStatisticFromColumnDeclaration(col_decl);
+ if (!skip_checks && !context_->getSettingsRef().allow_experimental_statistics)
+ throw Exception(ErrorCodes::INCORRECT_QUERY, "Create table with statistics is now disabled. Turn on allow_experimental_statistics");
+ column.statistics = ColumnStatisticsDescription::fromColumnDeclaration(col_decl);
+ column.statistics.data_type = column.type;
}
if (col_decl.ttl)
@@ -754,7 +756,7 @@ InterpreterCreateQuery::TableProperties InterpreterCreateQuery::getTableProperti
throw Exception(ErrorCodes::SUPPORT_IS_DISABLED, "Experimental full-text index feature is not enabled (the setting 'allow_experimental_full_text_index')");
/// ----
/// Temporary check during a transition period. Please remove at the end of 2024.
- if (index_desc.type == INVERTED_INDEX_NAME && settings.allow_experimental_inverted_index) /// The funny condition is not a mistake, see 02346_fulltext_index_old_name.sql
+ if (index_desc.type == INVERTED_INDEX_NAME && !settings.allow_experimental_inverted_index)
throw Exception(ErrorCodes::ILLEGAL_INDEX, "Please use index type 'full_text' instead of 'inverted'");
/// ----
if (index_desc.type == "annoy" && !settings.allow_experimental_annoy_index)
diff --git a/src/Interpreters/InterpreterExplainQuery.cpp b/src/Interpreters/InterpreterExplainQuery.cpp
index 458be843b59..3a06e1b2301 100644
--- a/src/Interpreters/InterpreterExplainQuery.cpp
+++ b/src/Interpreters/InterpreterExplainQuery.cpp
@@ -67,8 +67,8 @@ namespace
static void visit(ASTSelectQuery & select, ASTPtr & node, Data & data)
{
- /// we need to read statistic when `allow_statistic_optimize` is enabled.
- bool only_analyze = !data.getContext()->getSettings().allow_statistic_optimize;
+ /// we need to read statistic when `allow_statistics_optimize` is enabled.
+ bool only_analyze = !data.getContext()->getSettings().allow_statistics_optimize;
InterpreterSelectQuery interpreter(
node, data.getContext(), SelectQueryOptions(QueryProcessingStage::FetchColumns).analyze(only_analyze).modify());
diff --git a/src/Interpreters/InterpreterSelectQuery.cpp b/src/Interpreters/InterpreterSelectQuery.cpp
index e72cf670f69..09f987a1c24 100644
--- a/src/Interpreters/InterpreterSelectQuery.cpp
+++ b/src/Interpreters/InterpreterSelectQuery.cpp
@@ -657,7 +657,7 @@ InterpreterSelectQuery::InterpreterSelectQuery(
MergeTreeWhereOptimizer where_optimizer{
std::move(column_compressed_sizes),
metadata_snapshot,
- storage->getConditionEstimatorByPredicate(storage_snapshot, nullptr, context),
+ storage->getConditionSelectivityEstimatorByPredicate(storage_snapshot, nullptr, context),
queried_columns,
supported_prewhere_columns,
log};
diff --git a/src/Interpreters/MutationsInterpreter.cpp b/src/Interpreters/MutationsInterpreter.cpp
index 4f6c1c5f18b..ba33b70b59c 100644
--- a/src/Interpreters/MutationsInterpreter.cpp
+++ b/src/Interpreters/MutationsInterpreter.cpp
@@ -55,7 +55,7 @@ namespace ErrorCodes
extern const int CANNOT_UPDATE_COLUMN;
extern const int UNEXPECTED_EXPRESSION;
extern const int THERE_IS_NO_COLUMN;
- extern const int ILLEGAL_STATISTIC;
+ extern const int ILLEGAL_STATISTICS;
}
@@ -781,7 +781,7 @@ void MutationsInterpreter::prepare(bool dry_run)
}
else if (command.type == MutationCommand::MATERIALIZE_INDEX)
{
- mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION);
+ mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION);
auto it = std::find_if(
std::cbegin(indices_desc), std::end(indices_desc),
[&](const IndexDescription & index)
@@ -801,20 +801,20 @@ void MutationsInterpreter::prepare(bool dry_run)
materialized_indices.emplace(command.index_name);
}
}
- else if (command.type == MutationCommand::MATERIALIZE_STATISTIC)
+ else if (command.type == MutationCommand::MATERIALIZE_STATISTICS)
{
- mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION);
- for (const auto & stat_column_name: command.statistic_columns)
+ mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION);
+ for (const auto & stat_column_name: command.statistics_columns)
{
- if (!columns_desc.has(stat_column_name) || !columns_desc.get(stat_column_name).stat)
- throw Exception(ErrorCodes::ILLEGAL_STATISTIC, "Unknown statistic column: {}", stat_column_name);
- dependencies.emplace(stat_column_name, ColumnDependency::STATISTIC);
+ if (!columns_desc.has(stat_column_name) || columns_desc.get(stat_column_name).statistics.empty())
+ throw Exception(ErrorCodes::ILLEGAL_STATISTICS, "Unknown statistics column: {}", stat_column_name);
+ dependencies.emplace(stat_column_name, ColumnDependency::STATISTICS);
materialized_statistics.emplace(stat_column_name);
}
}
else if (command.type == MutationCommand::MATERIALIZE_PROJECTION)
{
- mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION);
+ mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION);
const auto & projection = projections_desc.get(command.projection_name);
if (!source.hasProjection(projection.name) || source.hasBrokenProjection(projection.name))
{
@@ -825,18 +825,18 @@ void MutationsInterpreter::prepare(bool dry_run)
}
else if (command.type == MutationCommand::DROP_INDEX)
{
- mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION);
+ mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION);
materialized_indices.erase(command.index_name);
}
- else if (command.type == MutationCommand::DROP_STATISTIC)
+ else if (command.type == MutationCommand::DROP_STATISTICS)
{
- mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION);
- for (const auto & stat_column_name: command.statistic_columns)
+ mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION);
+ for (const auto & stat_column_name: command.statistics_columns)
materialized_statistics.erase(stat_column_name);
}
else if (command.type == MutationCommand::DROP_PROJECTION)
{
- mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION);
+ mutation_kind.set(MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION);
materialized_projections.erase(command.projection_name);
}
else if (command.type == MutationCommand::MATERIALIZE_TTL)
@@ -888,7 +888,7 @@ void MutationsInterpreter::prepare(bool dry_run)
{
if (dependency.kind == ColumnDependency::SKIP_INDEX
|| dependency.kind == ColumnDependency::PROJECTION
- || dependency.kind == ColumnDependency::STATISTIC)
+ || dependency.kind == ColumnDependency::STATISTICS)
dependencies.insert(dependency);
}
}
@@ -1360,7 +1360,7 @@ QueryPipelineBuilder MutationsInterpreter::execute()
Block MutationsInterpreter::getUpdatedHeader() const
{
// If it's an index/projection materialization, we don't write any data columns, thus empty header is used
- return mutation_kind.mutation_kind == MutationKind::MUTATE_INDEX_STATISTIC_PROJECTION ? Block{} : *updated_header;
+ return mutation_kind.mutation_kind == MutationKind::MUTATE_INDEX_STATISTICS_PROJECTION ? Block{} : *updated_header;
}
const ColumnDependencies & MutationsInterpreter::getColumnDependencies() const
diff --git a/src/Interpreters/MutationsInterpreter.h b/src/Interpreters/MutationsInterpreter.h
index 2d01c7154c8..6aaa233cda3 100644
--- a/src/Interpreters/MutationsInterpreter.h
+++ b/src/Interpreters/MutationsInterpreter.h
@@ -102,7 +102,7 @@ public:
enum MutationKindEnum
{
MUTATE_UNKNOWN,
- MUTATE_INDEX_STATISTIC_PROJECTION,
+ MUTATE_INDEX_STATISTICS_PROJECTION,
MUTATE_OTHER,
} mutation_kind = MUTATE_UNKNOWN;
diff --git a/src/Interpreters/SystemLog.cpp b/src/Interpreters/SystemLog.cpp
index 3af8761ff8e..5e0ce2cb0de 100644
--- a/src/Interpreters/SystemLog.cpp
+++ b/src/Interpreters/SystemLog.cpp
@@ -519,8 +519,7 @@ void SystemLog::flushImpl(const std::vector & to_flush,
// we need query context to do inserts to target table with MV containing subqueries or joins
auto insert_context = Context::createCopy(context);
insert_context->makeQueryContext();
- /// We always want to deliver the data to the original table regardless of the MVs
- insert_context->setSetting("materialized_views_ignore_errors", true);
+ addSettingsForQuery(insert_context, IAST::QueryKind::Insert);
InterpreterInsertQuery interpreter(query_ptr, insert_context);
BlockIO io = interpreter.execute();
@@ -541,13 +540,18 @@ void SystemLog::flushImpl(const std::vector & to_flush,
LOG_TRACE(log, "Flushed system log up to offset {}", to_flush_end);
}
+template
+StoragePtr SystemLog::getStorage() const
+{
+ return DatabaseCatalog::instance().tryGetTable(table_id, getContext());
+}
template
void SystemLog::prepareTable()
{
String description = table_id.getNameForLogs();
- auto table = DatabaseCatalog::instance().tryGetTable(table_id, getContext());
+ auto table = getStorage();
if (table)
{
if (old_create_query.empty())
@@ -596,10 +600,9 @@ void SystemLog::prepareTable()
merges_lock = table->getActionLock(ActionLocks::PartsMerge);
auto query_context = Context::createCopy(context);
- /// As this operation is performed automatically we don't want it to fail because of user dependencies on log tables
- query_context->setSetting("check_table_dependencies", Field{false});
- query_context->setSetting("check_referential_table_dependencies", Field{false});
query_context->makeQueryContext();
+ addSettingsForQuery(query_context, IAST::QueryKind::Rename);
+
InterpreterRenameQuery(rename, query_context).execute();
/// The required table will be created.
@@ -616,6 +619,7 @@ void SystemLog::prepareTable()
auto query_context = Context::createCopy(context);
query_context->makeQueryContext();
+ addSettingsForQuery(query_context, IAST::QueryKind::Create);
auto create_query_ast = getCreateTableQuery();
InterpreterCreateQuery interpreter(create_query_ast, query_context);
@@ -630,6 +634,22 @@ void SystemLog::prepareTable()
is_prepared = true;
}
+template
+void SystemLog::addSettingsForQuery(ContextMutablePtr & mutable_context, IAST::QueryKind query_kind) const
+{
+ if (query_kind == IAST::QueryKind::Insert)
+ {
+ /// We always want to deliver the data to the original table regardless of the MVs
+ mutable_context->setSetting("materialized_views_ignore_errors", true);
+ }
+ else if (query_kind == IAST::QueryKind::Rename)
+ {
+ /// As this operation is performed automatically we don't want it to fail because of user dependencies on log tables
+ mutable_context->setSetting("check_table_dependencies", Field{false});
+ mutable_context->setSetting("check_referential_table_dependencies", Field{false});
+ }
+}
+
template
ASTPtr SystemLog::getCreateTableQuery()
{
diff --git a/src/Interpreters/SystemLog.h b/src/Interpreters/SystemLog.h
index e5b79585701..af635ca1bdb 100644
--- a/src/Interpreters/SystemLog.h
+++ b/src/Interpreters/SystemLog.h
@@ -2,6 +2,7 @@
#include
#include
+#include
#include
@@ -139,6 +140,17 @@ protected:
using ISystemLog::thread_mutex;
using Base::queue;
+ StoragePtr getStorage() const;
+
+ /** Creates new table if it does not exist.
+ * Renames old table if its structure is not suitable.
+ * This cannot be done in constructor to avoid deadlock while renaming a table under locked Context when SystemLog object is created.
+ */
+ void prepareTable() override;
+
+ /// Some tables can override settings for internal queries
+ virtual void addSettingsForQuery(ContextMutablePtr & mutable_context, IAST::QueryKind query_kind) const;
+
private:
/* Saving thread data */
const StorageID table_id;
@@ -147,12 +159,6 @@ private:
String old_create_query;
bool is_prepared = false;
- /** Creates new table if it does not exist.
- * Renames old table if its structure is not suitable.
- * This cannot be done in constructor to avoid deadlock while renaming a table under locked Context when SystemLog object is created.
- */
- void prepareTable() override;
-
void savingThreadFunction() override;
/// flushImpl can be executed only in saving_thread.
diff --git a/src/Parsers/ASTAlterQuery.cpp b/src/Parsers/ASTAlterQuery.cpp
index f104e715452..90b63d2ce6f 100644
--- a/src/Parsers/ASTAlterQuery.cpp
+++ b/src/Parsers/ASTAlterQuery.cpp
@@ -42,8 +42,8 @@ ASTPtr ASTAlterCommand::clone() const
res->projection_decl = res->children.emplace_back(projection_decl->clone()).get();
if (projection)
res->projection = res->children.emplace_back(projection->clone()).get();
- if (statistic_decl)
- res->statistic_decl = res->children.emplace_back(statistic_decl->clone()).get();
+ if (statistics_decl)
+ res->statistics_decl = res->children.emplace_back(statistics_decl->clone()).get();
if (partition)
res->partition = res->children.emplace_back(partition->clone()).get();
if (predicate)
@@ -200,27 +200,33 @@ void ASTAlterCommand::formatImpl(const FormatSettings & settings, FormatState &
partition->formatImpl(settings, state, frame);
}
}
- else if (type == ASTAlterCommand::ADD_STATISTIC)
+ else if (type == ASTAlterCommand::ADD_STATISTICS)
{
- settings.ostr << (settings.hilite ? hilite_keyword : "") << "ADD STATISTIC " << (if_not_exists ? "IF NOT EXISTS " : "")
+ settings.ostr << (settings.hilite ? hilite_keyword : "") << "ADD STATISTICS " << (if_not_exists ? "IF NOT EXISTS " : "")
<< (settings.hilite ? hilite_none : "");
- statistic_decl->formatImpl(settings, state, frame);
+ statistics_decl->formatImpl(settings, state, frame);
}
- else if (type == ASTAlterCommand::DROP_STATISTIC)
+ else if (type == ASTAlterCommand::MODIFY_STATISTICS)
{
- settings.ostr << (settings.hilite ? hilite_keyword : "") << (clear_statistic ? "CLEAR " : "DROP ") << "STATISTIC "
+ settings.ostr << (settings.hilite ? hilite_keyword : "") << "MODIFY STATISTICS "
+ << (settings.hilite ? hilite_none : "");
+ statistics_decl->formatImpl(settings, state, frame);
+ }
+ else if (type == ASTAlterCommand::DROP_STATISTICS)
+ {
+ settings.ostr << (settings.hilite ? hilite_keyword : "") << (clear_statistics ? "CLEAR " : "DROP ") << "STATISTICS "
<< (if_exists ? "IF EXISTS " : "") << (settings.hilite ? hilite_none : "");
- statistic_decl->formatImpl(settings, state, frame);
+ statistics_decl->formatImpl(settings, state, frame);
if (partition)
{
settings.ostr << (settings.hilite ? hilite_keyword : "") << " IN PARTITION " << (settings.hilite ? hilite_none : "");
partition->formatImpl(settings, state, frame);
}
}
- else if (type == ASTAlterCommand::MATERIALIZE_STATISTIC)
+ else if (type == ASTAlterCommand::MATERIALIZE_STATISTICS)
{
- settings.ostr << (settings.hilite ? hilite_keyword : "") << "MATERIALIZE STATISTIC " << (settings.hilite ? hilite_none : "");
- statistic_decl->formatImpl(settings, state, frame);
+ settings.ostr << (settings.hilite ? hilite_keyword : "") << "MATERIALIZE STATISTICS " << (settings.hilite ? hilite_none : "");
+ statistics_decl->formatImpl(settings, state, frame);
if (partition)
{
settings.ostr << (settings.hilite ? hilite_keyword : "") << " IN PARTITION " << (settings.hilite ? hilite_none : "");
@@ -507,7 +513,7 @@ void ASTAlterCommand::forEachPointerToChild(std::function f)
f(reinterpret_cast(&constraint));
f(reinterpret_cast(&projection_decl));
f(reinterpret_cast(&projection));
- f(reinterpret_cast(&statistic_decl));
+ f(reinterpret_cast(&statistics_decl));
f(reinterpret_cast(&partition));
f(reinterpret_cast(&predicate));
f(reinterpret_cast(&update_assignments));
diff --git a/src/Parsers/ASTAlterQuery.h b/src/Parsers/ASTAlterQuery.h
index a3cab1688c2..d7269bed2da 100644
--- a/src/Parsers/ASTAlterQuery.h
+++ b/src/Parsers/ASTAlterQuery.h
@@ -55,9 +55,10 @@ public:
DROP_PROJECTION,
MATERIALIZE_PROJECTION,
- ADD_STATISTIC,
- DROP_STATISTIC,
- MATERIALIZE_STATISTIC,
+ ADD_STATISTICS,
+ DROP_STATISTICS,
+ MODIFY_STATISTICS,
+ MATERIALIZE_STATISTICS,
DROP_PARTITION,
DROP_DETACHED_PARTITION,
@@ -135,7 +136,7 @@ public:
*/
IAST * projection = nullptr;
- IAST * statistic_decl = nullptr;
+ IAST * statistics_decl = nullptr;
/** Used in DROP PARTITION, ATTACH PARTITION FROM, FORGET PARTITION, UPDATE, DELETE queries.
* The value or ID of the partition is stored here.
@@ -180,7 +181,7 @@ public:
bool clear_index = false; /// for CLEAR INDEX (do not drop index from metadata)
- bool clear_statistic = false; /// for CLEAR STATISTIC (do not drop statistic from metadata)
+ bool clear_statistics = false; /// for CLEAR STATISTICS (do not drop statistics from metadata)
bool clear_projection = false; /// for CLEAR PROJECTION (do not drop projection from metadata)
diff --git a/src/Parsers/ASTStatisticDeclaration.cpp b/src/Parsers/ASTStatisticDeclaration.cpp
deleted file mode 100644
index 0e20b020ab3..00000000000
--- a/src/Parsers/ASTStatisticDeclaration.cpp
+++ /dev/null
@@ -1,42 +0,0 @@
-#include
-#include
-
-#include
-#include
-#include
-
-
-namespace DB
-{
-
-ASTPtr ASTStatisticDeclaration::clone() const
-{
- auto res = std::make_shared();
-
- res->set(res->columns, columns->clone());
- res->type = type;
-
- return res;
-}
-
-std::vector ASTStatisticDeclaration::getColumnNames() const
-{
- std::vector result;
- result.reserve(columns->children.size());
- for (const ASTPtr & column_ast : columns->children)
- {
- result.push_back(column_ast->as().name());
- }
- return result;
-
-}
-
-void ASTStatisticDeclaration::formatImpl(const FormatSettings & s, FormatState & state, FormatStateStacked frame) const
-{
- columns->formatImpl(s, state, frame);
- s.ostr << (s.hilite ? hilite_keyword : "") << " TYPE " << (s.hilite ? hilite_none : "");
- s.ostr << backQuoteIfNeed(type);
-}
-
-}
-
diff --git a/src/Parsers/ASTStatisticsDeclaration.cpp b/src/Parsers/ASTStatisticsDeclaration.cpp
new file mode 100644
index 00000000000..f9b7a9e29db
--- /dev/null
+++ b/src/Parsers/ASTStatisticsDeclaration.cpp
@@ -0,0 +1,60 @@
+#include
+#include
+
+#include
+#include
+#include
+
+
+namespace DB
+{
+
+ASTPtr ASTStatisticsDeclaration::clone() const
+{
+ auto res = std::make_shared();
+
+ res->set(res->columns, columns->clone());
+ if (types)
+ res->set(res->types, types->clone());
+
+ return res;
+}
+
+std::vector ASTStatisticsDeclaration::getColumnNames() const
+{
+ std::vector result;
+ result.reserve(columns->children.size());
+ for (const ASTPtr & column_ast : columns->children)
+ {
+ result.push_back(column_ast->as().name());
+ }
+ return result;
+
+}
+
+std::vector ASTStatisticsDeclaration::getTypeNames() const
+{
+ chassert(types != nullptr);
+ std::vector result;
+ result.reserve(types->children.size());
+ for (const ASTPtr & column_ast : types->children)
+ {
+ result.push_back(column_ast->as().name);
+ }
+ return result;
+
+}
+
+void ASTStatisticsDeclaration::formatImpl(const FormatSettings & s, FormatState & state, FormatStateStacked frame) const
+{
+ columns->formatImpl(s, state, frame);
+ s.ostr << (s.hilite ? hilite_keyword : "");
+ if (types)
+ {
+ s.ostr << " TYPE " << (s.hilite ? hilite_none : "");
+ types->formatImpl(s, state, frame);
+ }
+}
+
+}
+
diff --git a/src/Parsers/ASTStatisticDeclaration.h b/src/Parsers/ASTStatisticsDeclaration.h
similarity index 74%
rename from src/Parsers/ASTStatisticDeclaration.h
rename to src/Parsers/ASTStatisticsDeclaration.h
index f936c93f2ba..f43567b3c70 100644
--- a/src/Parsers/ASTStatisticDeclaration.h
+++ b/src/Parsers/ASTStatisticsDeclaration.h
@@ -9,17 +9,17 @@ class ASTFunction;
/** name BY columns TYPE typename(args) in create query
*/
-class ASTStatisticDeclaration : public IAST
+class ASTStatisticsDeclaration : public IAST
{
public:
IAST * columns;
- /// TODO type should be a list of ASTFunction, for example, 'tdigest(256), hyperloglog(128)', etc.
- String type;
+ IAST * types;
/** Get the text that identifies this element. */
String getID(char) const override { return "Stat"; }
std::vector getColumnNames() const;
+ std::vector getTypeNames() const;
ASTPtr clone() const override;
void formatImpl(const FormatSettings & s, FormatState & state, FormatStateStacked frame) const override;
diff --git a/src/Parsers/CommonParsers.h b/src/Parsers/CommonParsers.h
index 97094b00bc6..f0cbe42da80 100644
--- a/src/Parsers/CommonParsers.h
+++ b/src/Parsers/CommonParsers.h
@@ -13,7 +13,7 @@ namespace DB
MR_MACROS(ADD_CONSTRAINT, "ADD CONSTRAINT") \
MR_MACROS(ADD_INDEX, "ADD INDEX") \
MR_MACROS(ADD_PROJECTION, "ADD PROJECTION") \
- MR_MACROS(ADD_STATISTIC, "ADD STATISTIC") \
+ MR_MACROS(ADD_STATISTICS, "ADD STATISTICS") \
MR_MACROS(ADD, "ADD") \
MR_MACROS(ADMIN_OPTION_FOR, "ADMIN OPTION FOR") \
MR_MACROS(AFTER, "AFTER") \
@@ -83,7 +83,7 @@ namespace DB
MR_MACROS(CLEAR_COLUMN, "CLEAR COLUMN") \
MR_MACROS(CLEAR_INDEX, "CLEAR INDEX") \
MR_MACROS(CLEAR_PROJECTION, "CLEAR PROJECTION") \
- MR_MACROS(CLEAR_STATISTIC, "CLEAR STATISTIC") \
+ MR_MACROS(CLEAR_STATISTICS, "CLEAR STATISTICS") \
MR_MACROS(CLUSTER, "CLUSTER") \
MR_MACROS(CLUSTERS, "CLUSTERS") \
MR_MACROS(CN, "CN") \
@@ -150,7 +150,7 @@ namespace DB
MR_MACROS(DROP_PART, "DROP PART") \
MR_MACROS(DROP_PARTITION, "DROP PARTITION") \
MR_MACROS(DROP_PROJECTION, "DROP PROJECTION") \
- MR_MACROS(DROP_STATISTIC, "DROP STATISTIC") \
+ MR_MACROS(DROP_STATISTICS, "DROP STATISTICS") \
MR_MACROS(DROP_TABLE, "DROP TABLE") \
MR_MACROS(DROP_TEMPORARY_TABLE, "DROP TEMPORARY TABLE") \
MR_MACROS(DROP, "DROP") \
@@ -279,7 +279,7 @@ namespace DB
MR_MACROS(MATERIALIZE_COLUMN, "MATERIALIZE COLUMN") \
MR_MACROS(MATERIALIZE_INDEX, "MATERIALIZE INDEX") \
MR_MACROS(MATERIALIZE_PROJECTION, "MATERIALIZE PROJECTION") \
- MR_MACROS(MATERIALIZE_STATISTIC, "MATERIALIZE STATISTIC") \
+ MR_MACROS(MATERIALIZE_STATISTICS, "MATERIALIZE STATISTICS") \
MR_MACROS(MATERIALIZE_TTL, "MATERIALIZE TTL") \
MR_MACROS(MATERIALIZE, "MATERIALIZE") \
MR_MACROS(MATERIALIZED, "MATERIALIZED") \
@@ -304,6 +304,7 @@ namespace DB
MR_MACROS(MODIFY_QUERY, "MODIFY QUERY") \
MR_MACROS(MODIFY_REFRESH, "MODIFY REFRESH") \
MR_MACROS(MODIFY_SAMPLE_BY, "MODIFY SAMPLE BY") \
+ MR_MACROS(MODIFY_STATISTICS, "MODIFY STATISTICS") \
MR_MACROS(MODIFY_SETTING, "MODIFY SETTING") \
MR_MACROS(MODIFY_SQL_SECURITY, "MODIFY SQL SECURITY") \
MR_MACROS(MODIFY_TTL, "MODIFY TTL") \
@@ -447,7 +448,7 @@ namespace DB
MR_MACROS(SQL_SECURITY, "SQL SECURITY") \
MR_MACROS(SS, "SS") \
MR_MACROS(START_TRANSACTION, "START TRANSACTION") \
- MR_MACROS(STATISTIC, "STATISTIC") \
+ MR_MACROS(STATISTICS, "STATISTICS") \
MR_MACROS(STEP, "STEP") \
MR_MACROS(STORAGE, "STORAGE") \
MR_MACROS(STRICT, "STRICT") \
diff --git a/src/Parsers/ExpressionElementParsers.cpp b/src/Parsers/ExpressionElementParsers.cpp
index 416f696323c..5997452bcf3 100644
--- a/src/Parsers/ExpressionElementParsers.cpp
+++ b/src/Parsers/ExpressionElementParsers.cpp
@@ -703,7 +703,7 @@ bool ParserCodec::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
return true;
}
-bool ParserStatisticType::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
+bool ParserStatisticsType::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
{
ParserList stat_type_parser(std::make_unique(),
std::make_unique(TokenType::Comma), false);
@@ -722,7 +722,7 @@ bool ParserStatisticType::parseImpl(Pos & pos, ASTPtr & node, Expected & expecte
++pos;
auto function_node = std::make_shared();
- function_node->name = "STATISTIC";
+ function_node->name = "STATISTICS";
function_node->arguments = stat_type;
function_node->children.push_back(function_node->arguments);
diff --git a/src/Parsers/ExpressionElementParsers.h b/src/Parsers/ExpressionElementParsers.h
index 14d501e50da..4e3f29bfe0c 100644
--- a/src/Parsers/ExpressionElementParsers.h
+++ b/src/Parsers/ExpressionElementParsers.h
@@ -202,11 +202,11 @@ protected:
bool parseImpl(Pos & pos, ASTPtr & node, Expected & expected) override;
};
-/// STATISTIC(tdigest(200))
-class ParserStatisticType : public IParserBase
+/// STATISTICS(tdigest(200))
+class ParserStatisticsType : public IParserBase
{
protected:
- const char * getName() const override { return "statistic"; }
+ const char * getName() const override { return "statistics"; }
bool parseImpl(Pos & pos, ASTPtr & node, Expected & expected) override;
};
diff --git a/src/Parsers/Lexer.cpp b/src/Parsers/Lexer.cpp
index 5f2bd50524c..b4601389696 100644
--- a/src/Parsers/Lexer.cpp
+++ b/src/Parsers/Lexer.cpp
@@ -59,9 +59,6 @@ Token quotedStringWithUnicodeQuotes(const char *& pos, const char * const token_
pos = find_first_symbols<'\xE2'>(pos, end);
if (pos + 2 >= end)
return Token(error_token, token_begin, end);
- /// Empty identifiers are not allowed, while empty strings are.
- if (success_token == TokenType::QuotedIdentifier && pos + 3 >= end)
- return Token(error_token, token_begin, end);
if (pos[0] == '\xE2' && pos[1] == '\x80' && pos[2] == expected_end_byte)
{
diff --git a/src/Parsers/ParserAlterQuery.cpp b/src/Parsers/ParserAlterQuery.cpp
index 6f48f79d942..28dbf781011 100644
--- a/src/Parsers/ParserAlterQuery.cpp
+++ b/src/Parsers/ParserAlterQuery.cpp
@@ -49,10 +49,11 @@ bool ParserAlterCommand::parseImpl(Pos & pos, ASTPtr & node, Expected & expected
ParserKeyword s_clear_index(Keyword::CLEAR_INDEX);
ParserKeyword s_materialize_index(Keyword::MATERIALIZE_INDEX);
- ParserKeyword s_add_statistic(Keyword::ADD_STATISTIC);
- ParserKeyword s_drop_statistic(Keyword::DROP_STATISTIC);
- ParserKeyword s_clear_statistic(Keyword::CLEAR_STATISTIC);
- ParserKeyword s_materialize_statistic(Keyword::MATERIALIZE_STATISTIC);
+ ParserKeyword s_add_statistics(Keyword::ADD_STATISTICS);
+ ParserKeyword s_drop_statistics(Keyword::DROP_STATISTICS);
+ ParserKeyword s_modify_statistics(Keyword::MODIFY_STATISTICS);
+ ParserKeyword s_clear_statistics(Keyword::CLEAR_STATISTICS);
+ ParserKeyword s_materialize_statistics(Keyword::MATERIALIZE_STATISTICS);
ParserKeyword s_add_constraint(Keyword::ADD_CONSTRAINT);
ParserKeyword s_drop_constraint(Keyword::DROP_CONSTRAINT);
@@ -126,7 +127,8 @@ bool ParserAlterCommand::parseImpl(Pos & pos, ASTPtr & node, Expected & expected
ParserIdentifier parser_remove_property;
ParserCompoundColumnDeclaration parser_col_decl;
ParserIndexDeclaration parser_idx_decl;
- ParserStatisticDeclaration parser_stat_decl;
+ ParserStatisticsDeclaration parser_stat_decl;
+ ParserStatisticsDeclarationWithoutTypes parser_stat_decl_without_types;
ParserConstraintDeclaration parser_constraint_decl;
ParserProjectionDeclaration parser_projection_decl;
ParserCompoundColumnDeclaration parser_modify_col_decl(false, false, true);
@@ -154,7 +156,7 @@ bool ParserAlterCommand::parseImpl(Pos & pos, ASTPtr & node, Expected & expected
ASTPtr command_constraint;
ASTPtr command_projection_decl;
ASTPtr command_projection;
- ASTPtr command_statistic_decl;
+ ASTPtr command_statistics_decl;
ASTPtr command_partition;
ASTPtr command_predicate;
ASTPtr command_update_assignments;
@@ -368,36 +370,43 @@ bool ParserAlterCommand::parseImpl(Pos & pos, ASTPtr & node, Expected & expected
return false;
}
}
- else if (s_add_statistic.ignore(pos, expected))
+ else if (s_add_statistics.ignore(pos, expected))
{
if (s_if_not_exists.ignore(pos, expected))
command->if_not_exists = true;
- if (!parser_stat_decl.parse(pos, command_statistic_decl, expected))
+ if (!parser_stat_decl.parse(pos, command_statistics_decl, expected))
return false;
- command->type = ASTAlterCommand::ADD_STATISTIC;
+ command->type = ASTAlterCommand::ADD_STATISTICS;
}
- else if (s_drop_statistic.ignore(pos, expected))
+ else if (s_modify_statistics.ignore(pos, expected))
+ {
+ if (!parser_stat_decl.parse(pos, command_statistics_decl, expected))
+ return false;
+
+ command->type = ASTAlterCommand::MODIFY_STATISTICS;
+ }
+ else if (s_drop_statistics.ignore(pos, expected))
{
if (s_if_exists.ignore(pos, expected))
command->if_exists = true;
- if (!parser_stat_decl.parse(pos, command_statistic_decl, expected))
+ if (!parser_stat_decl_without_types.parse(pos, command_statistics_decl, expected))
return false;
- command->type = ASTAlterCommand::DROP_STATISTIC;
+ command->type = ASTAlterCommand::DROP_STATISTICS;
}
- else if (s_clear_statistic.ignore(pos, expected))
+ else if (s_clear_statistics.ignore(pos, expected))
{
if (s_if_exists.ignore(pos, expected))
command->if_exists = true;
- if (!parser_stat_decl.parse(pos, command_statistic_decl, expected))
+ if (!parser_stat_decl_without_types.parse(pos, command_statistics_decl, expected))
return false;
- command->type = ASTAlterCommand::DROP_STATISTIC;
- command->clear_statistic = true;
+ command->type = ASTAlterCommand::DROP_STATISTICS;
+ command->clear_statistics = true;
command->detach = false;
if (s_in_partition.ignore(pos, expected))
@@ -406,15 +415,15 @@ bool ParserAlterCommand::parseImpl(Pos & pos, ASTPtr & node, Expected & expected
return false;
}
}
- else if (s_materialize_statistic.ignore(pos, expected))
+ else if (s_materialize_statistics.ignore(pos, expected))
{
if (s_if_exists.ignore(pos, expected))
command->if_exists = true;
- if (!parser_stat_decl.parse(pos, command_statistic_decl, expected))
+ if (!parser_stat_decl_without_types.parse(pos, command_statistics_decl, expected))
return false;
- command->type = ASTAlterCommand::MATERIALIZE_STATISTIC;
+ command->type = ASTAlterCommand::MATERIALIZE_STATISTICS;
command->detach = false;
if (s_in_partition.ignore(pos, expected))
@@ -931,8 +940,8 @@ bool ParserAlterCommand::parseImpl(Pos & pos, ASTPtr & node, Expected & expected
command->projection_decl = command->children.emplace_back(std::move(command_projection_decl)).get();
if (command_projection)
command->projection = command->children.emplace_back(std::move(command_projection)).get();
- if (command_statistic_decl)
- command->statistic_decl = command->children.emplace_back(std::move(command_statistic_decl)).get();
+ if (command_statistics_decl)
+ command->statistics_decl = command->children.emplace_back(std::move(command_statistics_decl)).get();
if (command_partition)
command->partition = command->children.emplace_back(std::move(command_partition)).get();
if (command_predicate)
diff --git a/src/Parsers/ParserCreateQuery.cpp b/src/Parsers/ParserCreateQuery.cpp
index c1b45871577..014dc7bd3bf 100644
--- a/src/Parsers/ParserCreateQuery.cpp
+++ b/src/Parsers/ParserCreateQuery.cpp
@@ -7,7 +7,7 @@
#include
#include
#include
-#include
+#include
#include
#include
#include
@@ -225,15 +225,15 @@ bool ParserIndexDeclaration::parseImpl(Pos & pos, ASTPtr & node, Expected & expe
return true;
}
-bool ParserStatisticDeclaration::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
+bool ParserStatisticsDeclaration::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
{
ParserKeyword s_type(Keyword::TYPE);
ParserList columns_p(std::make_unique(), std::make_unique(TokenType::Comma), false);
- ParserIdentifier type_p;
+ ParserList types_p(std::make_unique(), std::make_unique(TokenType::Comma), false);
ASTPtr columns;
- ASTPtr type;
+ ASTPtr types;
if (!columns_p.parse(pos, columns, expected))
return false;
@@ -241,12 +241,29 @@ bool ParserStatisticDeclaration::parseImpl(Pos & pos, ASTPtr & node, Expected &
if (!s_type.ignore(pos, expected))
return false;
- if (!type_p.parse(pos, type, expected))
+ if (!types_p.parse(pos, types, expected))
return false;
- auto stat = std::make_shared();
+ auto stat = std::make_shared();
+ stat->set(stat->columns, columns);
+ stat->set(stat->types, types);
+ node = stat;
+
+ return true;
+}
+
+bool ParserStatisticsDeclarationWithoutTypes::parseImpl(Pos & pos, ASTPtr & node, Expected & expected)
+{
+
+ ParserList columns_p(std::make_unique(), std::make_unique(TokenType::Comma), false);
+
+ ASTPtr columns;
+
+ if (!columns_p.parse(pos, columns, expected))
+ return false;
+
+ auto stat = std::make_shared();
stat->set(stat->columns, columns);
- stat->type = type->as().name();
node = stat;
return true;
diff --git a/src/Parsers/ParserCreateQuery.h b/src/Parsers/ParserCreateQuery.h
index d001c097114..27bb524970d 100644
--- a/src/Parsers/ParserCreateQuery.h
+++ b/src/Parsers/ParserCreateQuery.h
@@ -138,7 +138,7 @@ bool IParserColumnDeclaration::parseImpl(Pos & pos, ASTPtr & node, E
ParserKeyword s_auto_increment{Keyword::AUTO_INCREMENT};
ParserKeyword s_comment{Keyword::COMMENT};
ParserKeyword s_codec{Keyword::CODEC};
- ParserKeyword s_stat{Keyword::STATISTIC};
+ ParserKeyword s_stat{Keyword::STATISTICS};
ParserKeyword s_ttl{Keyword::TTL};
ParserKeyword s_remove{Keyword::REMOVE};
ParserKeyword s_modify_setting(Keyword::MODIFY_SETTING);
@@ -155,7 +155,7 @@ bool IParserColumnDeclaration::parseImpl(Pos & pos, ASTPtr & node, E
ParserLiteral literal_parser;
ParserCodec codec_parser;
ParserCollation collation_parser;
- ParserStatisticType stat_type_parser;
+ ParserStatisticsType stat_type_parser;
ParserExpression expression_parser;
ParserSetQuery settings_parser(true);
@@ -452,16 +452,27 @@ protected:
bool parseImpl(Pos & pos, ASTPtr & node, Expected & expected) override;
};
-class ParserStatisticDeclaration : public IParserBase
+class ParserStatisticsDeclaration : public IParserBase
{
public:
- ParserStatisticDeclaration() = default;
+ ParserStatisticsDeclaration() = default;
protected:
const char * getName() const override { return "statistics declaration"; }
bool parseImpl(Pos & pos, ASTPtr & node, Expected & expected) override;
};
+class ParserStatisticsDeclarationWithoutTypes : public IParserBase
+{
+public:
+ ParserStatisticsDeclarationWithoutTypes() = default;
+
+protected:
+ const char * getName() const override { return "statistics declaration"; }
+ bool parseImpl(Pos & pos, ASTPtr & node, Expected & expected) override;
+};
+
+
class ParserConstraintDeclaration : public IParserBase
{
protected:
diff --git a/src/Planner/PlannerJoinTree.cpp b/src/Planner/PlannerJoinTree.cpp
index 26e78ea69ac..6ec9a1e2f0d 100644
--- a/src/Planner/PlannerJoinTree.cpp
+++ b/src/Planner/PlannerJoinTree.cpp
@@ -1592,6 +1592,8 @@ JoinTreeQueryPlan buildQueryPlanForJoinNode(const QueryTreeNodePtr & join_table_
left_join_tree_query_plan.actions_dags.emplace_back(std::move(join_clauses_and_actions.left_join_expressions_actions));
if (join_clauses_and_actions.right_join_expressions_actions)
left_join_tree_query_plan.actions_dags.emplace_back(std::move(join_clauses_and_actions.right_join_expressions_actions));
+ if (join_clauses_and_actions.mixed_join_expressions_actions)
+ left_join_tree_query_plan.actions_dags.push_back(join_clauses_and_actions.mixed_join_expressions_actions);
auto mapping = std::move(left_join_tree_query_plan.query_node_to_plan_step_mapping);
auto & r_mapping = right_join_tree_query_plan.query_node_to_plan_step_mapping;
diff --git a/src/Processors/QueryPlan/Optimizations/optimizePrewhere.cpp b/src/Processors/QueryPlan/Optimizations/optimizePrewhere.cpp
index fbd9b451ddc..e374d8811db 100644
--- a/src/Processors/QueryPlan/Optimizations/optimizePrewhere.cpp
+++ b/src/Processors/QueryPlan/Optimizations/optimizePrewhere.cpp
@@ -83,7 +83,7 @@ void optimizePrewhere(Stack & stack, QueryPlan::Nodes &)
MergeTreeWhereOptimizer where_optimizer{
std::move(column_compressed_sizes),
storage_metadata,
- storage.getConditionEstimatorByPredicate(storage_snapshot, source_step_with_filter->getFilterActionsDAG(), context),
+ storage.getConditionSelectivityEstimatorByPredicate(storage_snapshot, source_step_with_filter->getFilterActionsDAG(), context),
queried_columns,
storage.supportedPrewhereColumns(),
getLogger("QueryPlanOptimizePrewhere")};
diff --git a/src/Processors/QueryPlan/Optimizations/optimizeTree.cpp b/src/Processors/QueryPlan/Optimizations/optimizeTree.cpp
index df9e095af30..136d474751a 100644
--- a/src/Processors/QueryPlan/Optimizations/optimizeTree.cpp
+++ b/src/Processors/QueryPlan/Optimizations/optimizeTree.cpp
@@ -118,7 +118,7 @@ void optimizeTreeSecondPass(const QueryPlanOptimizationSettings & optimization_s
optimizePrimaryKeyCondition(stack);
/// NOTE: optimizePrewhere can modify the stack.
- /// Prewhere optimization relies on PK optimization (getConditionEstimatorByPredicate)
+ /// Prewhere optimization relies on PK optimization (getConditionSelectivityEstimatorByPredicate)
if (optimization_settings.optimize_prewhere)
optimizePrewhere(stack, nodes);
diff --git a/src/Server/HTTPHandler.cpp b/src/Server/HTTPHandler.cpp
index d1db4cb3951..02d0959ff50 100644
--- a/src/Server/HTTPHandler.cpp
+++ b/src/Server/HTTPHandler.cpp
@@ -67,6 +67,8 @@ namespace ErrorCodes
{
extern const int BAD_ARGUMENTS;
extern const int LOGICAL_ERROR;
+ extern const int CANNOT_COMPILE_REGEXP;
+ extern const int CANNOT_OPEN_FILE;
extern const int CANNOT_PARSE_TEXT;
extern const int CANNOT_PARSE_ESCAPE_SEQUENCE;
extern const int CANNOT_PARSE_QUOTED_STRING;
@@ -78,8 +80,7 @@ namespace ErrorCodes
extern const int CANNOT_PARSE_IPV6;
extern const int CANNOT_PARSE_UUID;
extern const int CANNOT_PARSE_INPUT_ASSERTION_FAILED;
- extern const int CANNOT_OPEN_FILE;
- extern const int CANNOT_COMPILE_REGEXP;
+ extern const int CANNOT_SCHEDULE_TASK;
extern const int DUPLICATE_COLUMN;
extern const int ILLEGAL_COLUMN;
extern const int THERE_IS_NO_COLUMN;
@@ -267,6 +268,10 @@ static Poco::Net::HTTPResponse::HTTPStatus exceptionCodeToHTTPStatus(int excepti
{
return HTTPResponse::HTTP_REQUEST_TIMEOUT;
}
+ else if (exception_code == ErrorCodes::CANNOT_SCHEDULE_TASK)
+ {
+ return HTTPResponse::HTTP_SERVICE_UNAVAILABLE;
+ }
return HTTPResponse::HTTP_INTERNAL_SERVER_ERROR;
}
diff --git a/src/Storages/AlterCommands.cpp b/src/Storages/AlterCommands.cpp
index 4879d1a16dc..3a4c1d94750 100644
--- a/src/Storages/AlterCommands.cpp
+++ b/src/Storages/AlterCommands.cpp
@@ -25,7 +25,7 @@
#include
#include
#include
-#include
+#include
#include
#include
#include
@@ -44,7 +44,7 @@ namespace DB
namespace ErrorCodes
{
extern const int ILLEGAL_COLUMN;
- extern const int ILLEGAL_STATISTIC;
+ extern const int ILLEGAL_STATISTICS;
extern const int BAD_ARGUMENTS;
extern const int NOT_FOUND_COLUMN_IN_BLOCK;
extern const int LOGICAL_ERROR;
@@ -263,17 +263,32 @@ std::optional AlterCommand::parse(const ASTAlterCommand * command_
return command;
}
- else if (command_ast->type == ASTAlterCommand::ADD_STATISTIC)
+ else if (command_ast->type == ASTAlterCommand::ADD_STATISTICS)
{
AlterCommand command;
command.ast = command_ast->clone();
- command.statistic_decl = command_ast->statistic_decl->clone();
- command.type = AlterCommand::ADD_STATISTIC;
+ command.statistics_decl = command_ast->statistics_decl->clone();
+ command.type = AlterCommand::ADD_STATISTICS;
- const auto & ast_stat_decl = command_ast->statistic_decl->as();
+ const auto & ast_stat_decl = command_ast->statistics_decl->as();
- command.statistic_columns = ast_stat_decl.getColumnNames();
- command.statistic_type = ast_stat_decl.type;
+ command.statistics_columns = ast_stat_decl.getColumnNames();
+ command.statistics_types = ast_stat_decl.getTypeNames();
+ command.if_not_exists = command_ast->if_not_exists;
+
+ return command;
+ }
+ else if (command_ast->type == ASTAlterCommand::MODIFY_STATISTICS)
+ {
+ AlterCommand command;
+ command.ast = command_ast->clone();
+ command.statistics_decl = command_ast->statistics_decl->clone();
+ command.type = AlterCommand::MODIFY_STATISTICS;
+
+ const auto & ast_stat_decl = command_ast->statistics_decl->as();
+
+ command.statistics_columns = ast_stat_decl.getColumnNames();
+ command.statistics_types = ast_stat_decl.getTypeNames();
command.if_not_exists = command_ast->if_not_exists;
return command;
@@ -337,17 +352,17 @@ std::optional AlterCommand::parse(const ASTAlterCommand * command_
return command;
}
- else if (command_ast->type == ASTAlterCommand::DROP_STATISTIC)
+ else if (command_ast->type == ASTAlterCommand::DROP_STATISTICS)
{
AlterCommand command;
command.ast = command_ast->clone();
- command.type = AlterCommand::DROP_STATISTIC;
- const auto & ast_stat_decl = command_ast->statistic_decl->as();
+ command.statistics_decl = command_ast->statistics_decl->clone();
+ command.type = AlterCommand::DROP_STATISTICS;
+ const auto & ast_stat_decl = command_ast->statistics_decl->as();
- command.statistic_columns = ast_stat_decl.getColumnNames();
- command.statistic_type = ast_stat_decl.type;
+ command.statistics_columns = ast_stat_decl.getColumnNames();
command.if_exists = command_ast->if_exists;
- command.clear = command_ast->clear_statistic;
+ command.clear = command_ast->clear_statistics;
if (command_ast->partition)
command.partition = command_ast->partition->clone();
@@ -676,41 +691,56 @@ void AlterCommand::apply(StorageInMemoryMetadata & metadata, ContextPtr context)
metadata.secondary_indices.erase(erase_it);
}
}
- else if (type == ADD_STATISTIC)
+ else if (type == ADD_STATISTICS)
{
- for (const auto & statistic_column_name : statistic_columns)
+ for (const auto & statistics_column_name : statistics_columns)
{
- if (!metadata.columns.has(statistic_column_name))
+ if (!metadata.columns.has(statistics_column_name))
{
- throw Exception(ErrorCodes::ILLEGAL_STATISTIC, "Cannot add statistic {} with type {}: this column is not found", statistic_column_name, statistic_type);
+ throw Exception(ErrorCodes::ILLEGAL_STATISTICS, "Cannot add statistics for column {}: this column is not found", statistics_column_name);
}
- if (!if_exists && metadata.columns.get(statistic_column_name).stat)
- throw Exception(ErrorCodes::ILLEGAL_STATISTIC, "Cannot add statistic {} with type {}: statistic on this column with this type already exists", statistic_column_name, statistic_type);
}
- auto stats = StatisticDescription::getStatisticsFromAST(statistic_decl, metadata.columns);
- for (auto && stat : stats)
+ auto stats_vec = ColumnStatisticsDescription::fromAST(statistics_decl, metadata.columns);
+ for (const auto & stats : stats_vec)
{
- metadata.columns.modify(stat.column_name,
- [&](ColumnDescription & column) { column.stat = std::move(stat); });
+ metadata.columns.modify(stats.column_name,
+ [&](ColumnDescription & column) { column.statistics.merge(stats, column.name, column.type, if_not_exists); });
}
}
- else if (type == DROP_STATISTIC)
+ else if (type == DROP_STATISTICS)
{
- for (const auto & stat_column_name : statistic_columns)
+ for (const auto & statistics_column_name : statistics_columns)
{
- if (!metadata.columns.has(stat_column_name) || !metadata.columns.get(stat_column_name).stat)
+ if (!metadata.columns.has(statistics_column_name)
+ || metadata.columns.get(statistics_column_name).statistics.empty())
{
if (if_exists)
return;
- throw Exception(ErrorCodes::ILLEGAL_STATISTIC, "Wrong statistic name. Cannot find statistic {} with type {} to drop", backQuote(stat_column_name), statistic_type);
+ throw Exception(ErrorCodes::ILLEGAL_STATISTICS, "Wrong statistics name. Cannot find statistics {} to drop", backQuote(statistics_column_name));
}
- if (!partition && !clear)
+
+ if (!clear && !partition)
+ metadata.columns.modify(statistics_column_name,
+ [&](ColumnDescription & column) { column.statistics.clear(); });
+ }
+ }
+ else if (type == MODIFY_STATISTICS)
+ {
+ for (const auto & statistics_column_name : statistics_columns)
+ {
+ if (!metadata.columns.has(statistics_column_name))
{
- metadata.columns.modify(stat_column_name,
- [&](ColumnDescription & column) { column.stat = std::nullopt; });
+ throw Exception(ErrorCodes::ILLEGAL_STATISTICS, "Cannot add statistics for column {}: this column is not found", statistics_column_name);
}
}
+
+ auto stats_vec = ColumnStatisticsDescription::fromAST(statistics_decl, metadata.columns);
+ for (const auto & stats : stats_vec)
+ {
+ metadata.columns.modify(stats.column_name,
+ [&](ColumnDescription & column) { column.statistics.assign(stats); });
+ }
}
else if (type == ADD_CONSTRAINT)
{
@@ -833,8 +863,8 @@ void AlterCommand::apply(StorageInMemoryMetadata & metadata, ContextPtr context)
rename_visitor.visit(column_to_modify.default_desc.expression);
if (column_to_modify.ttl)
rename_visitor.visit(column_to_modify.ttl);
- if (column_to_modify.name == column_name && column_to_modify.stat)
- column_to_modify.stat->column_name = rename_to;
+ if (column_to_modify.name == column_name && !column_to_modify.statistics.empty())
+ column_to_modify.statistics.column_name = rename_to;
});
}
if (metadata.table_ttl.definition_ast)
@@ -958,7 +988,7 @@ bool AlterCommand::isRequireMutationStage(const StorageInMemoryMetadata & metada
if (isRemovingProperty() || type == REMOVE_TTL || type == REMOVE_SAMPLE_BY)
return false;
- if (type == DROP_INDEX || type == DROP_PROJECTION || type == RENAME_COLUMN || type == DROP_STATISTIC)
+ if (type == DROP_INDEX || type == DROP_PROJECTION || type == RENAME_COLUMN || type == DROP_STATISTICS)
return true;
/// Drop alias is metadata alter, in other case mutation is required.
@@ -1065,10 +1095,10 @@ std::optional AlterCommand::tryConvertToMutationCommand(Storage
result.predicate = nullptr;
}
- else if (type == DROP_STATISTIC)
+ else if (type == DROP_STATISTICS)
{
- result.type = MutationCommand::Type::DROP_STATISTIC;
- result.statistic_columns = statistic_columns;
+ result.type = MutationCommand::Type::DROP_STATISTICS;
+ result.statistics_columns = statistics_columns;
if (clear)
result.clear = true;
diff --git a/src/Storages/AlterCommands.h b/src/Storages/AlterCommands.h
index 46abffab8ad..a91bac10214 100644
--- a/src/Storages/AlterCommands.h
+++ b/src/Storages/AlterCommands.h
@@ -38,8 +38,9 @@ struct AlterCommand
DROP_CONSTRAINT,
ADD_PROJECTION,
DROP_PROJECTION,
- ADD_STATISTIC,
- DROP_STATISTIC,
+ ADD_STATISTICS,
+ DROP_STATISTICS,
+ MODIFY_STATISTICS,
MODIFY_TTL,
MODIFY_SETTING,
RESET_SETTING,
@@ -123,9 +124,9 @@ struct AlterCommand
/// For ADD/DROP PROJECTION
String projection_name;
- ASTPtr statistic_decl = nullptr;
- std::vector statistic_columns;
- String statistic_type;
+ ASTPtr statistics_decl = nullptr;
+ std::vector statistics_columns;
+ std::vector statistics_types;
/// For MODIFY TTL
ASTPtr ttl = nullptr;
diff --git a/src/Storages/ColumnDependency.h b/src/Storages/ColumnDependency.h
index b9088dd0227..dcbda7a4b86 100644
--- a/src/Storages/ColumnDependency.h
+++ b/src/Storages/ColumnDependency.h
@@ -26,8 +26,8 @@ struct ColumnDependency
/// TTL is set for @column_name.
TTL_TARGET,
- /// Exists any statistic, that requires @column_name
- STATISTIC,
+ /// Exists any statistics, that requires @column_name
+ STATISTICS,
};
ColumnDependency(const String & column_name_, Kind kind_)
diff --git a/src/Storages/ColumnsDescription.cpp b/src/Storages/ColumnsDescription.cpp
index a8869970300..69e39323219 100644
--- a/src/Storages/ColumnsDescription.cpp
+++ b/src/Storages/ColumnsDescription.cpp
@@ -72,7 +72,7 @@ ColumnDescription & ColumnDescription::operator=(const ColumnDescription & other
codec = other.codec ? other.codec->clone() : nullptr;
settings = other.settings;
ttl = other.ttl ? other.ttl->clone() : nullptr;
- stat = other.stat;
+ statistics = other.statistics;
return *this;
}
@@ -95,7 +95,7 @@ ColumnDescription & ColumnDescription::operator=(ColumnDescription && other) noe
ttl = other.ttl ? other.ttl->clone() : nullptr;
other.ttl.reset();
- stat = std::move(other.stat);
+ statistics = std::move(other.statistics);
return *this;
}
@@ -107,7 +107,7 @@ bool ColumnDescription::operator==(const ColumnDescription & other) const
return name == other.name
&& type->equals(*other.type)
&& default_desc == other.default_desc
- && stat == other.stat
+ && statistics == other.statistics
&& ast_to_str(codec) == ast_to_str(other.codec)
&& settings == other.settings
&& ast_to_str(ttl) == ast_to_str(other.ttl);
@@ -154,10 +154,10 @@ void ColumnDescription::writeText(WriteBuffer & buf) const
DB::writeText(")", buf);
}
- if (stat)
+ if (!statistics.empty())
{
writeChar('\t', buf);
- writeEscapedString(queryToString(stat->ast), buf);
+ writeEscapedString(queryToString(statistics.getAST()), buf);
}
if (ttl)
diff --git a/src/Storages/ColumnsDescription.h b/src/Storages/ColumnsDescription.h
index 79e43d0a4e4..f0760160f0a 100644
--- a/src/Storages/ColumnsDescription.h
+++ b/src/Storages/ColumnsDescription.h
@@ -89,7 +89,7 @@ struct ColumnDescription
ASTPtr codec;
SettingsChanges settings;
ASTPtr ttl;
- std::optional stat;
+ ColumnStatisticsDescription statistics;
ColumnDescription() = default;
ColumnDescription(const ColumnDescription & other) { *this = other; }
diff --git a/src/Storages/IStorage.cpp b/src/Storages/IStorage.cpp
index 9afafe9f52b..1f7ac23eb82 100644
--- a/src/Storages/IStorage.cpp
+++ b/src/Storages/IStorage.cpp
@@ -12,7 +12,7 @@
#include
#include
#include
-#include
+#include
#include
#include
@@ -236,7 +236,7 @@ StorageID IStorage::getStorageID() const
return storage_id;
}
-ConditionEstimator IStorage::getConditionEstimatorByPredicate(const StorageSnapshotPtr &, const ActionsDAGPtr &, ContextPtr) const
+ConditionSelectivityEstimator IStorage::getConditionSelectivityEstimatorByPredicate(const StorageSnapshotPtr &, const ActionsDAGPtr &, ContextPtr) const
{
return {};
}
diff --git a/src/Storages/IStorage.h b/src/Storages/IStorage.h
index 9d6b3457a24..98afd844046 100644
--- a/src/Storages/IStorage.h
+++ b/src/Storages/IStorage.h
@@ -68,7 +68,7 @@ using DatabaseAndTableName = std::pair;
class BackupEntriesCollector;
class RestorerFromBackup;
-class ConditionEstimator;
+class ConditionSelectivityEstimator;
struct ColumnSize
{
@@ -135,7 +135,7 @@ public:
/// Returns true if the storage supports queries with the PREWHERE section.
virtual bool supportsPrewhere() const { return false; }
- virtual ConditionEstimator getConditionEstimatorByPredicate(const StorageSnapshotPtr &, const ActionsDAGPtr &, ContextPtr) const;
+ virtual ConditionSelectivityEstimator getConditionSelectivityEstimatorByPredicate(const StorageSnapshotPtr &, const ActionsDAGPtr &, ContextPtr) const;
/// Returns which columns supports PREWHERE, or empty std::nullopt if all columns is supported.
/// This is needed for engines whose aggregates data from multiple tables, like Merge.
diff --git a/src/Storages/MergeTree/IMergeTreeDataPart.cpp b/src/Storages/MergeTree/IMergeTreeDataPart.cpp
index c276361559c..4c8f1240cf5 100644
--- a/src/Storages/MergeTree/IMergeTreeDataPart.cpp
+++ b/src/Storages/MergeTree/IMergeTreeDataPart.cpp
@@ -673,16 +673,16 @@ String IMergeTreeDataPart::getColumnNameWithMinimumCompressedSize(bool with_subc
return *minimum_size_column;
}
-Statistics IMergeTreeDataPart::loadStatistics() const
+ColumnsStatistics IMergeTreeDataPart::loadStatistics() const
{
const auto & metadata_snaphost = storage.getInMemoryMetadata();
auto total_statistics = MergeTreeStatisticsFactory::instance().getMany(metadata_snaphost.getColumns());
- Statistics result;
+ ColumnsStatistics result;
for (auto & stat : total_statistics)
{
- String file_name = stat->getFileName() + STAT_FILE_SUFFIX;
+ String file_name = stat->getFileName() + STATS_FILE_SUFFIX;
String file_path = fs::path(getDataPartStorage().getRelativePath()) / file_name;
if (!metadata_manager->exists(file_name))
diff --git a/src/Storages/MergeTree/IMergeTreeDataPart.h b/src/Storages/MergeTree/IMergeTreeDataPart.h
index bd3814bf415..c9b3ec26df0 100644
--- a/src/Storages/MergeTree/IMergeTreeDataPart.h
+++ b/src/Storages/MergeTree/IMergeTreeDataPart.h
@@ -171,7 +171,7 @@ public:
void remove();
- Statistics loadStatistics() const;
+ ColumnsStatistics loadStatistics() const;
/// Initialize columns (from columns.txt if exists, or create from column files if not).
/// Load various metadata into memory: checksums from checksums.txt, index if required, etc.
diff --git a/src/Storages/MergeTree/IMergeTreeDataPartWriter.cpp b/src/Storages/MergeTree/IMergeTreeDataPartWriter.cpp
index 891ba1b9660..6152da78395 100644
--- a/src/Storages/MergeTree/IMergeTreeDataPartWriter.cpp
+++ b/src/Storages/MergeTree/IMergeTreeDataPartWriter.cpp
@@ -119,7 +119,7 @@ MergeTreeDataPartWriterPtr createMergeTreeDataPartCompactWriter(
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & writer_settings,
@@ -136,7 +136,7 @@ MergeTreeDataPartWriterPtr createMergeTreeDataPartWideWriter(
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & writer_settings,
@@ -156,7 +156,7 @@ MergeTreeDataPartWriterPtr createMergeTreeDataPartWriter(
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & writer_settings,
diff --git a/src/Storages/MergeTree/IMergeTreeDataPartWriter.h b/src/Storages/MergeTree/IMergeTreeDataPartWriter.h
index f04beb37ebb..d9e9a433827 100644
--- a/src/Storages/MergeTree/IMergeTreeDataPartWriter.h
+++ b/src/Storages/MergeTree/IMergeTreeDataPartWriter.h
@@ -84,7 +84,7 @@ MergeTreeDataPartWriterPtr createMergeTreeDataPartWriter(
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns_,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & writer_settings,
diff --git a/src/Storages/MergeTree/MergeTask.cpp b/src/Storages/MergeTree/MergeTask.cpp
index f1f856da3a2..62bc3b42d1f 100644
--- a/src/Storages/MergeTree/MergeTask.cpp
+++ b/src/Storages/MergeTree/MergeTask.cpp
@@ -658,7 +658,7 @@ void MergeTask::VerticalMergeStage::prepareVerticalMergeForOneColumn() const
/// because all of them were already recalculated and written
/// as key part of vertical merge
std::vector{},
- std::vector{}, /// TODO: think about it
+ ColumnsStatistics{}, /// TODO(hanfei)
&global_ctx->written_offset_columns,
global_ctx->to->getIndexGranularity());
diff --git a/src/Storages/MergeTree/MergeTreeData.cpp b/src/Storages/MergeTree/MergeTreeData.cpp
index 32d08d1562e..14a310364dc 100644
--- a/src/Storages/MergeTree/MergeTreeData.cpp
+++ b/src/Storages/MergeTree/MergeTreeData.cpp
@@ -73,7 +73,7 @@
#include
#include
#include
-#include
+#include
#include
#include
#include
@@ -471,10 +471,10 @@ StoragePolicyPtr MergeTreeData::getStoragePolicy() const
return storage_policy;
}
-ConditionEstimator MergeTreeData::getConditionEstimatorByPredicate(
+ConditionSelectivityEstimator MergeTreeData::getConditionSelectivityEstimatorByPredicate(
const StorageSnapshotPtr & storage_snapshot, const ActionsDAGPtr & filter_dag, ContextPtr local_context) const
{
- if (!local_context->getSettings().allow_statistic_optimize)
+ if (!local_context->getSettings().allow_statistics_optimize)
return {};
const auto & parts = assert_cast(*storage_snapshot->data).parts;
@@ -486,23 +486,29 @@ ConditionEstimator MergeTreeData::getConditionEstimatorByPredicate(
ASTPtr expression_ast;
- ConditionEstimator result;
+ ConditionSelectivityEstimator result;
PartitionPruner partition_pruner(storage_snapshot->metadata, filter_dag, local_context);
if (partition_pruner.isUseless())
{
/// Read all partitions.
for (const auto & part : parts)
+ try
{
auto stats = part->loadStatistics();
/// TODO: We only have one stats file for every part.
for (const auto & stat : stats)
result.merge(part->info.getPartNameV1(), part->rows_count, stat);
}
+ catch (...)
+ {
+ tryLogCurrentException(log, fmt::format("while loading statistics on part {}", part->info.getPartNameV1()));
+ }
}
else
{
for (const auto & part : parts)
+ try
{
if (!partition_pruner.canBePruned(*part))
{
@@ -511,6 +517,10 @@ ConditionEstimator MergeTreeData::getConditionEstimatorByPredicate(
result.merge(part->info.getPartNameV1(), part->rows_count, stat);
}
}
+ catch (...)
+ {
+ tryLogCurrentException(log, fmt::format("while loading statistics on part {}", part->info.getPartNameV1()));
+ }
}
return result;
@@ -691,8 +701,8 @@ void MergeTreeData::checkProperties(
for (const auto & col : new_metadata.columns)
{
- if (col.stat)
- MergeTreeStatisticsFactory::instance().validate(*col.stat, col.type);
+ if (!col.statistics.empty())
+ MergeTreeStatisticsFactory::instance().validate(col.statistics, col.type);
}
checkKeyExpression(*new_sorting_key.expression, new_sorting_key.sample_block, "Sorting", allow_nullable_key_);
@@ -3469,13 +3479,13 @@ void MergeTreeData::checkAlterIsPossible(const AlterCommands & commands, Context
new_metadata.getColumns().getPhysical(command.column_name));
const auto & old_column = old_metadata.getColumns().get(command.column_name);
- if (old_column.stat)
+ if (!old_column.statistics.empty())
{
const auto & new_column = new_metadata.getColumns().get(command.column_name);
if (!old_column.type->equals(*new_column.type))
throw Exception(ErrorCodes::ALTER_OF_COLUMN_IS_FORBIDDEN,
- "ALTER types of column {} with statistic is not not safe "
- "because it can change the representation of statistic",
+ "ALTER types of column {} with statistics is not not safe "
+ "because it can change the representation of statistics",
backQuoteIfNeed(command.column_name));
}
}
@@ -8514,7 +8524,7 @@ std::pair MergeTreeData::createE
const auto & index_factory = MergeTreeIndexFactory::instance();
MergedBlockOutputStream out(new_data_part, metadata_snapshot, columns,
index_factory.getMany(metadata_snapshot->getSecondaryIndices()),
- Statistics{},
+ ColumnsStatistics{},
compression_codec, txn ? txn->tid : Tx::PrehistoricTID);
bool sync_on_insert = settings->fsync_after_insert;
diff --git a/src/Storages/MergeTree/MergeTreeData.h b/src/Storages/MergeTree/MergeTreeData.h
index 440daaf6ced..c6f736a4afd 100644
--- a/src/Storages/MergeTree/MergeTreeData.h
+++ b/src/Storages/MergeTree/MergeTreeData.h
@@ -426,7 +426,7 @@ public:
bool supportsPrewhere() const override { return true; }
- ConditionEstimator getConditionEstimatorByPredicate(const StorageSnapshotPtr &, const ActionsDAGPtr &, ContextPtr) const override;
+ ConditionSelectivityEstimator getConditionSelectivityEstimatorByPredicate(const StorageSnapshotPtr &, const ActionsDAGPtr &, ContextPtr) const override;
bool supportsFinal() const override;
diff --git a/src/Storages/MergeTree/MergeTreeDataPartCompact.cpp b/src/Storages/MergeTree/MergeTreeDataPartCompact.cpp
index 4a160e5e229..d628fd6b529 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartCompact.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataPartCompact.cpp
@@ -59,7 +59,7 @@ MergeTreeDataPartWriterPtr createMergeTreeDataPartCompactWriter(
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & writer_settings,
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWide.cpp b/src/Storages/MergeTree/MergeTreeDataPartWide.cpp
index 149f86cef00..379c465a409 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWide.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataPartWide.cpp
@@ -64,7 +64,7 @@ MergeTreeDataPartWriterPtr createMergeTreeDataPartWideWriter(
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & writer_settings,
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.cpp b/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.cpp
index fb0f0ba9154..21d046c76f2 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.cpp
@@ -20,7 +20,7 @@ MergeTreeDataPartWriterCompact::MergeTreeDataPartWriterCompact(
const StorageMetadataPtr & metadata_snapshot_,
const VirtualsDescriptionPtr & virtual_columns_,
const std::vector & indices_to_recalc_,
- const Statistics & stats_to_recalc,
+ const ColumnsStatistics & stats_to_recalc,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & settings_,
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.h b/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.h
index a5527b74e69..b440a37222d 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.h
+++ b/src/Storages/MergeTree/MergeTreeDataPartWriterCompact.h
@@ -21,7 +21,7 @@ public:
const StorageMetadataPtr & metadata_snapshot_,
const VirtualsDescriptionPtr & virtual_columns_,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc,
+ const ColumnsStatistics & stats_to_recalc,
const String & marks_file_extension,
const CompressionCodecPtr & default_codec,
const MergeTreeWriterSettings & settings,
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.cpp b/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.cpp
index bcf51bfcd3d..a576720294f 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.cpp
@@ -150,7 +150,7 @@ MergeTreeDataPartWriterOnDisk::MergeTreeDataPartWriterOnDisk(
const StorageMetadataPtr & metadata_snapshot_,
const VirtualsDescriptionPtr & virtual_columns_,
const MergeTreeIndices & indices_to_recalc_,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & settings_,
@@ -265,7 +265,7 @@ void MergeTreeDataPartWriterOnDisk::initStatistics()
stats_streams.emplace_back(std::make_unique>(
stats_name,
data_part_storage,
- stats_name, STAT_FILE_SUFFIX,
+ stats_name, STATS_FILE_SUFFIX,
default_codec, settings.max_compress_block_size,
settings.query_write_settings));
}
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.h b/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.h
index 0c31cabc8c4..bdf0fdb7f32 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.h
+++ b/src/Storages/MergeTree/MergeTreeDataPartWriterOnDisk.h
@@ -111,7 +111,7 @@ public:
const StorageMetadataPtr & metadata_snapshot_,
const VirtualsDescriptionPtr & virtual_columns_,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension,
const CompressionCodecPtr & default_codec,
const MergeTreeWriterSettings & settings,
@@ -155,7 +155,7 @@ protected:
const MergeTreeIndices skip_indices;
- const Statistics stats;
+ const ColumnsStatistics stats;
std::vector stats_streams;
const String marks_file_extension;
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWriterWide.cpp b/src/Storages/MergeTree/MergeTreeDataPartWriterWide.cpp
index afa14d8a98a..5ba326cef0c 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWriterWide.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataPartWriterWide.cpp
@@ -86,7 +86,7 @@ MergeTreeDataPartWriterWide::MergeTreeDataPartWriterWide(
const StorageMetadataPtr & metadata_snapshot_,
const VirtualsDescriptionPtr & virtual_columns_,
const std::vector & indices_to_recalc_,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension_,
const CompressionCodecPtr & default_codec_,
const MergeTreeWriterSettings & settings_,
diff --git a/src/Storages/MergeTree/MergeTreeDataPartWriterWide.h b/src/Storages/MergeTree/MergeTreeDataPartWriterWide.h
index 9d18ac76880..ab86ed27c7e 100644
--- a/src/Storages/MergeTree/MergeTreeDataPartWriterWide.h
+++ b/src/Storages/MergeTree/MergeTreeDataPartWriterWide.h
@@ -31,7 +31,7 @@ public:
const StorageMetadataPtr & metadata_snapshot,
const VirtualsDescriptionPtr & virtual_columns_,
const std::vector & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
const String & marks_file_extension,
const CompressionCodecPtr & default_codec,
const MergeTreeWriterSettings & settings,
diff --git a/src/Storages/MergeTree/MergeTreeDataSelectExecutor.cpp b/src/Storages/MergeTree/MergeTreeDataSelectExecutor.cpp
index 11058c542a6..2e287ff3042 100644
--- a/src/Storages/MergeTree/MergeTreeDataSelectExecutor.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataSelectExecutor.cpp
@@ -760,9 +760,16 @@ RangesInDataParts MergeTreeDataSelectExecutor::filterPartsByPrimaryKeyAndSkipInd
CurrentMetrics::MergeTreeDataSelectExecutorThreadsScheduled,
num_threads);
+
+ /// Instances of ThreadPool "borrow" threads from the global thread pool.
+ /// We intentionally use scheduleOrThrow here to avoid a deadlock.
+ /// For example, queries can already be running with threads from the
+ /// global pool, and if we saturate max_thread_pool_size whilst requesting
+ /// more in this loop, queries will block infinitely.
+ /// So we wait until lock_acquire_timeout, and then raise an exception.
for (size_t part_index = 0; part_index < parts.size(); ++part_index)
{
- pool.scheduleOrThrowOnError([&, part_index, thread_group = CurrentThread::getGroup()]
+ pool.scheduleOrThrow([&, part_index, thread_group = CurrentThread::getGroup()]
{
setThreadName("MergeTreeIndex");
@@ -774,7 +781,7 @@ RangesInDataParts MergeTreeDataSelectExecutor::filterPartsByPrimaryKeyAndSkipInd
CurrentThread::attachToGroupIfDetached(thread_group);
process_part(part_index);
- });
+ }, Priority{}, context->getSettingsRef().lock_acquire_timeout.totalMicroseconds());
}
pool.wait();
diff --git a/src/Storages/MergeTree/MergeTreeDataWriter.cpp b/src/Storages/MergeTree/MergeTreeDataWriter.cpp
index 04182062b12..8e304936747 100644
--- a/src/Storages/MergeTree/MergeTreeDataWriter.cpp
+++ b/src/Storages/MergeTree/MergeTreeDataWriter.cpp
@@ -469,7 +469,7 @@ MergeTreeDataWriter::TemporaryPart MergeTreeDataWriter::writeTempPartImpl(
if (context->getSettingsRef().materialize_skip_indexes_on_insert)
indices = MergeTreeIndexFactory::instance().getMany(metadata_snapshot->getSecondaryIndices());
- Statistics statistics;
+ ColumnsStatistics statistics;
if (context->getSettingsRef().materialize_statistics_on_insert)
statistics = MergeTreeStatisticsFactory::instance().getMany(metadata_snapshot->getColumns());
@@ -754,7 +754,8 @@ MergeTreeDataWriter::TemporaryPart MergeTreeDataWriter::writeProjectionPartImpl(
metadata_snapshot,
columns,
MergeTreeIndices{},
- Statistics{}, /// TODO(hanfei): It should be helpful to write statistics for projection result.
+ /// TODO(hanfei): It should be helpful to write statistics for projection result.
+ ColumnsStatistics{},
compression_codec,
Tx::PrehistoricTID,
false, false, data.getContext()->getWriteSettings());
diff --git a/src/Storages/MergeTree/MergeTreeIndexBloomFilterText.cpp b/src/Storages/MergeTree/MergeTreeIndexBloomFilterText.cpp
index 6f46ee0c184..8cf58687125 100644
--- a/src/Storages/MergeTree/MergeTreeIndexBloomFilterText.cpp
+++ b/src/Storages/MergeTree/MergeTreeIndexBloomFilterText.cpp
@@ -566,7 +566,7 @@ bool MergeTreeConditionBloomFilterText::traverseTreeEquals(
out.function = RPNElement::FUNCTION_EQUALS;
out.bloom_filter = std::make_unique(params);
const auto & value = const_value.get();
- token_extractor->stringToBloomFilter(value.data(), value.size(), *out.bloom_filter);
+ token_extractor->substringToBloomFilter(value.data(), value.size(), *out.bloom_filter, true, false);
return true;
}
else if (function_name == "endsWith")
@@ -575,7 +575,7 @@ bool MergeTreeConditionBloomFilterText::traverseTreeEquals(
out.function = RPNElement::FUNCTION_EQUALS;
out.bloom_filter = std::make_unique(params);
const auto & value = const_value.get();
- token_extractor->stringToBloomFilter(value.data(), value.size(), *out.bloom_filter);
+ token_extractor->substringToBloomFilter(value.data(), value.size(), *out.bloom_filter, false, true);
return true;
}
else if (function_name == "multiSearchAny"
@@ -596,7 +596,15 @@ bool MergeTreeConditionBloomFilterText::traverseTreeEquals(
bloom_filters.back().emplace_back(params);
const auto & value = element.get();
- token_extractor->stringToBloomFilter(value.data(), value.size(), bloom_filters.back().back());
+
+ if (function_name == "multiSearchAny")
+ {
+ token_extractor->substringToBloomFilter(value.data(), value.size(), bloom_filters.back().back(), false, false);
+ }
+ else
+ {
+ token_extractor->stringToBloomFilter(value.data(), value.size(), bloom_filters.back().back());
+ }
}
out.set_bloom_filters = std::move(bloom_filters);
return true;
@@ -625,12 +633,12 @@ bool MergeTreeConditionBloomFilterText::traverseTreeEquals(
for (const auto & alternative : alternatives)
{
bloom_filters.back().emplace_back(params);
- token_extractor->stringToBloomFilter(alternative.data(), alternative.size(), bloom_filters.back().back());
+ token_extractor->substringToBloomFilter(alternative.data(), alternative.size(), bloom_filters.back().back(), false, false);
}
out.set_bloom_filters = std::move(bloom_filters);
}
else
- token_extractor->stringToBloomFilter(required_substring.data(), required_substring.size(), *out.bloom_filter);
+ token_extractor->substringToBloomFilter(required_substring.data(), required_substring.size(), *out.bloom_filter, false, false);
return true;
}
diff --git a/src/Storages/MergeTree/MergeTreeIndexFullText.cpp b/src/Storages/MergeTree/MergeTreeIndexFullText.cpp
index af9ee710f88..47ce24b91eb 100644
--- a/src/Storages/MergeTree/MergeTreeIndexFullText.cpp
+++ b/src/Storages/MergeTree/MergeTreeIndexFullText.cpp
@@ -595,7 +595,7 @@ bool MergeTreeConditionFullText::traverseASTEquals(
out.function = RPNElement::FUNCTION_EQUALS;
out.gin_filter = std::make_unique(params);
const auto & value = const_value.get();
- token_extractor->stringToGinFilter(value.data(), value.size(), *out.gin_filter);
+ token_extractor->substringToGinFilter(value.data(), value.size(), *out.gin_filter, true, false);
return true;
}
else if (function_name == "endsWith")
@@ -604,7 +604,7 @@ bool MergeTreeConditionFullText::traverseASTEquals(
out.function = RPNElement::FUNCTION_EQUALS;
out.gin_filter = std::make_unique(params);
const auto & value = const_value.get();
- token_extractor->stringToGinFilter(value.data(), value.size(), *out.gin_filter);
+ token_extractor->substringToGinFilter(value.data(), value.size(), *out.gin_filter, false, true);
return true;
}
else if (function_name == "multiSearchAny")
@@ -622,7 +622,7 @@ bool MergeTreeConditionFullText::traverseASTEquals(
gin_filters.back().emplace_back(params);
const auto & value = element.get();
- token_extractor->stringToGinFilter(value.data(), value.size(), gin_filters.back().back());
+ token_extractor->substringToGinFilter(value.data(), value.size(), gin_filters.back().back(), false, false);
}
out.set_gin_filters = std::move(gin_filters);
return true;
@@ -650,14 +650,14 @@ bool MergeTreeConditionFullText::traverseASTEquals(
for (const auto & alternative : alternatives)
{
gin_filters.back().emplace_back(params);
- token_extractor->stringToGinFilter(alternative.data(), alternative.size(), gin_filters.back().back());
+ token_extractor->substringToGinFilter(alternative.data(), alternative.size(), gin_filters.back().back(), false, false);
}
out.set_gin_filters = std::move(gin_filters);
}
else
{
out.gin_filter = std::make_unique(params);
- token_extractor->stringToGinFilter(required_substring.data(), required_substring.size(), *out.gin_filter);
+ token_extractor->substringToGinFilter(required_substring.data(), required_substring.size(), *out.gin_filter, false, false);
}
return true;
@@ -742,6 +742,7 @@ bool MergeTreeConditionFullText::tryPrepareSetGinFilter(
MergeTreeIndexGranulePtr MergeTreeIndexFullText::createIndexGranule() const
{
+ /// ------
/// Index type 'inverted' was renamed to 'full_text' in May 2024.
/// Tables with old indexes can be loaded during a transition period. We still want let users know that they should drop existing
/// indexes and re-create them. Function `createIndexGranule` is called whenever the index is used by queries. Reject the query if we
@@ -749,6 +750,7 @@ MergeTreeIndexGranulePtr MergeTreeIndexFullText::createIndexGranule() const
/// TODO: remove this at the end of 2024.
if (index.type == INVERTED_INDEX_NAME)
throw Exception(ErrorCodes::ILLEGAL_INDEX, "Indexes of type 'inverted' are no longer supported. Please drop and recreate the index as type 'full-text'");
+ /// ------
return std::make_shared(index.name, index.column_names.size(), params);
}
diff --git a/src/Storages/MergeTree/MergeTreeWhereOptimizer.cpp b/src/Storages/MergeTree/MergeTreeWhereOptimizer.cpp
index 3844ac18268..a9a5fddace4 100644
--- a/src/Storages/MergeTree/MergeTreeWhereOptimizer.cpp
+++ b/src/Storages/MergeTree/MergeTreeWhereOptimizer.cpp
@@ -53,7 +53,7 @@ static Int64 findMinPosition(const NameSet & condition_table_columns, const Name
MergeTreeWhereOptimizer::MergeTreeWhereOptimizer(
std::unordered_map column_sizes_,
const StorageMetadataPtr & metadata_snapshot,
- const ConditionEstimator & estimator_,
+ const ConditionSelectivityEstimator & estimator_,
const Names & queried_columns_,
const std::optional & supported_columns_,
LoggerPtr log_)
@@ -92,7 +92,7 @@ void MergeTreeWhereOptimizer::optimize(SelectQueryInfo & select_query_info, cons
where_optimizer_context.move_all_conditions_to_prewhere = context->getSettingsRef().move_all_conditions_to_prewhere;
where_optimizer_context.move_primary_key_columns_to_end_of_prewhere = context->getSettingsRef().move_primary_key_columns_to_end_of_prewhere;
where_optimizer_context.is_final = select.final();
- where_optimizer_context.use_statistic = context->getSettingsRef().allow_statistic_optimize;
+ where_optimizer_context.use_statistics = context->getSettingsRef().allow_statistics_optimize;
RPNBuilderTreeContext tree_context(context, std::move(block_with_constants), {} /*prepared_sets*/);
RPNBuilderTreeNode node(select.where().get(), tree_context);
@@ -123,7 +123,7 @@ MergeTreeWhereOptimizer::FilterActionsOptimizeResult MergeTreeWhereOptimizer::op
where_optimizer_context.move_all_conditions_to_prewhere = context->getSettingsRef().move_all_conditions_to_prewhere;
where_optimizer_context.move_primary_key_columns_to_end_of_prewhere = context->getSettingsRef().move_primary_key_columns_to_end_of_prewhere;
where_optimizer_context.is_final = is_final;
- where_optimizer_context.use_statistic = context->getSettingsRef().allow_statistic_optimize;
+ where_optimizer_context.use_statistics = context->getSettingsRef().allow_statistics_optimize;
RPNBuilderTreeContext tree_context(context);
RPNBuilderTreeNode node(&filter_dag->findInOutputs(filter_column_name), tree_context);
@@ -273,15 +273,17 @@ void MergeTreeWhereOptimizer::analyzeImpl(Conditions & res, const RPNBuilderTree
/// Do not move conditions involving all queried columns.
&& cond.table_columns.size() < queried_columns.size();
- if (where_optimizer_context.use_statistic)
+ if (cond.viable)
+ cond.good = isConditionGood(node, table_columns);
+
+ if (where_optimizer_context.use_statistics)
{
cond.good = cond.viable;
- cond.selectivity = estimator.estimateSelectivity(node);
- LOG_TEST(log, "Condition {} has selectivity {}", node.getColumnName(), cond.selectivity);
- }
- else if (cond.viable)
- {
- cond.good = isConditionGood(node, table_columns);
+
+ cond.estimated_row_count = estimator.estimateRowCount(node);
+
+ if (node.getASTNode() != nullptr)
+ LOG_DEBUG(log, "Condition {} has estimated row count {}", node.getASTNode()->dumpTree(), cond.estimated_row_count);
}
if (where_optimizer_context.move_primary_key_columns_to_end_of_prewhere)
diff --git a/src/Storages/MergeTree/MergeTreeWhereOptimizer.h b/src/Storages/MergeTree/MergeTreeWhereOptimizer.h
index 6c5ff29bc76..ba6b4660924 100644
--- a/src/Storages/MergeTree/MergeTreeWhereOptimizer.h
+++ b/src/Storages/MergeTree/MergeTreeWhereOptimizer.h
@@ -4,7 +4,7 @@
#include
#include
#include
-#include
+#include
#include
@@ -38,7 +38,7 @@ public:
MergeTreeWhereOptimizer(
std::unordered_map column_sizes_,
const StorageMetadataPtr & metadata_snapshot,
- const ConditionEstimator & estimator_,
+ const ConditionSelectivityEstimator & estimator_,
const Names & queried_columns_,
const std::optional & supported_columns_,
LoggerPtr log_);
@@ -76,7 +76,7 @@ private:
bool good = false;
/// the lower the better
- Float64 selectivity = 1.0;
+ Float64 estimated_row_count = 0;
/// Does the condition contain primary key column?
/// If so, it is better to move it further to the end of PREWHERE chain depending on minimal position in PK of any
@@ -85,7 +85,7 @@ private:
auto tuple() const
{
- return std::make_tuple(!viable, !good, -min_position_in_primary_key, selectivity, columns_size, table_columns.size());
+ return std::make_tuple(!viable, !good, -min_position_in_primary_key, estimated_row_count, columns_size, table_columns.size());
}
/// Is condition a better candidate for moving to PREWHERE?
@@ -104,7 +104,7 @@ private:
bool move_all_conditions_to_prewhere = false;
bool move_primary_key_columns_to_end_of_prewhere = false;
bool is_final = false;
- bool use_statistic = false;
+ bool use_statistics = false;
};
struct OptimizeResult
@@ -147,7 +147,7 @@ private:
static NameSet determineArrayJoinedNames(const ASTSelectQuery & select);
- const ConditionEstimator estimator;
+ const ConditionSelectivityEstimator estimator;
const NameSet table_columns;
const Names queried_columns;
diff --git a/src/Storages/MergeTree/MergedBlockOutputStream.cpp b/src/Storages/MergeTree/MergedBlockOutputStream.cpp
index c5799fab09f..164658c914e 100644
--- a/src/Storages/MergeTree/MergedBlockOutputStream.cpp
+++ b/src/Storages/MergeTree/MergedBlockOutputStream.cpp
@@ -20,7 +20,7 @@ MergedBlockOutputStream::MergedBlockOutputStream(
const StorageMetadataPtr & metadata_snapshot_,
const NamesAndTypesList & columns_list_,
const MergeTreeIndices & skip_indices,
- const Statistics & statistics,
+ const ColumnsStatistics & statistics,
CompressionCodecPtr default_codec_,
TransactionID tid,
bool reset_columns_,
diff --git a/src/Storages/MergeTree/MergedBlockOutputStream.h b/src/Storages/MergeTree/MergedBlockOutputStream.h
index c1e3d75fefc..e212fe5bb5a 100644
--- a/src/Storages/MergeTree/MergedBlockOutputStream.h
+++ b/src/Storages/MergeTree/MergedBlockOutputStream.h
@@ -20,7 +20,7 @@ public:
const StorageMetadataPtr & metadata_snapshot_,
const NamesAndTypesList & columns_list_,
const MergeTreeIndices & skip_indices,
- const Statistics & statistics,
+ const ColumnsStatistics & statistics,
CompressionCodecPtr default_codec_,
TransactionID tid,
bool reset_columns_ = false,
diff --git a/src/Storages/MergeTree/MergedColumnOnlyOutputStream.cpp b/src/Storages/MergeTree/MergedColumnOnlyOutputStream.cpp
index 674a9bd498f..54d177943d0 100644
--- a/src/Storages/MergeTree/MergedColumnOnlyOutputStream.cpp
+++ b/src/Storages/MergeTree/MergedColumnOnlyOutputStream.cpp
@@ -16,7 +16,7 @@ MergedColumnOnlyOutputStream::MergedColumnOnlyOutputStream(
const Block & header_,
CompressionCodecPtr default_codec,
const MergeTreeIndices & indices_to_recalc,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
WrittenOffsetColumns * offset_columns_,
const MergeTreeIndexGranularity & index_granularity,
const MergeTreeIndexGranularityInfo * index_granularity_info)
diff --git a/src/Storages/MergeTree/MergedColumnOnlyOutputStream.h b/src/Storages/MergeTree/MergedColumnOnlyOutputStream.h
index ad3cabe459e..16a54ff33b6 100644
--- a/src/Storages/MergeTree/MergedColumnOnlyOutputStream.h
+++ b/src/Storages/MergeTree/MergedColumnOnlyOutputStream.h
@@ -20,7 +20,7 @@ public:
const Block & header_,
CompressionCodecPtr default_codec_,
const MergeTreeIndices & indices_to_recalc_,
- const Statistics & stats_to_recalc_,
+ const ColumnsStatistics & stats_to_recalc_,
WrittenOffsetColumns * offset_columns_ = nullptr,
const MergeTreeIndexGranularity & index_granularity = {},
const MergeTreeIndexGranularityInfo * index_granularity_info_ = nullptr);
diff --git a/src/Storages/MergeTree/MutateTask.cpp b/src/Storages/MergeTree/MutateTask.cpp
index 1828b8a7eeb..1318563e469 100644
--- a/src/Storages/MergeTree/MutateTask.cpp
+++ b/src/Storages/MergeTree/MutateTask.cpp
@@ -130,7 +130,7 @@ static void splitAndModifyMutationCommands(
}
}
if (command.type == MutationCommand::Type::MATERIALIZE_INDEX
- || command.type == MutationCommand::Type::MATERIALIZE_STATISTIC
+ || command.type == MutationCommand::Type::MATERIALIZE_STATISTICS
|| command.type == MutationCommand::Type::MATERIALIZE_PROJECTION
|| command.type == MutationCommand::Type::MATERIALIZE_TTL
|| command.type == MutationCommand::Type::DELETE
@@ -143,7 +143,7 @@ static void splitAndModifyMutationCommands(
}
else if (command.type == MutationCommand::Type::DROP_INDEX
|| command.type == MutationCommand::Type::DROP_PROJECTION
- || command.type == MutationCommand::Type::DROP_STATISTIC)
+ || command.type == MutationCommand::Type::DROP_STATISTICS)
{
for_file_renames.push_back(command);
}
@@ -258,7 +258,7 @@ static void splitAndModifyMutationCommands(
for_interpreter.push_back(command);
}
else if (command.type == MutationCommand::Type::MATERIALIZE_INDEX
- || command.type == MutationCommand::Type::MATERIALIZE_STATISTIC
+ || command.type == MutationCommand::Type::MATERIALIZE_STATISTICS
|| command.type == MutationCommand::Type::MATERIALIZE_PROJECTION
|| command.type == MutationCommand::Type::MATERIALIZE_TTL
|| command.type == MutationCommand::Type::DELETE
@@ -269,7 +269,7 @@ static void splitAndModifyMutationCommands(
}
else if (command.type == MutationCommand::Type::DROP_INDEX
|| command.type == MutationCommand::Type::DROP_PROJECTION
- || command.type == MutationCommand::Type::DROP_STATISTIC)
+ || command.type == MutationCommand::Type::DROP_STATISTICS)
{
for_file_renames.push_back(command);
}
@@ -532,16 +532,16 @@ static ExecuteTTLType shouldExecuteTTL(const StorageMetadataPtr & metadata_snaps
return has_ttl_expression ? ExecuteTTLType::RECALCULATE : ExecuteTTLType::NONE;
}
-static std::set getStatisticsToRecalculate(const StorageMetadataPtr & metadata_snapshot, const NameSet & materialized_stats)
+static std::set getStatisticsToRecalculate(const StorageMetadataPtr & metadata_snapshot, const NameSet & materialized_stats)
{
const auto & stats_factory = MergeTreeStatisticsFactory::instance();
- std::set stats_to_recalc;
+ std::set stats_to_recalc;
const auto & columns = metadata_snapshot->getColumns();
for (const auto & col_desc : columns)
{
- if (col_desc.stat && materialized_stats.contains(col_desc.name))
+ if (!col_desc.statistics.empty() && materialized_stats.contains(col_desc.name))
{
- stats_to_recalc.insert(stats_factory.get(*col_desc.stat));
+ stats_to_recalc.insert(stats_factory.get(col_desc.statistics));
}
}
return stats_to_recalc;
@@ -655,7 +655,7 @@ static NameSet collectFilesToSkip(
const std::set & indices_to_recalc,
const String & mrk_extension,
const std::set & projections_to_recalc,
- const std::set & stats_to_recalc)
+ const std::set & stats_to_recalc)
{
NameSet files_to_skip = source_part->getFileNamesWithoutChecksums();
@@ -683,7 +683,7 @@ static NameSet collectFilesToSkip(
files_to_skip.insert(projection->getDirectoryName());
for (const auto & stat : stats_to_recalc)
- files_to_skip.insert(stat->getFileName() + STAT_FILE_SUFFIX);
+ files_to_skip.insert(stat->getFileName() + STATS_FILE_SUFFIX);
if (isWidePart(source_part))
{
@@ -772,11 +772,11 @@ static NameToNameVector collectFilesForRenames(
if (source_part->checksums.has(command.column_name + ".proj"))
add_rename(command.column_name + ".proj", "");
}
- else if (command.type == MutationCommand::Type::DROP_STATISTIC)
+ else if (command.type == MutationCommand::Type::DROP_STATISTICS)
{
- for (const auto & statistic_column_name : command.statistic_columns)
- if (source_part->checksums.has(STAT_FILE_PREFIX + statistic_column_name + STAT_FILE_SUFFIX))
- add_rename(STAT_FILE_PREFIX + statistic_column_name + STAT_FILE_SUFFIX, "");
+ for (const auto & statistics_column_name : command.statistics_columns)
+ if (source_part->checksums.has(STATS_FILE_PREFIX + statistics_column_name + STATS_FILE_SUFFIX))
+ add_rename(STATS_FILE_PREFIX + statistics_column_name + STATS_FILE_SUFFIX, "");
}
else if (isWidePart(source_part))
{
@@ -797,9 +797,9 @@ static NameToNameVector collectFilesForRenames(
if (auto serialization = source_part->tryGetSerialization(command.column_name))
serialization->enumerateStreams(callback);
- /// if we drop a column with statistic, we should also drop the stat file.
- if (source_part->checksums.has(STAT_FILE_PREFIX + command.column_name + STAT_FILE_SUFFIX))
- add_rename(STAT_FILE_PREFIX + command.column_name + STAT_FILE_SUFFIX, "");
+ /// if we drop a column with statistics, we should also drop the stat file.
+ if (source_part->checksums.has(STATS_FILE_PREFIX + command.column_name + STATS_FILE_SUFFIX))
+ add_rename(STATS_FILE_PREFIX + command.column_name + STATS_FILE_SUFFIX, "");
}
else if (command.type == MutationCommand::Type::RENAME_COLUMN)
{
@@ -833,9 +833,9 @@ static NameToNameVector collectFilesForRenames(
if (auto serialization = source_part->tryGetSerialization(command.column_name))
serialization->enumerateStreams(callback);
- /// if we rename a column with statistic, we should also rename the stat file.
- if (source_part->checksums.has(STAT_FILE_PREFIX + command.column_name + STAT_FILE_SUFFIX))
- add_rename(STAT_FILE_PREFIX + command.column_name + STAT_FILE_SUFFIX, STAT_FILE_PREFIX + command.rename_to + STAT_FILE_SUFFIX);
+ /// if we rename a column with statistics, we should also rename the stat file.
+ if (source_part->checksums.has(STATS_FILE_PREFIX + command.column_name + STATS_FILE_SUFFIX))
+ add_rename(STATS_FILE_PREFIX + command.column_name + STATS_FILE_SUFFIX, STATS_FILE_PREFIX + command.rename_to + STATS_FILE_SUFFIX);
}
else if (command.type == MutationCommand::Type::READ_COLUMN)
{
@@ -1022,7 +1022,7 @@ struct MutationContext
IMergeTreeDataPart::MinMaxIndexPtr minmax_idx;
std::set indices_to_recalc;
- std::set stats_to_recalc;
+ std::set stats_to_recalc;
std::set projections_to_recalc;
MergeTreeData::DataPart::Checksums existing_indices_stats_checksums;
NameSet files_to_skip;
@@ -1473,12 +1473,12 @@ private:
{
if (command.type == MutationCommand::DROP_INDEX)
removed_indices.insert(command.column_name);
- else if (command.type == MutationCommand::DROP_STATISTIC)
- for (const auto & column_name : command.statistic_columns)
+ else if (command.type == MutationCommand::DROP_STATISTICS)
+ for (const auto & column_name : command.statistics_columns)
removed_stats.insert(column_name);
else if (command.type == MutationCommand::RENAME_COLUMN
- && ctx->source_part->checksums.files.contains(STAT_FILE_PREFIX + command.column_name + STAT_FILE_SUFFIX))
- renamed_stats[STAT_FILE_PREFIX + command.column_name + STAT_FILE_SUFFIX] = STAT_FILE_PREFIX + command.rename_to + STAT_FILE_SUFFIX;
+ && ctx->source_part->checksums.files.contains(STATS_FILE_PREFIX + command.column_name + STATS_FILE_SUFFIX))
+ renamed_stats[STATS_FILE_PREFIX + command.column_name + STATS_FILE_SUFFIX] = STATS_FILE_PREFIX + command.rename_to + STATS_FILE_SUFFIX;
}
bool is_full_part_storage = isFullPartStorage(ctx->new_data_part->getDataPartStorage());
@@ -1514,23 +1514,23 @@ private:
}
}
- Statistics stats_to_rewrite;
+ ColumnsStatistics stats_to_rewrite;
const auto & columns = ctx->metadata_snapshot->getColumns();
for (const auto & col : columns)
{
- if (!col.stat || removed_stats.contains(col.name))
+ if (col.statistics.empty() || removed_stats.contains(col.name))
continue;
if (ctx->materialized_statistics.contains(col.name))
{
- stats_to_rewrite.push_back(MergeTreeStatisticsFactory::instance().get(*col.stat));
+ stats_to_rewrite.push_back(MergeTreeStatisticsFactory::instance().get(col.statistics));
}
else
{
/// We do not hard-link statistics which
- /// 1. In `DROP STATISTIC` statement. It is filtered by `removed_stats`
+ /// 1. In `DROP STATISTICS` statement. It is filtered by `removed_stats`
/// 2. Not in column list anymore, including `DROP COLUMN`. It is not touched by this loop.
- String stat_file_name = STAT_FILE_PREFIX + col.name + STAT_FILE_SUFFIX;
+ String stat_file_name = STATS_FILE_PREFIX + col.name + STATS_FILE_SUFFIX;
auto it = ctx->source_part->checksums.files.find(stat_file_name);
if (it != ctx->source_part->checksums.files.end())
{
@@ -1904,7 +1904,7 @@ private:
ctx->updated_header,
ctx->compression_codec,
std::vector(ctx->indices_to_recalc.begin(), ctx->indices_to_recalc.end()),
- Statistics(ctx->stats_to_recalc.begin(), ctx->stats_to_recalc.end()),
+ ColumnsStatistics(ctx->stats_to_recalc.begin(), ctx->stats_to_recalc.end()),
nullptr,
ctx->source_part->index_granularity,
&ctx->source_part->index_granularity_info
diff --git a/src/Storages/MutationCommands.cpp b/src/Storages/MutationCommands.cpp
index aaf5c1b5d87..f736c863eee 100644
--- a/src/Storages/MutationCommands.cpp
+++ b/src/Storages/MutationCommands.cpp
@@ -6,7 +6,7 @@
#include
#include
#include
-#include
+#include
#include
#include
#include
@@ -83,15 +83,15 @@ std::optional MutationCommand::parse(ASTAlterCommand * command,
res.index_name = command->index->as().name();
return res;
}
- else if (command->type == ASTAlterCommand::MATERIALIZE_STATISTIC)
+ else if (command->type == ASTAlterCommand::MATERIALIZE_STATISTICS)
{
MutationCommand res;
res.ast = command->ptr();
- res.type = MATERIALIZE_STATISTIC;
+ res.type = MATERIALIZE_STATISTICS;
if (command->partition)
res.partition = command->partition->clone();
res.predicate = nullptr;
- res.statistic_columns = command->statistic_decl->as().getColumnNames();
+ res.statistics_columns = command->statistics_decl->as().getColumnNames();
return res;
}
else if (command->type == ASTAlterCommand::MATERIALIZE_PROJECTION)
@@ -150,16 +150,16 @@ std::optional MutationCommand::parse(ASTAlterCommand * command,
res.clear = true;
return res;
}
- else if (parse_alter_commands && command->type == ASTAlterCommand::DROP_STATISTIC)
+ else if (parse_alter_commands && command->type == ASTAlterCommand::DROP_STATISTICS)
{
MutationCommand res;
res.ast = command->ptr();
- res.type = MutationCommand::Type::DROP_STATISTIC;
+ res.type = MutationCommand::Type::DROP_STATISTICS;
if (command->partition)
res.partition = command->partition->clone();
if (command->clear_index)
res.clear = true;
- res.statistic_columns = command->statistic_decl->as().getColumnNames();
+ res.statistics_columns = command->statistics_decl->as().getColumnNames();
return res;
}
else if (parse_alter_commands && command->type == ASTAlterCommand::DROP_PROJECTION)
diff --git a/src/Storages/MutationCommands.h b/src/Storages/MutationCommands.h
index 6e10f7d9b2d..f999aab1f4d 100644
--- a/src/Storages/MutationCommands.h
+++ b/src/Storages/MutationCommands.h
@@ -30,12 +30,12 @@ struct MutationCommand
UPDATE,
MATERIALIZE_INDEX,
MATERIALIZE_PROJECTION,
- MATERIALIZE_STATISTIC,
+ MATERIALIZE_STATISTICS,
READ_COLUMN, /// Read column and apply conversions (MODIFY COLUMN alter query).
DROP_COLUMN,
DROP_INDEX,
DROP_PROJECTION,
- DROP_STATISTIC,
+ DROP_STATISTICS,
MATERIALIZE_TTL,
RENAME_COLUMN,
MATERIALIZE_COLUMN,
@@ -51,10 +51,11 @@ struct MutationCommand
/// Columns with corresponding actions
std::unordered_map column_to_update_expression = {};
- /// For MATERIALIZE INDEX and PROJECTION and STATISTIC
+ /// For MATERIALIZE INDEX and PROJECTION and STATISTICS
String index_name = {};
String projection_name = {};
- std::vector statistic_columns = {};
+ std::vector statistics_columns = {};
+ std::vector statistics_types = {};
/// For MATERIALIZE INDEX, UPDATE and DELETE.
ASTPtr partition = {};
diff --git a/src/Storages/S3Queue/S3QueueMetadata.cpp b/src/Storages/S3Queue/S3QueueMetadata.cpp
index f4c8c5c5ef2..9c77bb2d24c 100644
--- a/src/Storages/S3Queue/S3QueueMetadata.cpp
+++ b/src/Storages/S3Queue/S3QueueMetadata.cpp
@@ -167,7 +167,7 @@ S3QueueMetadata::FileMetadataPtr S3QueueMetadata::getFileMetadata(
S3QueueOrderedFileMetadata::BucketInfoPtr bucket_info)
{
auto file_status = local_file_statuses->get(path, /* create */true);
- switch (settings.mode)
+ switch (settings.mode.value)
{
case S3QueueMode::ORDERED:
return std::make_shared(
diff --git a/src/Storages/S3Queue/S3QueueMetadata.h b/src/Storages/S3Queue/S3QueueMetadata.h
index ef4a9808c68..25d01fb52b9 100644
--- a/src/Storages/S3Queue/S3QueueMetadata.h
+++ b/src/Storages/S3Queue/S3QueueMetadata.h
@@ -82,7 +82,6 @@ private:
const fs::path zookeeper_path;
const size_t buckets_num;
- bool initialized = false;
LoggerPtr log;
std::atomic_bool shutdown_called = false;
diff --git a/src/Storages/S3Queue/S3QueueOrderedFileMetadata.cpp b/src/Storages/S3Queue/S3QueueOrderedFileMetadata.cpp
index d1298b8c4fa..bac87c95cc9 100644
--- a/src/Storages/S3Queue/S3QueueOrderedFileMetadata.cpp
+++ b/src/Storages/S3Queue/S3QueueOrderedFileMetadata.cpp
@@ -371,7 +371,6 @@ void S3QueueOrderedFileMetadata::setProcessedImpl()
};
const auto zk_client = getZooKeeper();
- const auto node_metadata_str = node_metadata.toString();
std::string failure_reason;
while (true)
diff --git a/src/Storages/S3Queue/S3QueueSettings.h b/src/Storages/S3Queue/S3QueueSettings.h
index c486a7fbb5d..4a92d99c411 100644
--- a/src/Storages/S3Queue/S3QueueSettings.h
+++ b/src/Storages/S3Queue/S3QueueSettings.h
@@ -13,7 +13,7 @@ class ASTStorage;
#define S3QUEUE_RELATED_SETTINGS(M, ALIAS) \
M(S3QueueMode, \
mode, \
- S3QueueMode::UNORDERED, \
+ S3QueueMode::ORDERED, \
"With unordered mode, the set of all already processed files is tracked with persistent nodes in ZooKepeer." \
"With ordered mode, only the max name of the successfully consumed file stored.", \
0) \
diff --git a/src/Storages/S3Queue/StorageS3Queue.cpp b/src/Storages/S3Queue/StorageS3Queue.cpp
index 0844d0a479e..afb75a21b21 100644
--- a/src/Storages/S3Queue/StorageS3Queue.cpp
+++ b/src/Storages/S3Queue/StorageS3Queue.cpp
@@ -71,8 +71,14 @@ namespace
return zkutil::extractZooKeeperPath(result_zk_path, true);
}
- void checkAndAdjustSettings(S3QueueSettings & s3queue_settings, const Settings & settings)
+ void checkAndAdjustSettings(S3QueueSettings & s3queue_settings, const Settings & settings, bool is_attach)
{
+ if (!is_attach && !s3queue_settings.mode.changed)
+ {
+ throw Exception(ErrorCodes::BAD_ARGUMENTS, "Setting `mode` (Unordered/Ordered) is not specified, but is required.");
+ }
+ /// In case !is_attach, we leave Ordered mode as default for compatibility.
+
if (!s3queue_settings.s3queue_processing_threads_num)
{
throw Exception(ErrorCodes::BAD_ARGUMENTS, "Setting `s3queue_processing_threads_num` cannot be set to zero");
@@ -125,15 +131,7 @@ StorageS3Queue::StorageS3Queue(
throw Exception(ErrorCodes::QUERY_NOT_ALLOWED, "S3Queue url must either end with '/' or contain globs");
}
- if (mode == LoadingStrictnessLevel::CREATE
- && !context_->getSettingsRef().s3queue_allow_experimental_sharded_mode
- && s3queue_settings->mode == S3QueueMode::ORDERED
- && (s3queue_settings->s3queue_buckets > 1 || s3queue_settings->s3queue_processing_threads_num > 1))
- {
- throw Exception(ErrorCodes::QUERY_NOT_ALLOWED, "S3Queue sharded mode is not allowed. To enable use `s3queue_allow_experimental_sharded_mode`");
- }
-
- checkAndAdjustSettings(*s3queue_settings, context_->getSettingsRef());
+ checkAndAdjustSettings(*s3queue_settings, context_->getSettingsRef(), mode > LoadingStrictnessLevel::CREATE);
object_storage = configuration->createObjectStorage(context_, /* is_readonly */true);
FormatFactory::instance().checkFormatName(configuration->format);
diff --git a/src/Storages/Statistics/ConditionSelectivityEstimator.cpp b/src/Storages/Statistics/ConditionSelectivityEstimator.cpp
new file mode 100644
index 00000000000..757136fdf42
--- /dev/null
+++ b/src/Storages/Statistics/ConditionSelectivityEstimator.cpp
@@ -0,0 +1,201 @@
+#include
+#include
+
+namespace DB
+{
+
+namespace ErrorCodes
+{
+ extern const int LOGICAL_ERROR;
+}
+
+void ConditionSelectivityEstimator::ColumnSelectivityEstimator::merge(String part_name, ColumnStatisticsPtr stats)
+{
+ if (part_statistics.contains(part_name))
+ throw Exception(ErrorCodes::LOGICAL_ERROR, "part {} has been added in column {}", part_name, stats->columnName());
+ part_statistics[part_name] = stats;
+}
+
+Float64 ConditionSelectivityEstimator::ColumnSelectivityEstimator::estimateLess(Float64 val, Float64 rows) const
+{
+ if (part_statistics.empty())
+ return default_normal_cond_factor * rows;
+ Float64 result = 0;
+ Float64 part_rows = 0;
+ for (const auto & [key, estimator] : part_statistics)
+ {
+ result += estimator->estimateLess(val);
+ part_rows += estimator->rowCount();
+ }
+ return result * rows / part_rows;
+}
+
+Float64 ConditionSelectivityEstimator::ColumnSelectivityEstimator::estimateGreater(Float64 val, Float64 rows) const
+{
+ return rows - estimateLess(val, rows);
+}
+
+Float64 ConditionSelectivityEstimator::ColumnSelectivityEstimator::estimateEqual(Float64 val, Float64 rows) const
+{
+ if (part_statistics.empty())
+ {
+ if (val < - threshold || val > threshold)
+ return default_normal_cond_factor * rows;
+ else
+ return default_good_cond_factor * rows;
+ }
+ Float64 result = 0;
+ Float64 partial_cnt = 0;
+ for (const auto & [key, estimator] : part_statistics)
+ {
+ result += estimator->estimateEqual(val);
+ partial_cnt += estimator->rowCount();
+ }
+ return result * rows / partial_cnt;
+}
+
+/// second return value represents how many columns in the node.
+static std::pair tryToExtractSingleColumn(const RPNBuilderTreeNode & node)
+{
+ if (node.isConstant())
+ {
+ return {};
+ }
+
+ if (!node.isFunction())
+ {
+ auto column_name = node.getColumnName();
+ return {column_name, 1};
+ }
+
+ auto function_node = node.toFunctionNode();
+ size_t arguments_size = function_node.getArgumentsSize();
+ std::pair result;
+ for (size_t i = 0; i < arguments_size; ++i)
+ {
+ auto function_argument = function_node.getArgumentAt(i);
+ auto subresult = tryToExtractSingleColumn(function_argument);
+ if (subresult.second == 0) /// the subnode contains 0 column
+ continue;
+ else if (subresult.second > 1) /// the subnode contains more than 1 column
+ return subresult;
+ else if (result.second == 0 || result.first == subresult.first) /// subnodes contain same column.
+ result = subresult;
+ else
+ return {"", 2};
+ }
+ return result;
+}
+
+std::pair ConditionSelectivityEstimator::extractBinaryOp(const RPNBuilderTreeNode & node, const String & column_name) const
+{
+ if (!node.isFunction())
+ return {};
+
+ auto function_node = node.toFunctionNode();
+ if (function_node.getArgumentsSize() != 2)
+ return {};
+
+ String function_name = function_node.getFunctionName();
+
+ auto lhs_argument = function_node.getArgumentAt(0);
+ auto rhs_argument = function_node.getArgumentAt(1);
+
+ auto lhs_argument_column_name = lhs_argument.getColumnName();
+ auto rhs_argument_column_name = rhs_argument.getColumnName();
+
+ bool lhs_argument_is_column = column_name == (lhs_argument_column_name);
+ bool rhs_argument_is_column = column_name == (rhs_argument_column_name);
+
+ bool lhs_argument_is_constant = lhs_argument.isConstant();
+ bool rhs_argument_is_constant = rhs_argument.isConstant();
+
+ RPNBuilderTreeNode * constant_node = nullptr;
+
+ if (lhs_argument_is_column && rhs_argument_is_constant)
+ constant_node = &rhs_argument;
+ else if (lhs_argument_is_constant && rhs_argument_is_column)
+ constant_node = &lhs_argument;
+ else
+ return {};
+
+ Field output_value;
+ DataTypePtr output_type;
+ if (!constant_node->tryGetConstant(output_value, output_type))
+ return {};
+
+ const auto type = output_value.getType();
+ Float64 value;
+ if (type == Field::Types::Int64)
+ value = output_value.get();
+ else if (type == Field::Types::UInt64)
+ value = output_value.get();
+ else if (type == Field::Types::Float64)
+ value = output_value.get();
+ else
+ return {};
+ return std::make_pair(function_name, value);
+}
+
+Float64 ConditionSelectivityEstimator::estimateRowCount(const RPNBuilderTreeNode & node) const
+{
+ auto result = tryToExtractSingleColumn(node);
+ if (result.second != 1)
+ {
+ return default_unknown_cond_factor;
+ }
+ String col = result.first;
+ auto it = column_estimators.find(col);
+
+ /// If there the estimator of the column is not found or there are no data at all,
+ /// we use dummy estimation.
+ bool dummy = total_rows == 0;
+ ColumnSelectivityEstimator estimator;
+ if (it != column_estimators.end())
+ {
+ estimator = it->second;
+ }
+ else
+ {
+ dummy = true;
+ }
+ auto [op, val] = extractBinaryOp(node, col);
+ if (op == "equals")
+ {
+ if (dummy)
+ {
+ if (val < - threshold || val > threshold)
+ return default_normal_cond_factor * total_rows;
+ else
+ return default_good_cond_factor * total_rows;
+ }
+ return estimator.estimateEqual(val, total_rows);
+ }
+ else if (op == "less" || op == "lessOrEquals")
+ {
+ if (dummy)
+ return default_normal_cond_factor * total_rows;
+ return estimator.estimateLess(val, total_rows);
+ }
+ else if (op == "greater" || op == "greaterOrEquals")
+ {
+ if (dummy)
+ return default_normal_cond_factor * total_rows;
+ return estimator.estimateGreater(val, total_rows);
+ }
+ else
+ return default_unknown_cond_factor * total_rows;
+}
+
+void ConditionSelectivityEstimator::merge(String part_name, UInt64 part_rows, ColumnStatisticsPtr column_stat)
+{
+ if (!part_names.contains(part_name))
+ {
+ total_rows += part_rows;
+ part_names.insert(part_name);
+ }
+ if (column_stat != nullptr)
+ column_estimators[column_stat->columnName()].merge(part_name, column_stat);
+}
+
+}
diff --git a/src/Storages/Statistics/ConditionSelectivityEstimator.h b/src/Storages/Statistics/ConditionSelectivityEstimator.h
new file mode 100644
index 00000000000..f0599742276
--- /dev/null
+++ b/src/Storages/Statistics/ConditionSelectivityEstimator.h
@@ -0,0 +1,50 @@
+#pragma once
+
+#include
+
+namespace DB
+{
+
+class RPNBuilderTreeNode;
+
+/// It estimates the selectivity of a condition.
+class ConditionSelectivityEstimator
+{
+private:
+ friend class ColumnStatistics;
+ struct ColumnSelectivityEstimator
+ {
+ /// We store the part_name and part_statistics.
+ /// then simply get selectivity for every part_statistics and combine them.
+ std::map part_statistics;
+
+ void merge(String part_name, ColumnStatisticsPtr stats);
+
+ Float64 estimateLess(Float64 val, Float64 rows) const;
+
+ Float64 estimateGreater(Float64 val, Float64 rows) const;
+
+ Float64 estimateEqual(Float64 val, Float64 rows) const;
+ };
+
+ static constexpr auto default_good_cond_factor = 0.1;
+ static constexpr auto default_normal_cond_factor = 0.5;
+ static constexpr auto default_unknown_cond_factor = 1.0;
+ /// Conditions like "x = N" are considered good if abs(N) > threshold.
+ /// This is used to assume that condition is likely to have good selectivity.
+ static constexpr auto threshold = 2;
+
+ UInt64 total_rows = 0;
+ std::set part_names;
+ std::map column_estimators;
+ std::pair