Lightweight Update
Lightweight Update
When lightweight updates are enabled, updated rows are marked as updated immediately and subsequent SELECT
queries will automatically return with the changed values. When lightweight updates are not enabled, you may have to wait for your mutations to be applied via a background process to see the changed values.
Lightweight updates can be enabled for MergeTree
-family tables by enabling the query-level setting apply_mutations_on_fly
.
SET apply_mutations_on_fly = 1;
Example
Let's create a table and run some mutations:
CREATE TABLE test_on_fly_mutations (id UInt64, v String)
ENGINE = MergeTree ORDER BY id;
-- Disable background materialization of mutations to showcase
-- default behavior when lightweight updates are not enabled
SYSTEM STOP MERGES test_on_fly_mutations;
SET mutations_sync = 0;
-- Insert some rows in our new table
INSERT INTO test_on_fly_mutations VALUES (1, 'a'), (2, 'b'), (3, 'c');
-- Update the values of the rows
ALTER TABLE test_on_fly_mutations UPDATE v = 'd' WHERE id = 1;
ALTER TABLE test_on_fly_mutations DELETE WHERE v = 'd';
ALTER TABLE test_on_fly_mutations UPDATE v = 'e' WHERE id = 2;
ALTER TABLE test_on_fly_mutations DELETE WHERE v = 'e';
Let's check the result of the updates via a SELECT
query:
-- Explicitly disable lightweight updates
SET apply_mutations_on_fly = 0;
SELECT id, v FROM test_on_fly_mutations ORDER BY id;
Note that the values of the rows have not yet been updated when we query the new table:
┌─id─┬─v─┐
│ 1 │ a │
│ 2 │ b │
│ 3 │ c │
└────┴───┘
Let's now see what happens when we enable lightweight updates:
-- Enable lightweight updates
SET apply_mutations_on_fly = 1;
SELECT id, v FROM test_on_fly_mutations ORDER BY id;
The SELECT
query now returns the correct result immediately, without having to wait for the mutations to be applied:
┌─id─┬─v─┐
│ 3 │ c │
└────┴───┘
Performance Impact
When lightweight updates are enabled, mutations are not materialized immediately but will only be applied during SELECT
queries. However, please note that mutations are still being materialized asynchronously in the background, which is a heavy process.
If the number of submitted mutations constantly exceeds the number of mutations that are processed in the background over some time interval, the queue of unmaterialized mutations that have to be applied will continue to grow. This will result in the eventual degradation of SELECT
query performance.
We suggest enabling the setting apply_mutations_on_fly
together with other MergeTree
-level settings such as number_of_mutations_to_throw
and number_of_mutations_to_delay
to restrict the infinite growth of unmaterialized mutations.
Support for subqueries and non-deterministic functions
Lightweight updates have limited support with subqueries and non-deterministic functions. Only scalar subqueries with a result that have a reasonable size (controlled by the setting mutations_max_literal_size_to_replace
) are supported. Only constant non-deterministic functions are supported (e.g. the function now()
).
These behaviours are controlled by the following settings:
mutations_execute_nondeterministic_on_initiator
- if true, non-deterministic functions are executed on the initiator replica and are replaced as literals inUPDATE
andDELETE
queries. Default value:false
.mutations_execute_subqueries_on_initiator
- if true, scalar subqueries are executed on the initiator replica and are replaced as literals inUPDATE
andDELETE
queries. Default value:false
.mutations_max_literal_size_to_replace
- The maximum size of serialized literals in bytes to replace inUPDATE
andDELETE
queries. Default value:16384
(16 KiB).