2014-03-09 17:36:01 +00:00
|
|
|
#pragma once
|
|
|
|
|
2017-06-25 00:01:10 +00:00
|
|
|
#include <Common/SimpleIncrement.h>
|
2020-05-20 20:16:32 +00:00
|
|
|
#include <Common/MultiVersion.h>
|
2019-05-03 02:00:57 +00:00
|
|
|
#include <Storages/IStorage.h>
|
2021-09-06 12:01:16 +00:00
|
|
|
#include <Storages/MergeTree/BackgroundJobsAssignee.h>
|
2019-01-17 12:11:36 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreeIndices.h>
|
2017-08-14 18:16:11 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreePartInfo.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreeSettings.h>
|
2019-05-03 02:00:57 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreeMutationStatus.h>
|
2019-09-03 11:32:25 +00:00
|
|
|
#include <Storages/MergeTree/MergeList.h>
|
2020-05-22 13:29:33 +00:00
|
|
|
#include <Storages/DataDestinationType.h>
|
2017-04-01 09:19:00 +00:00
|
|
|
#include <IO/ReadBufferFromString.h>
|
|
|
|
#include <IO/WriteBufferFromFile.h>
|
|
|
|
#include <IO/ReadBufferFromFile.h>
|
|
|
|
#include <DataTypes/DataTypeString.h>
|
|
|
|
#include <DataTypes/DataTypesNumber.h>
|
2020-04-14 09:21:24 +00:00
|
|
|
#include <Processors/Merges/Algorithms/Graphite.h>
|
2019-10-10 16:30:30 +00:00
|
|
|
#include <Storages/MergeTree/IMergeTreeDataPart.h>
|
2020-04-20 01:38:38 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreeDataPartInMemory.h>
|
2019-01-20 18:43:49 +00:00
|
|
|
#include <Storages/IndicesDescription.h>
|
2019-09-05 13:12:29 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreePartsMover.h>
|
2020-04-14 19:47:19 +00:00
|
|
|
#include <Storages/MergeTree/MergeTreeWriteAheadLog.h>
|
2020-11-24 14:24:48 +00:00
|
|
|
#include <Storages/MergeTree/PinnedPartUUIDs.h>
|
2019-09-03 11:32:25 +00:00
|
|
|
#include <Interpreters/PartLog.h>
|
2020-05-04 20:15:38 +00:00
|
|
|
#include <Disks/StoragePolicy.h>
|
2020-06-16 08:39:12 +00:00
|
|
|
#include <Storages/extractKeyExpressionList.h>
|
2020-07-13 17:27:52 +00:00
|
|
|
#include <Storages/PartitionCommands.h>
|
2022-01-17 11:52:51 +00:00
|
|
|
#include <Storages/MergeTree/ZeroCopyLock.h>
|
|
|
|
|
2014-04-02 13:45:39 +00:00
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
#include <boost/multi_index_container.hpp>
|
|
|
|
#include <boost/multi_index/ordered_index.hpp>
|
|
|
|
#include <boost/multi_index/global_fun.hpp>
|
|
|
|
#include <boost/range/iterator_range_core.hpp>
|
2018-05-21 13:49:54 +00:00
|
|
|
|
2014-04-02 13:45:39 +00:00
|
|
|
|
2014-03-09 17:36:01 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
2018-12-25 23:11:36 +00:00
|
|
|
class AlterCommands;
|
2019-09-05 13:12:29 +00:00
|
|
|
class MergeTreePartsMover;
|
2021-09-10 21:16:09 +00:00
|
|
|
class MergeTreeDataMergerMutator;
|
2020-03-24 17:05:38 +00:00
|
|
|
class MutationCommands;
|
2020-05-20 20:16:32 +00:00
|
|
|
class Context;
|
2022-02-02 18:33:19 +00:00
|
|
|
using PartitionIdToMaxBlock = std::unordered_map<String, Int64>;
|
2020-10-16 10:12:31 +00:00
|
|
|
struct JobAndPool;
|
2022-01-17 11:52:51 +00:00
|
|
|
struct ZeroCopyLock;
|
2018-12-25 23:11:36 +00:00
|
|
|
|
2021-02-18 08:50:31 +00:00
|
|
|
/// Auxiliary struct holding information about the future merged or mutated part.
|
|
|
|
struct EmergingPartInfo
|
|
|
|
{
|
|
|
|
String disk_name;
|
|
|
|
String partition_id;
|
|
|
|
size_t estimate_bytes;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct CurrentlySubmergingEmergingTagger;
|
|
|
|
|
2021-02-10 14:12:49 +00:00
|
|
|
struct SelectQueryOptions;
|
2020-02-10 15:50:12 +00:00
|
|
|
class ExpressionActions;
|
|
|
|
using ExpressionActionsPtr = std::shared_ptr<ExpressionActions>;
|
2020-12-10 23:56:57 +00:00
|
|
|
using ManyExpressionActions = std::vector<ExpressionActionsPtr>;
|
2021-04-02 16:45:18 +00:00
|
|
|
class MergeTreeDeduplicationLog;
|
2020-02-10 15:50:12 +00:00
|
|
|
|
2016-01-11 21:46:36 +00:00
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
2017-04-01 07:20:54 +00:00
|
|
|
extern const int LOGICAL_ERROR;
|
2016-01-11 21:46:36 +00:00
|
|
|
}
|
|
|
|
|
2017-03-07 17:13:54 +00:00
|
|
|
|
2017-02-07 17:52:41 +00:00
|
|
|
/// Data structure for *MergeTree engines.
|
|
|
|
/// Merge tree is used for incremental sorting of data.
|
|
|
|
/// The table consists of several sorted parts.
|
|
|
|
/// During insertion new data is sorted according to the primary key and is written to the new part.
|
|
|
|
/// Parts are merged in the background according to a heuristic algorithm.
|
|
|
|
/// For each part the index file is created containing primary key values for every n-th row.
|
|
|
|
/// This allows efficient selection by primary key range predicate.
|
|
|
|
///
|
|
|
|
/// Additionally:
|
|
|
|
///
|
|
|
|
/// The date column is specified. For each part min and max dates are remembered.
|
|
|
|
/// Essentially it is an index too.
|
|
|
|
///
|
2017-09-13 16:22:04 +00:00
|
|
|
/// Data is partitioned by the value of the partitioning expression.
|
|
|
|
/// Parts belonging to different partitions are not merged - for the ease of administration (data sync and backup).
|
2017-02-07 17:52:41 +00:00
|
|
|
///
|
2017-09-13 16:22:04 +00:00
|
|
|
/// File structure of old-style month-partitioned tables (format_version = 0):
|
2017-02-07 17:52:41 +00:00
|
|
|
/// Part directory - / min-date _ max-date _ min-id _ max-id _ level /
|
|
|
|
/// Inside the part directory:
|
|
|
|
/// checksums.txt - contains the list of all files along with their sizes and checksums.
|
|
|
|
/// columns.txt - contains the list of all columns and their types.
|
|
|
|
/// primary.idx - contains the primary index.
|
|
|
|
/// [Column].bin - contains compressed column data.
|
|
|
|
/// [Column].mrk - marks, pointing to seek positions allowing to skip n * k rows.
|
|
|
|
///
|
2017-09-13 16:22:04 +00:00
|
|
|
/// File structure of tables with custom partitioning (format_version >= 1):
|
2020-05-21 09:00:44 +00:00
|
|
|
/// Part directory - / partition-id _ min-id _ max-id _ level /
|
2017-09-13 16:22:04 +00:00
|
|
|
/// Inside the part directory:
|
|
|
|
/// The same files as for month-partitioned tables, plus
|
2017-10-24 14:11:53 +00:00
|
|
|
/// count.txt - contains total number of rows in this part.
|
|
|
|
/// partition.dat - contains the value of the partitioning expression.
|
2020-01-17 12:24:27 +00:00
|
|
|
/// minmax_[Column].idx - MinMax indexes (see IMergeTreeDataPart::MinMaxIndex class) for the columns required by the partitioning expression.
|
2017-09-13 16:22:04 +00:00
|
|
|
///
|
2017-02-07 17:52:41 +00:00
|
|
|
/// Several modes are implemented. Modes determine additional actions during merge:
|
|
|
|
/// - Ordinary - don't do anything special
|
|
|
|
/// - Collapsing - collapse pairs of rows with the opposite values of sign_columns for the same values
|
|
|
|
/// of primary key (cf. CollapsingSortedBlockInputStream.h)
|
|
|
|
/// - Replacing - for all rows with the same primary key keep only the latest one. Or, if the version
|
|
|
|
/// column is set, keep the latest row with the maximal version.
|
|
|
|
/// - Summing - sum all numeric columns not contained in the primary key for all rows with the same primary key.
|
|
|
|
/// - Aggregating - merge columns containing aggregate function states for all rows with the same primary key.
|
|
|
|
/// - Graphite - performs coarsening of historical data for Graphite (a system for quantitative monitoring).
|
|
|
|
|
|
|
|
/// The MergeTreeData class contains a list of parts and the data structure parameters.
|
|
|
|
/// To read and modify the data use other classes:
|
|
|
|
/// - MergeTreeDataSelectExecutor
|
|
|
|
/// - MergeTreeDataWriter
|
2018-04-20 16:18:16 +00:00
|
|
|
/// - MergeTreeDataMergerMutator
|
2017-02-07 17:52:41 +00:00
|
|
|
|
2021-05-31 14:49:02 +00:00
|
|
|
class MergeTreeData : public IStorage, public WithMutableContext
|
2014-03-09 17:36:01 +00:00
|
|
|
{
|
|
|
|
public:
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Function to call if the part is suspected to contain corrupt data.
|
|
|
|
using BrokenPartCallback = std::function<void (const String &)>;
|
2019-10-10 16:30:30 +00:00
|
|
|
using DataPart = IMergeTreeDataPart;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
using MutableDataPartPtr = std::shared_ptr<DataPart>;
|
2018-05-21 13:49:54 +00:00
|
|
|
using MutableDataPartsVector = std::vector<MutableDataPartPtr>;
|
2017-04-01 07:20:54 +00:00
|
|
|
/// After the DataPart is added to the working set, it cannot be changed.
|
|
|
|
using DataPartPtr = std::shared_ptr<const DataPart>;
|
2017-08-15 11:59:08 +00:00
|
|
|
|
2019-10-10 16:30:30 +00:00
|
|
|
using DataPartState = IMergeTreeDataPart::State;
|
2017-10-03 19:04:56 +00:00
|
|
|
using DataPartStates = std::initializer_list<DataPartState>;
|
2017-10-06 16:48:41 +00:00
|
|
|
using DataPartStateVector = std::vector<DataPartState>;
|
2017-09-11 22:40:51 +00:00
|
|
|
|
2020-11-24 14:24:48 +00:00
|
|
|
using PinnedPartUUIDsPtr = std::shared_ptr<const PinnedPartUUIDs>;
|
|
|
|
|
2021-03-10 02:10:25 +00:00
|
|
|
constexpr static auto FORMAT_VERSION_FILE_NAME = "format_version.txt";
|
|
|
|
constexpr static auto DETACHED_DIR_NAME = "detached";
|
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
/// Auxiliary structure for index comparison. Keep in mind lifetime of MergeTreePartInfo.
|
|
|
|
struct DataPartStateAndInfo
|
|
|
|
{
|
|
|
|
DataPartState state;
|
|
|
|
const MergeTreePartInfo & info;
|
2018-05-21 13:49:54 +00:00
|
|
|
};
|
2017-11-20 19:33:12 +00:00
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
/// Auxiliary structure for index comparison
|
|
|
|
struct DataPartStateAndPartitionID
|
|
|
|
{
|
|
|
|
DataPartState state;
|
|
|
|
String partition_id;
|
2017-11-20 19:33:12 +00:00
|
|
|
};
|
|
|
|
|
2018-06-03 20:39:06 +00:00
|
|
|
STRONG_TYPEDEF(String, PartitionID)
|
2018-05-21 13:49:54 +00:00
|
|
|
|
2020-04-03 10:40:46 +00:00
|
|
|
/// Alter conversions which should be applied on-fly for part. Build from of
|
|
|
|
/// the most recent mutation commands for part. Now we have only rename_map
|
|
|
|
/// here (from ALTER_RENAME) command, because for all other type of alters
|
|
|
|
/// we can deduce conversions for part from difference between
|
|
|
|
/// part->getColumns() and storage->getColumns().
|
2020-03-24 17:05:38 +00:00
|
|
|
struct AlterConversions
|
|
|
|
{
|
2020-04-03 10:40:46 +00:00
|
|
|
/// Rename map new_name -> old_name
|
2020-03-24 17:05:38 +00:00
|
|
|
std::unordered_map<String, String> rename_map;
|
2020-04-03 10:40:46 +00:00
|
|
|
|
|
|
|
bool isColumnRenamed(const String & new_name) const { return rename_map.count(new_name) > 0; }
|
|
|
|
String getColumnOldName(const String & new_name) const { return rename_map.at(new_name); }
|
2020-03-24 17:05:38 +00:00
|
|
|
};
|
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
struct LessDataPart
|
2017-08-15 11:59:08 +00:00
|
|
|
{
|
|
|
|
using is_transparent = void;
|
|
|
|
|
|
|
|
bool operator()(const DataPartPtr & lhs, const MergeTreePartInfo & rhs) const { return lhs->info < rhs; }
|
|
|
|
bool operator()(const MergeTreePartInfo & lhs, const DataPartPtr & rhs) const { return lhs < rhs->info; }
|
|
|
|
bool operator()(const DataPartPtr & lhs, const DataPartPtr & rhs) const { return lhs->info < rhs->info; }
|
2018-05-21 13:49:54 +00:00
|
|
|
bool operator()(const MergeTreePartInfo & lhs, const PartitionID & rhs) const { return lhs.partition_id < rhs.toUnderType(); }
|
|
|
|
bool operator()(const PartitionID & lhs, const MergeTreePartInfo & rhs) const { return lhs.toUnderType() < rhs.partition_id; }
|
2017-08-15 11:59:08 +00:00
|
|
|
};
|
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
struct LessStateDataPart
|
|
|
|
{
|
|
|
|
using is_transparent = void;
|
|
|
|
|
|
|
|
bool operator() (const DataPartStateAndInfo & lhs, const DataPartStateAndInfo & rhs) const
|
|
|
|
{
|
|
|
|
return std::forward_as_tuple(static_cast<UInt8>(lhs.state), lhs.info)
|
|
|
|
< std::forward_as_tuple(static_cast<UInt8>(rhs.state), rhs.info);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator() (DataPartStateAndInfo info, const DataPartState & state) const
|
|
|
|
{
|
|
|
|
return static_cast<size_t>(info.state) < static_cast<size_t>(state);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator() (const DataPartState & state, DataPartStateAndInfo info) const
|
|
|
|
{
|
|
|
|
return static_cast<size_t>(state) < static_cast<size_t>(info.state);
|
|
|
|
}
|
2018-05-21 13:49:54 +00:00
|
|
|
|
|
|
|
bool operator() (const DataPartStateAndInfo & lhs, const DataPartStateAndPartitionID & rhs) const
|
|
|
|
{
|
|
|
|
return std::forward_as_tuple(static_cast<UInt8>(lhs.state), lhs.info.partition_id)
|
|
|
|
< std::forward_as_tuple(static_cast<UInt8>(rhs.state), rhs.partition_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator() (const DataPartStateAndPartitionID & lhs, const DataPartStateAndInfo & rhs) const
|
|
|
|
{
|
|
|
|
return std::forward_as_tuple(static_cast<UInt8>(lhs.state), lhs.partition_id)
|
|
|
|
< std::forward_as_tuple(static_cast<UInt8>(rhs.state), rhs.info.partition_id);
|
|
|
|
}
|
2017-11-20 19:33:12 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
using DataParts = std::set<DataPartPtr, LessDataPart>;
|
2017-04-01 07:20:54 +00:00
|
|
|
using DataPartsVector = std::vector<DataPartPtr>;
|
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
using DataPartsLock = std::unique_lock<std::mutex>;
|
|
|
|
DataPartsLock lockParts() const { return DataPartsLock(data_parts_mutex); }
|
|
|
|
|
2020-02-13 14:42:48 +00:00
|
|
|
MergeTreeDataPartType choosePartType(size_t bytes_uncompressed, size_t rows_count) const;
|
2020-04-14 19:47:19 +00:00
|
|
|
MergeTreeDataPartType choosePartTypeOnDisk(size_t bytes_uncompressed, size_t rows_count) const;
|
2019-12-18 16:41:11 +00:00
|
|
|
|
2020-02-13 14:42:48 +00:00
|
|
|
/// After this method setColumns must be called
|
2019-11-21 16:10:22 +00:00
|
|
|
MutableDataPartPtr createPart(const String & name,
|
2019-11-22 12:51:00 +00:00
|
|
|
MergeTreeDataPartType type, const MergeTreePartInfo & part_info,
|
2021-02-10 14:12:49 +00:00
|
|
|
const VolumePtr & volume, const String & relative_path, const IMergeTreeDataPart * parent_part = nullptr) const;
|
2019-11-21 16:10:22 +00:00
|
|
|
|
2020-06-03 09:51:23 +00:00
|
|
|
/// Create part, that already exists on filesystem.
|
|
|
|
/// After this methods 'loadColumnsChecksumsIndexes' must be called.
|
2019-11-21 16:10:22 +00:00
|
|
|
MutableDataPartPtr createPart(const String & name,
|
2021-02-10 14:12:49 +00:00
|
|
|
const VolumePtr & volume, const String & relative_path, const IMergeTreeDataPart * parent_part = nullptr) const;
|
2019-12-18 16:41:11 +00:00
|
|
|
|
2019-11-21 16:10:22 +00:00
|
|
|
MutableDataPartPtr createPart(const String & name, const MergeTreePartInfo & part_info,
|
2021-02-10 14:12:49 +00:00
|
|
|
const VolumePtr & volume, const String & relative_path, const IMergeTreeDataPart * parent_part = nullptr) const;
|
2019-11-21 16:10:22 +00:00
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
/// Auxiliary object to add a set of parts into the working set in two steps:
|
2021-12-30 14:27:22 +00:00
|
|
|
/// * First, as PreActive parts (the parts are ready, but not yet in the active set).
|
2018-02-19 15:31:43 +00:00
|
|
|
/// * Next, if commit() is called, the parts are added to the active set and the parts that are
|
|
|
|
/// covered by them are marked Outdated.
|
2017-04-01 07:20:54 +00:00
|
|
|
/// If neither commit() nor rollback() was called, the destructor rollbacks the operation.
|
|
|
|
class Transaction : private boost::noncopyable
|
|
|
|
{
|
|
|
|
public:
|
2018-09-20 14:30:52 +00:00
|
|
|
Transaction(MergeTreeData & data_) : data(data_) {}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
DataPartsVector commit(MergeTreeData::DataPartsLock * acquired_parts_lock = nullptr);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-05-24 20:19:29 +00:00
|
|
|
void rollback();
|
2020-09-17 15:33:50 +00:00
|
|
|
|
2020-09-17 19:30:17 +00:00
|
|
|
/// Immediately remove parts from table's data_parts set and change part
|
|
|
|
/// state to temporary. Useful for new parts which not present in table.
|
|
|
|
void rollbackPartsToTemporaryState();
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-05-13 00:24:23 +00:00
|
|
|
size_t size() const { return precommitted_parts.size(); }
|
|
|
|
bool isEmpty() const { return precommitted_parts.empty(); }
|
2017-11-20 19:33:12 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
~Transaction()
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
rollback();
|
|
|
|
}
|
2018-11-23 18:52:00 +00:00
|
|
|
catch (...)
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
tryLogCurrentException("~MergeTreeData::Transaction");
|
|
|
|
}
|
|
|
|
}
|
2018-02-19 15:31:43 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
private:
|
|
|
|
friend class MergeTreeData;
|
|
|
|
|
2018-09-20 14:30:52 +00:00
|
|
|
MergeTreeData & data;
|
2018-02-19 15:31:43 +00:00
|
|
|
DataParts precommitted_parts;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-09-20 14:30:52 +00:00
|
|
|
void clear() { precommitted_parts.clear(); }
|
2017-04-01 07:20:54 +00:00
|
|
|
};
|
|
|
|
|
2021-09-16 21:19:58 +00:00
|
|
|
using TransactionUniquePtr = std::unique_ptr<Transaction>;
|
|
|
|
|
2020-03-19 16:37:55 +00:00
|
|
|
using PathWithDisk = std::pair<String, DiskPtr>;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-07-30 17:24:40 +00:00
|
|
|
struct PartsTemporaryRename : private boost::noncopyable
|
|
|
|
{
|
2019-08-29 16:17:47 +00:00
|
|
|
PartsTemporaryRename(
|
|
|
|
const MergeTreeData & storage_,
|
|
|
|
const String & source_dir_)
|
|
|
|
: storage(storage_)
|
|
|
|
, source_dir(source_dir_)
|
|
|
|
{
|
|
|
|
}
|
2019-07-30 17:24:40 +00:00
|
|
|
|
2021-11-29 18:20:44 +00:00
|
|
|
/// Adds part to rename. Both names are relative to relative_data_path.
|
2021-11-24 19:45:10 +00:00
|
|
|
void addPart(const String & old_name, const String & new_name, const DiskPtr & disk);
|
2019-07-30 17:24:40 +00:00
|
|
|
|
2019-07-31 14:44:55 +00:00
|
|
|
/// Renames part from old_name to new_name
|
|
|
|
void tryRenameAll();
|
|
|
|
|
2019-07-30 17:24:40 +00:00
|
|
|
/// Renames all added parts from new_name to old_name if old name is not empty
|
|
|
|
~PartsTemporaryRename();
|
|
|
|
|
2021-11-24 19:45:10 +00:00
|
|
|
struct RenameInfo
|
|
|
|
{
|
|
|
|
String old_name;
|
|
|
|
String new_name;
|
2021-11-29 18:20:44 +00:00
|
|
|
/// Disk cannot be changed
|
2021-11-24 19:45:10 +00:00
|
|
|
DiskPtr disk;
|
|
|
|
};
|
|
|
|
|
2019-07-31 14:44:55 +00:00
|
|
|
const MergeTreeData & storage;
|
2019-08-29 16:17:47 +00:00
|
|
|
const String source_dir;
|
2021-11-24 19:45:10 +00:00
|
|
|
std::vector<RenameInfo> old_and_new_names;
|
2019-07-31 14:44:55 +00:00
|
|
|
bool renamed = false;
|
2019-07-30 17:24:40 +00:00
|
|
|
};
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
/// Parameters for various modes.
|
|
|
|
struct MergingParams
|
|
|
|
{
|
|
|
|
/// Merging mode. See above.
|
|
|
|
enum Mode
|
|
|
|
{
|
2018-02-02 09:46:54 +00:00
|
|
|
Ordinary = 0, /// Enum values are saved. Do not change them.
|
|
|
|
Collapsing = 1,
|
|
|
|
Summing = 2,
|
|
|
|
Aggregating = 3,
|
|
|
|
Replacing = 5,
|
|
|
|
Graphite = 6,
|
|
|
|
VersionedCollapsing = 7,
|
2017-04-01 07:20:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Mode mode;
|
|
|
|
|
2018-02-02 09:46:54 +00:00
|
|
|
/// For Collapsing and VersionedCollapsing mode.
|
2017-04-01 07:20:54 +00:00
|
|
|
String sign_column;
|
|
|
|
|
|
|
|
/// For Summing mode. If empty - columns_to_sum is determined automatically.
|
|
|
|
Names columns_to_sum;
|
|
|
|
|
2018-02-02 09:46:54 +00:00
|
|
|
/// For Replacing and VersionedCollapsing mode. Can be empty for Replacing.
|
2017-04-01 07:20:54 +00:00
|
|
|
String version_column;
|
|
|
|
|
|
|
|
/// For Graphite mode.
|
|
|
|
Graphite::Params graphite_params;
|
|
|
|
|
|
|
|
/// Check that needed columns are present and have correct types.
|
2020-07-06 14:33:31 +00:00
|
|
|
void check(const StorageInMemoryMetadata & metadata) const;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
String getModeName() const;
|
|
|
|
};
|
|
|
|
|
2019-05-24 19:03:07 +00:00
|
|
|
/// Attach the table corresponding to the directory in full_path inside policy (must end with /), with the given columns.
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Correctness of names and paths is not checked.
|
|
|
|
///
|
2017-09-08 18:11:09 +00:00
|
|
|
/// date_column_name - if not empty, the name of the Date column used for partitioning by month.
|
2018-11-06 18:25:36 +00:00
|
|
|
/// Otherwise, partition_by_ast is used for partitioning.
|
|
|
|
///
|
|
|
|
/// order_by_ast - a single expression or a tuple. It is used as a sorting key
|
|
|
|
/// (an ASTExpressionList used for sorting data in parts);
|
|
|
|
/// primary_key_ast - can be nullptr, an expression, or a tuple.
|
|
|
|
/// Used to determine an ASTExpressionList values of which are written in the primary.idx file
|
|
|
|
/// for one row in every `index_granularity` rows to speed up range queries.
|
2018-10-15 18:02:07 +00:00
|
|
|
/// Primary key must be a prefix of the sorting key;
|
2018-11-06 18:25:36 +00:00
|
|
|
/// If it is nullptr, then it will be determined from order_by_ast.
|
2018-10-15 18:02:07 +00:00
|
|
|
///
|
2017-04-01 07:20:54 +00:00
|
|
|
/// require_part_metadata - should checksums.txt and columns.txt exist in the part directory.
|
|
|
|
/// attach - whether the existing table is attached or the new table is created.
|
2019-12-04 16:06:55 +00:00
|
|
|
MergeTreeData(const StorageID & table_id_,
|
2019-10-28 20:12:14 +00:00
|
|
|
const String & relative_data_path_,
|
2020-06-09 17:28:29 +00:00
|
|
|
const StorageInMemoryMetadata & metadata_,
|
2021-05-31 14:49:02 +00:00
|
|
|
ContextMutablePtr context_,
|
2017-12-03 02:15:35 +00:00
|
|
|
const String & date_column_name,
|
|
|
|
const MergingParams & merging_params_,
|
2019-08-26 14:24:29 +00:00
|
|
|
std::unique_ptr<MergeTreeSettings> settings_,
|
2017-12-03 02:15:35 +00:00
|
|
|
bool require_part_metadata_,
|
|
|
|
bool attach,
|
|
|
|
BrokenPartCallback broken_part_callback_ = [](const String &){});
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-10-13 17:31:37 +00:00
|
|
|
/// Build a block of minmax and count values of a MergeTree table. These values are extracted
|
|
|
|
/// from minmax_indices, the first expression of primary key, and part rows.
|
|
|
|
///
|
|
|
|
/// query_info - used to filter unneeded parts
|
|
|
|
///
|
|
|
|
/// parts - part set to filter
|
|
|
|
///
|
|
|
|
/// normal_parts - collects parts that don't have all the needed values to form the block.
|
|
|
|
/// Specifically, this is when a part doesn't contain a final mark and the related max value is
|
|
|
|
/// required.
|
2021-08-27 18:35:13 +00:00
|
|
|
Block getMinMaxCountProjectionBlock(
|
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
|
|
|
const Names & required_columns,
|
|
|
|
const SelectQueryInfo & query_info,
|
2021-10-09 09:00:39 +00:00
|
|
|
const DataPartsVector & parts,
|
|
|
|
DataPartsVector & normal_parts,
|
2022-02-02 18:33:19 +00:00
|
|
|
const PartitionIdToMaxBlock * max_block_numbers_to_read,
|
2021-08-27 18:35:13 +00:00
|
|
|
ContextPtr query_context) const;
|
|
|
|
|
2021-11-19 03:51:05 +00:00
|
|
|
std::optional<ProjectionCandidate> getQueryProcessingStageWithAggregateProjection(
|
2021-07-09 03:15:41 +00:00
|
|
|
ContextPtr query_context, const StorageSnapshotPtr & storage_snapshot, SelectQueryInfo & query_info) const;
|
2021-05-02 13:38:19 +00:00
|
|
|
|
2021-04-21 16:00:27 +00:00
|
|
|
QueryProcessingStage::Enum getQueryProcessingStage(
|
|
|
|
ContextPtr query_context,
|
|
|
|
QueryProcessingStage::Enum to_stage,
|
2021-07-09 03:15:41 +00:00
|
|
|
const StorageSnapshotPtr & storage_snapshot,
|
2021-04-21 16:00:27 +00:00
|
|
|
SelectQueryInfo & info) const override;
|
2021-02-10 14:12:49 +00:00
|
|
|
|
|
|
|
ReservationPtr reserveSpace(UInt64 expected_size, VolumePtr & volume) const;
|
|
|
|
|
|
|
|
static bool partsContainSameProjections(const DataPartPtr & left, const DataPartPtr & right);
|
2019-12-26 18:17:05 +00:00
|
|
|
|
2020-01-09 14:50:34 +00:00
|
|
|
StoragePolicyPtr getStoragePolicy() const override;
|
2019-05-24 19:03:07 +00:00
|
|
|
|
2019-05-03 02:00:57 +00:00
|
|
|
bool supportsPrewhere() const override { return true; }
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-01-17 04:33:47 +00:00
|
|
|
bool supportsFinal() const override;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-12-22 16:40:53 +00:00
|
|
|
bool supportsSubcolumns() const override { return true; }
|
|
|
|
|
2020-04-28 10:38:57 +00:00
|
|
|
NamesAndTypesList getVirtuals() const override;
|
2019-08-07 15:21:45 +00:00
|
|
|
|
2021-04-10 23:33:54 +00:00
|
|
|
bool mayBenefitFromIndexForIn(const ASTPtr & left_in_operand, ContextPtr, const StorageMetadataPtr & metadata_snapshot) const override;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-07-23 16:30:18 +00:00
|
|
|
struct SnapshotData : public StorageSnapshot::Data
|
|
|
|
{
|
|
|
|
DataPartsVector parts;
|
|
|
|
};
|
|
|
|
|
2021-07-09 03:15:41 +00:00
|
|
|
StorageSnapshotPtr getStorageSnapshot(const StorageMetadataPtr & metadata_snapshot) const override;
|
|
|
|
|
2019-05-04 03:45:58 +00:00
|
|
|
/// Load the set of data parts from disk. Call once - immediately after the object is created.
|
|
|
|
void loadDataParts(bool skip_sanity_checks);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
String getLogName() const { return log_name; }
|
|
|
|
|
2019-05-04 03:45:58 +00:00
|
|
|
Int64 getMaxBlockNumber() const;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Returns a copy of the list so that the caller shouldn't worry about locks.
|
2017-10-03 19:04:56 +00:00
|
|
|
DataParts getDataParts(const DataPartStates & affordable_states) const;
|
2021-06-20 08:24:43 +00:00
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
/// Returns sorted list of the parts with specified states
|
|
|
|
/// out_states will contain snapshot of each part state
|
2021-02-10 14:12:49 +00:00
|
|
|
DataPartsVector getDataPartsVector(
|
|
|
|
const DataPartStates & affordable_states, DataPartStateVector * out_states = nullptr, bool require_projection_parts = false) const;
|
2017-09-11 22:40:51 +00:00
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
/// Returns absolutely all parts (and snapshot of their states)
|
2021-02-10 14:12:49 +00:00
|
|
|
DataPartsVector getAllDataPartsVector(DataPartStateVector * out_states = nullptr, bool require_projection_parts = false) const;
|
2017-09-11 22:40:51 +00:00
|
|
|
|
2019-05-20 16:24:36 +00:00
|
|
|
/// Returns all detached parts
|
2019-07-31 14:44:55 +00:00
|
|
|
DetachedPartsInfo getDetachedParts() const;
|
2019-05-20 16:24:36 +00:00
|
|
|
|
2021-11-29 18:20:44 +00:00
|
|
|
static void validateDetachedPartName(const String & name);
|
2019-07-26 20:04:45 +00:00
|
|
|
|
2022-01-17 04:33:47 +00:00
|
|
|
void dropDetached(const ASTPtr & partition, bool part, ContextPtr local_context);
|
2019-07-30 17:24:40 +00:00
|
|
|
|
2019-07-30 19:11:15 +00:00
|
|
|
MutableDataPartsVector tryLoadPartsToAttach(const ASTPtr & partition, bool attach_part,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr context, PartsTemporaryRename & renamed_parts);
|
2019-05-20 16:24:36 +00:00
|
|
|
|
2021-12-30 14:27:22 +00:00
|
|
|
/// Returns Active parts
|
2017-04-01 07:20:54 +00:00
|
|
|
DataParts getDataParts() const;
|
|
|
|
DataPartsVector getDataPartsVector() const;
|
2017-09-11 22:40:51 +00:00
|
|
|
|
2017-12-19 14:55:50 +00:00
|
|
|
/// Returns a committed part with the given name or a part containing it. If there is no such part, returns nullptr.
|
2019-10-28 17:27:43 +00:00
|
|
|
DataPartPtr getActiveContainingPart(const String & part_name) const;
|
|
|
|
DataPartPtr getActiveContainingPart(const MergeTreePartInfo & part_info) const;
|
|
|
|
DataPartPtr getActiveContainingPart(const MergeTreePartInfo & part_info, DataPartState state, DataPartsLock & lock) const;
|
2018-05-21 13:49:54 +00:00
|
|
|
|
2019-08-16 15:57:19 +00:00
|
|
|
/// Swap part with it's identical copy (possible with another path on another disk).
|
|
|
|
/// If original part is not active or doesn't exist exception will be thrown.
|
2019-08-19 14:40:12 +00:00
|
|
|
void swapActivePart(MergeTreeData::DataPartPtr part_copy);
|
2019-06-07 19:16:42 +00:00
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
/// Returns all parts in specified partition
|
2021-08-18 22:19:14 +00:00
|
|
|
DataPartsVector getDataPartsVectorInPartition(DataPartState state, const String & partition_id) const;
|
|
|
|
DataPartsVector getDataPartsVectorInPartitions(DataPartState state, const std::unordered_set<String> & partition_ids) const;
|
2017-10-03 19:04:56 +00:00
|
|
|
|
2017-12-18 17:26:46 +00:00
|
|
|
/// Returns the part with the given name and state or nullptr if no such part.
|
|
|
|
DataPartPtr getPartIfExists(const String & part_name, const DataPartStates & valid_states);
|
2018-05-21 13:49:54 +00:00
|
|
|
DataPartPtr getPartIfExists(const MergeTreePartInfo & part_info, const DataPartStates & valid_states);
|
2017-10-03 19:04:56 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Total size of active parts in bytes.
|
|
|
|
size_t getTotalActiveSizeInBytes() const;
|
|
|
|
|
2019-10-28 17:27:43 +00:00
|
|
|
size_t getTotalActiveSizeInRows() const;
|
|
|
|
|
2019-05-02 14:48:54 +00:00
|
|
|
size_t getPartsCount() const;
|
2021-02-10 03:22:24 +00:00
|
|
|
size_t getMaxPartsCountForPartitionWithState(DataPartState state) const;
|
2017-08-14 18:16:11 +00:00
|
|
|
size_t getMaxPartsCountForPartition() const;
|
2021-02-10 03:22:24 +00:00
|
|
|
size_t getMaxInactivePartsCountForPartition() const;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-07-31 12:34:34 +00:00
|
|
|
/// Get min value of part->info.getDataVersion() for all active parts.
|
|
|
|
/// Makes sense only for ordinary MergeTree engines because for them block numbering doesn't depend on partition.
|
|
|
|
std::optional<Int64> getMinPartDataVersion() const;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// If the table contains too many active parts, sleep for a while to give them time to merge.
|
|
|
|
/// If until is non-null, wake up from the sleep earlier if the event happened.
|
2018-07-30 20:21:45 +00:00
|
|
|
void delayInsertOrThrowIfNeeded(Poco::Event * until = nullptr) const;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
/// Renames temporary part to a permanent part and adds it to the parts set.
|
2017-04-01 07:20:54 +00:00
|
|
|
/// It is assumed that the part does not intersect with existing parts.
|
2020-08-08 00:47:03 +00:00
|
|
|
/// If increment != nullptr, part index is determining using increment. Otherwise part index remains unchanged.
|
2021-12-30 14:27:22 +00:00
|
|
|
/// If out_transaction != nullptr, adds the part in the PreActive state (the part will be added to the
|
2018-02-19 15:31:43 +00:00
|
|
|
/// active set later with out_transaction->commit()).
|
|
|
|
/// Else, commits the part immediately.
|
2020-06-15 18:57:38 +00:00
|
|
|
/// Returns true if part was added. Returns false if part is covered by bigger part.
|
2022-01-03 23:04:56 +00:00
|
|
|
bool renameTempPartAndAdd(
|
|
|
|
MutableDataPartPtr & part,
|
|
|
|
SimpleIncrement * increment = nullptr,
|
|
|
|
Transaction * out_transaction = nullptr,
|
|
|
|
MergeTreeDeduplicationLog * deduplication_log = nullptr,
|
|
|
|
std::string_view deduplication_token = std::string_view());
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
/// The same as renameTempPartAndAdd but the block range of the part can contain existing parts.
|
|
|
|
/// Returns all parts covered by the added part (in ascending order).
|
|
|
|
/// If out_transaction == nullptr, marks covered parts as Outdated.
|
2017-04-01 07:20:54 +00:00
|
|
|
DataPartsVector renameTempPartAndReplace(
|
2021-04-02 16:45:18 +00:00
|
|
|
MutableDataPartPtr & part, SimpleIncrement * increment = nullptr, Transaction * out_transaction = nullptr, MergeTreeDeduplicationLog * deduplication_log = nullptr);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
/// Low-level version of previous one, doesn't lock mutex
|
2020-06-15 18:57:38 +00:00
|
|
|
bool renameTempPartAndReplace(
|
2022-01-03 23:04:56 +00:00
|
|
|
MutableDataPartPtr & part,
|
|
|
|
SimpleIncrement * increment,
|
|
|
|
Transaction * out_transaction,
|
|
|
|
DataPartsLock & lock,
|
|
|
|
DataPartsVector * out_covered_parts = nullptr,
|
|
|
|
MergeTreeDeduplicationLog * deduplication_log = nullptr,
|
|
|
|
std::string_view deduplication_token = std::string_view());
|
2020-09-17 15:33:50 +00:00
|
|
|
|
2020-09-17 19:30:17 +00:00
|
|
|
/// Remove parts from working set immediately (without wait for background
|
|
|
|
/// process). Transfer part state to temporary. Have very limited usage only
|
2021-03-17 15:49:04 +00:00
|
|
|
/// for new parts which aren't already present in table.
|
2020-09-17 15:33:50 +00:00
|
|
|
void removePartsFromWorkingSetImmediatelyAndSetTemporaryState(const DataPartsVector & remove);
|
|
|
|
|
2017-09-11 22:40:51 +00:00
|
|
|
/// Removes parts from the working set parts.
|
2021-12-30 14:27:22 +00:00
|
|
|
/// Parts in add must already be in data_parts with PreActive, Active, or Outdated states.
|
2017-04-01 07:20:54 +00:00
|
|
|
/// If clear_without_timeout is true, the parts will be deleted at once, or during the next call to
|
|
|
|
/// clearOldParts (ignoring old_parts_lifetime).
|
2018-05-21 13:49:54 +00:00
|
|
|
void removePartsFromWorkingSet(const DataPartsVector & remove, bool clear_without_timeout, DataPartsLock * acquired_lock = nullptr);
|
|
|
|
void removePartsFromWorkingSet(const DataPartsVector & remove, bool clear_without_timeout, DataPartsLock & acquired_lock);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
/// Removes all parts from the working set parts
|
|
|
|
/// for which (partition_id = drop_range.partition_id && min_block >= drop_range.min_block && max_block <= drop_range.max_block).
|
|
|
|
/// Used in REPLACE PARTITION command;
|
|
|
|
DataPartsVector removePartsInRangeFromWorkingSet(const MergeTreePartInfo & drop_range, bool clear_without_timeout,
|
2021-05-13 14:04:36 +00:00
|
|
|
DataPartsLock & lock);
|
2018-05-21 13:49:54 +00:00
|
|
|
|
|
|
|
/// Renames the part to detached/<prefix>_<part> and removes it from data_parts,
|
|
|
|
//// so it will not be deleted in clearOldParts.
|
2017-04-01 07:20:54 +00:00
|
|
|
/// If restore_covered is true, adds to the working set inactive parts, which were merged into the deleted part.
|
2018-05-21 13:49:54 +00:00
|
|
|
void forgetPartAndMoveToDetached(const DataPartPtr & part, const String & prefix = "", bool restore_covered = false);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-09-20 14:30:52 +00:00
|
|
|
/// If the part is Obsolete and not used by anybody else, immediately delete it from filesystem and remove from memory.
|
|
|
|
void tryRemovePartImmediately(DataPartPtr && part);
|
|
|
|
|
2020-01-13 14:45:13 +00:00
|
|
|
/// Returns old inactive parts that can be deleted. At the same time removes them from the list of parts but not from the disk.
|
|
|
|
/// If 'force' - don't wait for old_parts_lifetime.
|
2020-01-10 09:46:24 +00:00
|
|
|
DataPartsVector grabOldParts(bool force = false);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-09-11 22:40:51 +00:00
|
|
|
/// Reverts the changes made by grabOldParts(), parts should be in Deleting state.
|
|
|
|
void rollbackDeletingParts(const DataPartsVector & parts);
|
|
|
|
|
|
|
|
/// Removes parts from data_parts, they should be in Deleting state
|
|
|
|
void removePartsFinally(const DataPartsVector & parts);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-12-14 08:31:17 +00:00
|
|
|
/// When WAL is not enabled, the InMemoryParts need to be persistent.
|
|
|
|
void flushAllInMemoryPartsIfNeeded();
|
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
/// Delete irrelevant parts from memory and disk.
|
2020-01-13 14:45:13 +00:00
|
|
|
/// If 'force' - don't wait for old_parts_lifetime.
|
2021-11-09 12:26:51 +00:00
|
|
|
size_t clearOldPartsFromFilesystem(bool force = false);
|
2019-08-11 19:14:42 +00:00
|
|
|
void clearPartsFromFilesystem(const DataPartsVector & parts);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-05-27 20:05:55 +00:00
|
|
|
/// Delete WAL files containing parts, that all already stored on disk.
|
2021-11-09 12:26:51 +00:00
|
|
|
size_t clearOldWriteAheadLogs();
|
2020-05-27 20:05:55 +00:00
|
|
|
|
2018-09-20 14:30:52 +00:00
|
|
|
/// Delete all directories which names begin with "tmp"
|
2021-07-11 01:01:23 +00:00
|
|
|
/// Must be called with locked lockForShare() because it's using relative_data_path.
|
2021-11-09 12:26:51 +00:00
|
|
|
size_t clearOldTemporaryDirectories(const MergeTreeDataMergerMutator & merger_mutator, size_t custom_directories_lifetime_seconds);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-11-09 12:26:51 +00:00
|
|
|
size_t clearEmptyParts();
|
2020-11-11 16:18:21 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// After the call to dropAllData() no method can be called.
|
|
|
|
/// Deletes the data directory and flushes the uncompressed blocks cache and the marks cache.
|
|
|
|
void dropAllData();
|
|
|
|
|
2020-06-11 03:24:52 +00:00
|
|
|
/// Drop data directories if they are empty. It is safe to call this method if table creation was unsuccessful.
|
|
|
|
void dropIfEmpty();
|
|
|
|
|
2020-06-23 08:04:43 +00:00
|
|
|
/// Moves the entire data directory. Flushes the uncompressed blocks cache
|
|
|
|
/// and the marks cache. Must be called with locked lockExclusively()
|
|
|
|
/// because changes relative_data_path.
|
2020-04-07 14:05:51 +00:00
|
|
|
void rename(const String & new_table_path, const StorageID & new_table_id) override;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
/// Check if the ALTER can be performed:
|
|
|
|
/// - all needed columns are present.
|
|
|
|
/// - all type conversions can be done.
|
2019-01-16 16:53:38 +00:00
|
|
|
/// - columns corresponding to primary key, indices, sign, sampling expression and date are not affected.
|
2017-04-01 07:20:54 +00:00
|
|
|
/// If something is wrong, throws an exception.
|
2021-04-10 23:33:54 +00:00
|
|
|
void checkAlterIsPossible(const AlterCommands & commands, ContextPtr context) const override;
|
2021-02-25 10:07:48 +00:00
|
|
|
|
|
|
|
/// Checks if the Mutation can be performed.
|
2021-02-25 14:43:58 +00:00
|
|
|
/// (currently no additional checks: always ok)
|
2021-02-25 10:07:48 +00:00
|
|
|
void checkMutationIsPossible(const MutationCommands & commands, const Settings & settings) const override;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-07-14 08:19:39 +00:00
|
|
|
/// Checks that partition name in all commands is valid
|
2020-07-13 17:27:52 +00:00
|
|
|
void checkAlterPartitionIsPossible(const PartitionCommands & commands, const StorageMetadataPtr & metadata_snapshot, const Settings & settings) const override;
|
|
|
|
|
2019-08-27 09:34:53 +00:00
|
|
|
/// Change MergeTreeSettings
|
|
|
|
void changeSettings(
|
2020-07-10 08:13:21 +00:00
|
|
|
const ASTPtr & new_settings,
|
2021-10-25 17:49:49 +00:00
|
|
|
AlterLockHolder & table_lock_holder);
|
2019-08-06 13:04:29 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Should be called if part data is suspected to be corrupted.
|
2018-10-17 03:13:00 +00:00
|
|
|
void reportBrokenPart(const String & name) const
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
|
|
|
broken_part_callback(name);
|
|
|
|
}
|
|
|
|
|
2022-01-31 20:47:04 +00:00
|
|
|
/// Same as above but has the ability to check all other parts
|
|
|
|
/// which reside on the same disk of the suspicious part.
|
|
|
|
void reportBrokenPart(MergeTreeData::DataPartPtr & data_part) const;
|
|
|
|
|
2020-06-16 08:39:12 +00:00
|
|
|
/// TODO (alesap) Duplicate method required for compatibility.
|
|
|
|
/// Must be removed.
|
|
|
|
static ASTPtr extractKeyExpressionList(const ASTPtr & node)
|
|
|
|
{
|
|
|
|
return DB::extractKeyExpressionList(node);
|
|
|
|
}
|
2020-01-22 19:52:55 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/** Create local backup (snapshot) for parts with specified prefix.
|
|
|
|
* Backup is created in directory clickhouse_dir/shadow/i/, where i - incremental number,
|
|
|
|
* or if 'with_name' is specified - backup is created in directory with specified name.
|
|
|
|
*/
|
2021-03-02 20:28:42 +00:00
|
|
|
PartitionCommandsResultInfo freezePartition(
|
|
|
|
const ASTPtr & partition,
|
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
|
|
|
const String & with_name,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr context,
|
2021-03-02 20:28:42 +00:00
|
|
|
TableLockHolder & table_lock_holder);
|
|
|
|
|
|
|
|
/// Freezes all parts.
|
|
|
|
PartitionCommandsResultInfo freezeAll(
|
|
|
|
const String & with_name,
|
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr context,
|
2021-03-02 20:28:42 +00:00
|
|
|
TableLockHolder & table_lock_holder);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-03-02 20:28:42 +00:00
|
|
|
/// Unfreezes particular partition.
|
|
|
|
PartitionCommandsResultInfo unfreezePartition(
|
|
|
|
const ASTPtr & partition,
|
2021-02-24 14:26:46 +00:00
|
|
|
const String & backup_name,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr context,
|
2021-03-02 20:28:42 +00:00
|
|
|
TableLockHolder & table_lock_holder);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-03-02 20:28:42 +00:00
|
|
|
/// Unfreezes all parts.
|
|
|
|
PartitionCommandsResultInfo unfreezeAll(
|
|
|
|
const String & backup_name,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr context,
|
2021-03-02 20:28:42 +00:00
|
|
|
TableLockHolder & table_lock_holder);
|
2019-07-23 13:34:17 +00:00
|
|
|
|
2021-08-18 22:19:14 +00:00
|
|
|
/// Prepares entries to backup data of the storage.
|
2021-10-26 09:48:31 +00:00
|
|
|
BackupEntries backup(const ASTs & partitions, ContextPtr context) override;
|
2021-08-18 22:19:14 +00:00
|
|
|
static BackupEntries backupDataParts(const DataPartsVector & data_parts);
|
|
|
|
|
|
|
|
/// Extract data from the backup and put it to the storage.
|
|
|
|
RestoreDataTasks restoreDataPartsFromBackup(
|
|
|
|
const BackupPtr & backup,
|
|
|
|
const String & data_path_in_backup,
|
|
|
|
const std::unordered_set<String> & partition_ids,
|
|
|
|
SimpleIncrement * increment);
|
|
|
|
|
2019-07-23 13:34:17 +00:00
|
|
|
/// Moves partition to specified Disk
|
2021-04-10 23:33:54 +00:00
|
|
|
void movePartitionToDisk(const ASTPtr & partition, const String & name, bool moving_part, ContextPtr context);
|
2019-07-18 15:19:03 +00:00
|
|
|
|
2019-07-23 13:34:17 +00:00
|
|
|
/// Moves partition to specified Volume
|
2021-04-10 23:33:54 +00:00
|
|
|
void movePartitionToVolume(const ASTPtr & partition, const String & name, bool moving_part, ContextPtr context);
|
2019-07-23 13:34:17 +00:00
|
|
|
|
2020-07-13 16:19:08 +00:00
|
|
|
void checkPartitionCanBeDropped(const ASTPtr & partition) override;
|
|
|
|
|
2021-05-17 14:26:36 +00:00
|
|
|
void checkPartCanBeDropped(const String & part_name);
|
2020-11-02 17:30:53 +00:00
|
|
|
|
2020-11-11 13:34:07 +00:00
|
|
|
Pipe alterPartition(
|
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
|
|
|
const PartitionCommands & commands,
|
2021-04-10 23:33:54 +00:00
|
|
|
ContextPtr query_context) override;
|
2020-11-11 13:34:07 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
size_t getColumnCompressedSize(const std::string & name) const
|
|
|
|
{
|
2019-03-28 19:58:41 +00:00
|
|
|
auto lock = lockParts();
|
2017-04-01 07:20:54 +00:00
|
|
|
const auto it = column_sizes.find(name);
|
|
|
|
return it == std::end(column_sizes) ? 0 : it->second.data_compressed;
|
|
|
|
}
|
|
|
|
|
2019-07-16 17:13:12 +00:00
|
|
|
ColumnSizeByName getColumnSizes() const override
|
2017-04-01 07:20:54 +00:00
|
|
|
{
|
2019-03-28 19:58:41 +00:00
|
|
|
auto lock = lockParts();
|
2017-04-01 07:20:54 +00:00
|
|
|
return column_sizes;
|
|
|
|
}
|
|
|
|
|
2021-07-23 16:30:18 +00:00
|
|
|
static ColumnsDescription getObjectsDescription(const DataPartsVector & parts, const NameSet & object_names);
|
2021-04-24 04:09:01 +00:00
|
|
|
|
2021-10-08 13:13:56 +00:00
|
|
|
IndexSizeByName getSecondaryIndexSizes() const override
|
|
|
|
{
|
|
|
|
auto lock = lockParts();
|
|
|
|
return secondary_index_sizes;
|
|
|
|
}
|
|
|
|
|
2017-11-18 02:34:00 +00:00
|
|
|
/// For ATTACH/DETACH/DROP PARTITION.
|
2021-04-10 23:33:54 +00:00
|
|
|
String getPartitionIDFromQuery(const ASTPtr & ast, ContextPtr context) const;
|
2021-08-18 22:19:14 +00:00
|
|
|
std::unordered_set<String> getPartitionIDsFromQuery(const ASTs & asts, ContextPtr context) const;
|
2021-12-15 18:19:29 +00:00
|
|
|
std::set<String> getPartitionIdsAffectedByCommands(const MutationCommands & commands, ContextPtr query_context) const;
|
2017-08-19 18:11:20 +00:00
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
/// Extracts MergeTreeData of other *MergeTree* storage
|
|
|
|
/// and checks that their structure suitable for ALTER TABLE ATTACH PARTITION FROM
|
|
|
|
/// Tables structure should be locked.
|
2020-06-17 10:34:23 +00:00
|
|
|
MergeTreeData & checkStructureAndGetMergeTreeData(const StoragePtr & source_table, const StorageMetadataPtr & src_snapshot, const StorageMetadataPtr & my_snapshot) const;
|
|
|
|
MergeTreeData & checkStructureAndGetMergeTreeData(IStorage & source_table, const StorageMetadataPtr & src_snapshot, const StorageMetadataPtr & my_snapshot) const;
|
2018-05-21 13:49:54 +00:00
|
|
|
|
2019-11-03 22:27:05 +00:00
|
|
|
MergeTreeData::MutableDataPartPtr cloneAndLoadDataPartOnSameDisk(
|
2020-06-26 11:30:23 +00:00
|
|
|
const MergeTreeData::DataPartPtr & src_part, const String & tmp_part_prefix, const MergeTreePartInfo & dst_part_info, const StorageMetadataPtr & metadata_snapshot);
|
2018-03-06 20:18:34 +00:00
|
|
|
|
2019-05-03 02:00:57 +00:00
|
|
|
virtual std::vector<MergeTreeMutationStatus> getMutationsStatus() const = 0;
|
|
|
|
|
2019-08-12 13:30:29 +00:00
|
|
|
/// Returns true if table can create new parts with adaptive granularity
|
|
|
|
/// Has additional constraint in replicated version
|
|
|
|
virtual bool canUseAdaptiveGranularity() const
|
2019-06-19 14:46:06 +00:00
|
|
|
{
|
2019-08-26 14:24:29 +00:00
|
|
|
const auto settings = getSettings();
|
2019-08-13 10:29:31 +00:00
|
|
|
return settings->index_granularity_bytes != 0 &&
|
|
|
|
(settings->enable_mixed_granularity_parts || !has_non_adaptive_index_granularity_parts);
|
2019-06-19 14:46:06 +00:00
|
|
|
}
|
|
|
|
|
2019-08-28 18:23:20 +00:00
|
|
|
/// Get constant pointer to storage settings.
|
|
|
|
/// Copy this pointer into your scope and you will
|
|
|
|
/// get consistent settings.
|
|
|
|
MergeTreeSettingsPtr getSettings() const
|
|
|
|
{
|
|
|
|
return storage_settings.get();
|
|
|
|
}
|
2019-06-19 16:16:13 +00:00
|
|
|
|
2020-04-08 08:41:13 +00:00
|
|
|
String getRelativeDataPath() const { return relative_data_path; }
|
|
|
|
|
2019-09-06 15:09:20 +00:00
|
|
|
/// Get table path on disk
|
2019-11-27 09:39:44 +00:00
|
|
|
String getFullPathOnDisk(const DiskPtr & disk) const;
|
2019-09-06 15:09:20 +00:00
|
|
|
|
2021-11-24 19:45:10 +00:00
|
|
|
/// Looks for detached part on all disks,
|
|
|
|
/// returns pointer to the disk where part is found or nullptr (the second function throws an exception)
|
|
|
|
DiskPtr tryGetDiskForDetachedPart(const String & part_name) const;
|
|
|
|
DiskPtr getDiskForDetachedPart(const String & part_name) const;
|
2019-04-21 18:38:44 +00:00
|
|
|
|
2020-11-01 17:38:43 +00:00
|
|
|
bool storesDataOnDisk() const override { return true; }
|
2019-05-11 18:50:29 +00:00
|
|
|
Strings getDataPaths() const override;
|
2019-04-03 12:52:09 +00:00
|
|
|
|
2019-12-05 08:05:07 +00:00
|
|
|
/// Reserves space at least 1MB.
|
2019-12-12 09:10:50 +00:00
|
|
|
ReservationPtr reserveSpace(UInt64 expected_size) const;
|
2019-04-21 18:38:44 +00:00
|
|
|
|
2019-12-05 08:05:07 +00:00
|
|
|
/// Reserves space at least 1MB on specific disk or volume.
|
2020-03-18 00:57:00 +00:00
|
|
|
static ReservationPtr reserveSpace(UInt64 expected_size, SpacePtr space);
|
|
|
|
static ReservationPtr tryReserveSpace(UInt64 expected_size, SpacePtr space);
|
2019-12-05 08:05:07 +00:00
|
|
|
|
|
|
|
/// Reserves space at least 1MB preferring best destination according to `ttl_infos`.
|
2020-06-17 13:39:26 +00:00
|
|
|
ReservationPtr reserveSpacePreferringTTLRules(
|
2020-10-05 16:41:46 +00:00
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
2020-06-17 13:39:26 +00:00
|
|
|
UInt64 expected_size,
|
|
|
|
const IMergeTreeDataPart::TTLInfos & ttl_infos,
|
|
|
|
time_t time_of_move,
|
2020-09-18 15:30:00 +00:00
|
|
|
size_t min_volume_index = 0,
|
2021-02-18 08:50:31 +00:00
|
|
|
bool is_insert = false,
|
|
|
|
DiskPtr selected_disk = nullptr) const;
|
2020-06-17 13:39:26 +00:00
|
|
|
|
|
|
|
ReservationPtr tryReserveSpacePreferringTTLRules(
|
2020-10-05 16:41:46 +00:00
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
2020-06-17 13:39:26 +00:00
|
|
|
UInt64 expected_size,
|
|
|
|
const IMergeTreeDataPart::TTLInfos & ttl_infos,
|
|
|
|
time_t time_of_move,
|
2020-09-18 15:30:00 +00:00
|
|
|
size_t min_volume_index = 0,
|
2021-02-18 08:50:31 +00:00
|
|
|
bool is_insert = false,
|
|
|
|
DiskPtr selected_disk = nullptr) const;
|
|
|
|
|
|
|
|
/// Reserves space for the part based on the distribution of "big parts" in the same partition.
|
|
|
|
/// Parts with estimated size larger than `min_bytes_to_rebalance_partition_over_jbod` are
|
|
|
|
/// considered as big. The priority is lower than TTL. If reservation fails, return nullptr.
|
|
|
|
ReservationPtr balancedReservation(
|
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
|
|
|
size_t part_size,
|
|
|
|
size_t max_volume_index,
|
|
|
|
const String & part_name,
|
|
|
|
const MergeTreePartInfo & part_info,
|
|
|
|
MergeTreeData::DataPartsVector covered_parts,
|
|
|
|
std::optional<CurrentlySubmergingEmergingTagger> * tagger_ptr,
|
|
|
|
const IMergeTreeDataPart::TTLInfos * ttl_infos,
|
|
|
|
bool is_insert = false);
|
2020-06-17 13:39:26 +00:00
|
|
|
|
2019-05-12 14:57:23 +00:00
|
|
|
/// Choose disk with max available free space
|
|
|
|
/// Reserves 0 bytes
|
2021-04-10 23:33:54 +00:00
|
|
|
ReservationPtr makeEmptyReservationOnLargestDisk() const { return getStoragePolicy()->makeEmptyReservationOnLargestDisk(); }
|
2019-05-12 14:57:23 +00:00
|
|
|
|
2021-08-24 23:05:55 +00:00
|
|
|
Disks getDisks() const { return getStoragePolicy()->getDisks(); }
|
2020-10-14 15:05:59 +00:00
|
|
|
|
2020-04-03 10:40:46 +00:00
|
|
|
/// Return alter conversions for part which must be applied on fly.
|
2021-04-10 23:33:54 +00:00
|
|
|
AlterConversions getAlterConversionsForPart(MergeTreeDataPartPtr part) const;
|
2020-09-18 15:30:00 +00:00
|
|
|
/// Returns destination disk or volume for the TTL rule according to current storage policy
|
|
|
|
/// 'is_insert' - is TTL move performed on new data part insert.
|
2020-09-18 15:41:14 +00:00
|
|
|
SpacePtr getDestinationForMoveTTL(const TTLDescription & move_ttl, bool is_insert = false) const;
|
2020-05-25 17:07:14 +00:00
|
|
|
|
|
|
|
/// Checks if given part already belongs destination disk or volume for the
|
|
|
|
/// TTL rule.
|
2020-05-28 15:33:44 +00:00
|
|
|
bool isPartInTTLDestination(const TTLDescription & ttl, const IMergeTreeDataPart & part) const;
|
2020-03-24 17:05:38 +00:00
|
|
|
|
2020-09-04 10:08:09 +00:00
|
|
|
/// Get count of total merges with TTL in MergeList (system.merges) for all
|
|
|
|
/// tables (not only current table).
|
|
|
|
/// Method is cheap and doesn't require any locks.
|
|
|
|
size_t getTotalMergesWithTTLInMergeList() const;
|
|
|
|
|
2020-04-14 19:47:19 +00:00
|
|
|
using WriteAheadLogPtr = std::shared_ptr<MergeTreeWriteAheadLog>;
|
2020-06-30 18:47:12 +00:00
|
|
|
WriteAheadLogPtr getWriteAheadLog();
|
2020-04-14 19:47:19 +00:00
|
|
|
|
2017-08-25 20:41:45 +00:00
|
|
|
MergeTreeDataFormatVersion format_version;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Merging params - what additional actions to perform during merge.
|
|
|
|
const MergingParams merging_params;
|
|
|
|
|
2018-11-13 12:51:55 +00:00
|
|
|
bool is_custom_partitioned = false;
|
2017-08-19 18:11:20 +00:00
|
|
|
|
2021-03-02 10:57:09 +00:00
|
|
|
/// Used only for old syntax tables. Never changes after init.
|
2017-08-21 15:35:29 +00:00
|
|
|
Int64 minmax_idx_date_column_pos = -1; /// In a common case minmax index includes a date column.
|
2018-12-10 10:20:19 +00:00
|
|
|
Int64 minmax_idx_time_column_pos = -1; /// In other cases, minmax index often includes a dateTime column.
|
2017-08-21 15:35:29 +00:00
|
|
|
|
2021-03-02 10:57:09 +00:00
|
|
|
/// Get partition key expression on required columns
|
2021-03-05 15:21:14 +00:00
|
|
|
static ExpressionActionsPtr getMinMaxExpr(const KeyDescription & partition_key, const ExpressionActionsSettings & settings);
|
2021-03-02 10:57:09 +00:00
|
|
|
/// Get column names required for partition key
|
2021-03-02 16:13:36 +00:00
|
|
|
static Names getMinMaxColumnsNames(const KeyDescription & partition_key);
|
2021-03-02 10:57:09 +00:00
|
|
|
/// Get column types required for partition key
|
2021-03-02 16:13:36 +00:00
|
|
|
static DataTypes getMinMaxColumnsTypes(const KeyDescription & partition_key);
|
2021-03-02 10:33:54 +00:00
|
|
|
|
2020-06-17 09:38:47 +00:00
|
|
|
ExpressionActionsPtr getPrimaryKeyAndSkipIndicesExpression(const StorageMetadataPtr & metadata_snapshot) const;
|
|
|
|
ExpressionActionsPtr getSortingKeyAndSkipIndicesExpression(const StorageMetadataPtr & metadata_snapshot) const;
|
2017-08-21 15:35:29 +00:00
|
|
|
|
2020-09-07 07:59:14 +00:00
|
|
|
/// Get compression codec for part according to TTL rules and <compression>
|
|
|
|
/// section from config.xml.
|
2020-08-31 19:50:42 +00:00
|
|
|
CompressionCodecPtr getCompressionCodecForPart(size_t part_size_compressed, const IMergeTreeDataPart::TTLInfos & ttl_infos, time_t current_time) const;
|
2020-01-22 19:52:55 +00:00
|
|
|
|
2021-10-08 03:40:29 +00:00
|
|
|
std::lock_guard<std::mutex> getQueryIdSetLock() const { return std::lock_guard<std::mutex>(query_id_set_mutex); }
|
|
|
|
|
2021-01-25 05:01:39 +00:00
|
|
|
/// Record current query id where querying the table. Throw if there are already `max_queries` queries accessing the same table.
|
2021-10-08 03:40:29 +00:00
|
|
|
/// Returns false if the `query_id` already exists in the running set, otherwise return true.
|
|
|
|
bool insertQueryIdOrThrow(const String & query_id, size_t max_queries) const;
|
|
|
|
bool insertQueryIdOrThrowNoLock(const String & query_id, size_t max_queries, const std::lock_guard<std::mutex> &) const;
|
2021-01-25 05:01:39 +00:00
|
|
|
|
|
|
|
/// Remove current query id after query finished.
|
|
|
|
void removeQueryId(const String & query_id) const;
|
2021-10-08 03:40:29 +00:00
|
|
|
void removeQueryIdNoLock(const String & query_id, const std::lock_guard<std::mutex> &) const;
|
2021-01-25 05:01:39 +00:00
|
|
|
|
2021-04-27 08:15:59 +00:00
|
|
|
/// Return the partition expression types as a Tuple type. Return DataTypeUInt8 if partition expression is empty.
|
|
|
|
DataTypePtr getPartitionValueType() const;
|
|
|
|
|
2021-08-26 11:01:15 +00:00
|
|
|
/// Construct a sample block of virtual columns.
|
|
|
|
Block getSampleBlockWithVirtualColumns() const;
|
|
|
|
|
2021-04-27 08:15:59 +00:00
|
|
|
/// Construct a block consisting only of possible virtual columns for part pruning.
|
|
|
|
/// If one_part is true, fill in at most one part.
|
2021-08-27 18:35:13 +00:00
|
|
|
Block getBlockWithVirtualPartColumns(const MergeTreeData::DataPartsVector & parts, bool one_part, bool ignore_empty = false) const;
|
2017-04-06 13:03:23 +00:00
|
|
|
|
2017-06-25 00:01:10 +00:00
|
|
|
/// For generating names of temporary parts during insertion.
|
|
|
|
SimpleIncrement insert_increment;
|
|
|
|
|
2019-06-19 14:46:06 +00:00
|
|
|
bool has_non_adaptive_index_granularity_parts = false;
|
|
|
|
|
2019-09-05 13:12:29 +00:00
|
|
|
/// Parts that currently moving from disk/volume to another.
|
|
|
|
/// This set have to be used with `currently_processing_in_background_mutex`.
|
|
|
|
/// Moving may conflict with merges and mutations, but this is OK, because
|
|
|
|
/// if we decide to move some part to another disk, than we
|
|
|
|
/// assuredly will choose this disk for containing part, which will appear
|
|
|
|
/// as result of merge or mutation.
|
|
|
|
DataParts currently_moving_parts;
|
|
|
|
|
2019-09-06 15:09:20 +00:00
|
|
|
/// Mutex for currently_moving_parts
|
2019-09-05 15:53:23 +00:00
|
|
|
mutable std::mutex moving_parts_mutex;
|
2019-08-13 08:35:49 +00:00
|
|
|
|
2020-11-24 14:24:48 +00:00
|
|
|
PinnedPartUUIDsPtr getPinnedPartUUIDs() const;
|
|
|
|
|
2021-06-21 13:36:21 +00:00
|
|
|
/// Schedules background job to like merge/mutate/fetch an executor
|
2021-09-08 00:21:21 +00:00
|
|
|
virtual bool scheduleDataProcessingJob(BackgroundJobsAssignee & assignee) = 0;
|
2021-06-21 13:36:21 +00:00
|
|
|
/// Schedules job to move parts between disks/volumes and so on.
|
2021-09-08 00:21:21 +00:00
|
|
|
bool scheduleDataMovingJob(BackgroundJobsAssignee & assignee);
|
2020-10-14 07:22:48 +00:00
|
|
|
bool areBackgroundMovesNeeded() const;
|
2020-10-13 14:25:42 +00:00
|
|
|
|
2021-07-05 03:32:56 +00:00
|
|
|
/// Lock part in zookeeper for shared data in several nodes
|
2021-02-26 11:06:24 +00:00
|
|
|
/// Overridden in StorageReplicatedMergeTree
|
2021-02-26 09:48:57 +00:00
|
|
|
virtual void lockSharedData(const IMergeTreeDataPart &) const {}
|
|
|
|
|
2021-07-05 03:32:56 +00:00
|
|
|
/// Unlock shared data part in zookeeper
|
2021-02-26 11:06:24 +00:00
|
|
|
/// Overridden in StorageReplicatedMergeTree
|
2021-02-26 09:48:57 +00:00
|
|
|
virtual bool unlockSharedData(const IMergeTreeDataPart &) const { return true; }
|
|
|
|
|
2021-12-21 14:55:20 +00:00
|
|
|
/// Remove lock with old name for shared data part after rename
|
2021-12-01 13:11:26 +00:00
|
|
|
virtual bool unlockSharedData(const IMergeTreeDataPart &, const String &) const { return true; }
|
|
|
|
|
2021-02-26 09:48:57 +00:00
|
|
|
/// Fetch part only if some replica has it on shared storage like S3
|
2021-02-26 11:06:24 +00:00
|
|
|
/// Overridden in StorageReplicatedMergeTree
|
2021-03-05 17:24:06 +00:00
|
|
|
virtual bool tryToFetchIfShared(const IMergeTreeDataPart &, const DiskPtr &, const String &) { return false; }
|
2021-02-26 09:48:57 +00:00
|
|
|
|
2021-12-21 14:29:50 +00:00
|
|
|
/// Check shared data usage on other replicas for detached/freezed part
|
|
|
|
/// Remove local files and remote files if needed
|
|
|
|
virtual bool removeDetachedPart(DiskPtr disk, const String & path, const String & part_name, bool is_freezed);
|
|
|
|
|
2021-12-27 16:27:06 +00:00
|
|
|
/// Store metadata for replicated tables
|
|
|
|
/// Do nothing for non-replicated tables
|
|
|
|
virtual void createAndStoreFreezeMetadata(DiskPtr disk, DataPartPtr part, String backup_part_path) const;
|
2021-11-23 13:57:24 +00:00
|
|
|
|
2021-02-18 08:50:31 +00:00
|
|
|
/// Parts that currently submerging (merging to bigger parts) or emerging
|
2021-03-08 09:38:07 +00:00
|
|
|
/// (to be appeared after merging finished). These two variables have to be used
|
2021-02-18 08:50:31 +00:00
|
|
|
/// with `currently_submerging_emerging_mutex`.
|
2021-03-08 09:38:07 +00:00
|
|
|
DataParts currently_submerging_big_parts;
|
|
|
|
std::map<String, EmergingPartInfo> currently_emerging_big_parts;
|
2021-02-18 08:50:31 +00:00
|
|
|
/// Mutex for currently_submerging_parts and currently_emerging_parts
|
|
|
|
mutable std::mutex currently_submerging_emerging_mutex;
|
|
|
|
|
2019-05-03 02:00:57 +00:00
|
|
|
protected:
|
2019-08-13 08:35:49 +00:00
|
|
|
|
2019-10-10 16:30:30 +00:00
|
|
|
friend class IMergeTreeDataPart;
|
2018-11-09 19:01:39 +00:00
|
|
|
friend class MergeTreeDataMergerMutator;
|
|
|
|
friend struct ReplicatedMergeTreeTableMetadata;
|
2019-05-03 02:00:57 +00:00
|
|
|
friend class StorageReplicatedMergeTree;
|
2020-11-02 18:52:50 +00:00
|
|
|
friend class MergeTreeDataWriter;
|
2021-09-16 21:19:58 +00:00
|
|
|
friend class MergeTask;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-11-09 19:01:39 +00:00
|
|
|
bool require_part_metadata;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-09-25 12:07:38 +00:00
|
|
|
/// Relative path data, changes during rename for ordinary databases use
|
|
|
|
/// under lockForShare if rename is possible.
|
2019-10-28 20:12:14 +00:00
|
|
|
String relative_data_path;
|
2019-04-03 12:52:09 +00:00
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
|
|
|
|
/// Current column sizes in compressed and uncompressed form.
|
2018-03-26 14:18:04 +00:00
|
|
|
ColumnSizeByName column_sizes;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-10-08 13:13:56 +00:00
|
|
|
/// Current secondary index sizes in compressed and uncompressed form.
|
|
|
|
IndexSizeByName secondary_index_sizes;
|
|
|
|
|
2017-05-24 20:19:29 +00:00
|
|
|
/// Engine-specific methods
|
2017-04-01 07:20:54 +00:00
|
|
|
BrokenPartCallback broken_part_callback;
|
|
|
|
|
|
|
|
String log_name;
|
2020-05-30 21:57:37 +00:00
|
|
|
Poco::Logger * log;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2019-08-26 14:24:29 +00:00
|
|
|
/// Storage settings.
|
|
|
|
/// Use get and set to receive readonly versions.
|
|
|
|
MultiVersion<MergeTreeSettings> storage_settings;
|
2017-11-20 19:33:12 +00:00
|
|
|
|
2020-11-24 14:24:48 +00:00
|
|
|
/// Used to determine which UUIDs to send to root query executor for deduplication.
|
|
|
|
mutable std::shared_mutex pinned_part_uuids_mutex;
|
|
|
|
PinnedPartUUIDsPtr pinned_part_uuids;
|
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
/// Work with data parts
|
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
struct TagByInfo{};
|
|
|
|
struct TagByStateAndInfo{};
|
2017-11-20 19:33:12 +00:00
|
|
|
|
|
|
|
static const MergeTreePartInfo & dataPartPtrToInfo(const DataPartPtr & part)
|
|
|
|
{
|
|
|
|
return part->info;
|
|
|
|
}
|
|
|
|
|
|
|
|
static DataPartStateAndInfo dataPartPtrToStateAndInfo(const DataPartPtr & part)
|
|
|
|
{
|
2021-01-15 12:15:13 +00:00
|
|
|
return {part->getState(), part->info};
|
2018-06-03 20:39:06 +00:00
|
|
|
}
|
2017-11-20 19:33:12 +00:00
|
|
|
|
|
|
|
using DataPartsIndexes = boost::multi_index_container<DataPartPtr,
|
|
|
|
boost::multi_index::indexed_by<
|
2018-02-19 15:31:43 +00:00
|
|
|
/// Index by Info
|
2017-11-20 19:33:12 +00:00
|
|
|
boost::multi_index::ordered_unique<
|
2018-02-19 15:31:43 +00:00
|
|
|
boost::multi_index::tag<TagByInfo>,
|
2017-11-20 19:33:12 +00:00
|
|
|
boost::multi_index::global_fun<const DataPartPtr &, const MergeTreePartInfo &, dataPartPtrToInfo>
|
|
|
|
>,
|
2018-02-19 15:31:43 +00:00
|
|
|
/// Index by (State, Info), is used to obtain ordered slices of parts with the same state
|
2017-11-20 19:33:12 +00:00
|
|
|
boost::multi_index::ordered_unique<
|
2018-02-19 15:31:43 +00:00
|
|
|
boost::multi_index::tag<TagByStateAndInfo>,
|
2017-11-20 19:33:12 +00:00
|
|
|
boost::multi_index::global_fun<const DataPartPtr &, DataPartStateAndInfo, dataPartPtrToStateAndInfo>,
|
|
|
|
LessStateDataPart
|
|
|
|
>
|
|
|
|
>
|
|
|
|
>;
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Current set of data parts.
|
|
|
|
mutable std::mutex data_parts_mutex;
|
2017-11-20 19:33:12 +00:00
|
|
|
DataPartsIndexes data_parts_indexes;
|
2018-02-19 15:31:43 +00:00
|
|
|
DataPartsIndexes::index<TagByInfo>::type & data_parts_by_info;
|
|
|
|
DataPartsIndexes::index<TagByStateAndInfo>::type & data_parts_by_state_and_info;
|
2017-11-20 19:33:12 +00:00
|
|
|
|
2019-09-05 13:12:29 +00:00
|
|
|
MergeTreePartsMover parts_mover;
|
|
|
|
|
2021-09-02 21:31:32 +00:00
|
|
|
/// Executors are common for both ReplicatedMergeTree and plain MergeTree
|
|
|
|
/// but they are being started and finished in derived classes, so let them be protected.
|
2021-09-08 00:21:21 +00:00
|
|
|
///
|
|
|
|
/// Why there are two executors, not one? Or an executor for each kind of operation?
|
|
|
|
/// It is historically formed.
|
|
|
|
/// Another explanation is that moving operations are common for Replicated and Plain MergeTree classes.
|
|
|
|
/// Task that schedules this operations is executed with its own timetable and triggered in a specific places in code.
|
|
|
|
/// And for ReplicatedMergeTree we don't have LogEntry type for this operation.
|
|
|
|
BackgroundJobsAssignee background_operations_assignee;
|
|
|
|
BackgroundJobsAssignee background_moves_assignee;
|
|
|
|
|
|
|
|
/// Strongly connected with two fields above.
|
2021-09-02 21:31:32 +00:00
|
|
|
/// Every task that is finished will ask to assign a new one into an executor.
|
2021-09-08 00:21:21 +00:00
|
|
|
/// These callbacks will be passed to the constructor of each task.
|
2021-09-02 21:31:32 +00:00
|
|
|
std::function<void(bool)> common_assignee_trigger;
|
|
|
|
std::function<void(bool)> moves_assignee_trigger;
|
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
using DataPartIteratorByInfo = DataPartsIndexes::index<TagByInfo>::type::iterator;
|
|
|
|
using DataPartIteratorByStateAndInfo = DataPartsIndexes::index<TagByStateAndInfo>::type::iterator;
|
2017-11-20 19:33:12 +00:00
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
boost::iterator_range<DataPartIteratorByStateAndInfo> getDataPartsStateRange(DataPartState state) const
|
2017-11-20 19:33:12 +00:00
|
|
|
{
|
2018-02-19 15:31:43 +00:00
|
|
|
auto begin = data_parts_by_state_and_info.lower_bound(state, LessStateDataPart());
|
|
|
|
auto end = data_parts_by_state_and_info.upper_bound(state, LessStateDataPart());
|
2017-11-20 19:33:12 +00:00
|
|
|
return {begin, end};
|
|
|
|
}
|
|
|
|
|
2018-05-21 13:49:54 +00:00
|
|
|
boost::iterator_range<DataPartIteratorByInfo> getDataPartsPartitionRange(const String & partition_id) const
|
|
|
|
{
|
|
|
|
auto begin = data_parts_by_info.lower_bound(PartitionID(partition_id), LessDataPart());
|
|
|
|
auto end = data_parts_by_info.upper_bound(PartitionID(partition_id), LessDataPart());
|
|
|
|
return {begin, end};
|
|
|
|
}
|
|
|
|
|
2021-03-03 08:36:20 +00:00
|
|
|
std::optional<UInt64> totalRowsByPartitionPredicateImpl(
|
2021-04-10 23:33:54 +00:00
|
|
|
const SelectQueryInfo & query_info, ContextPtr context, const DataPartsVector & parts) const;
|
2021-03-03 08:36:20 +00:00
|
|
|
|
2017-11-20 19:33:12 +00:00
|
|
|
static decltype(auto) getStateModifier(DataPartState state)
|
|
|
|
{
|
2021-01-15 12:15:13 +00:00
|
|
|
return [state] (const DataPartPtr & part) { part->setState(state); };
|
2017-11-20 19:33:12 +00:00
|
|
|
}
|
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
void modifyPartState(DataPartIteratorByStateAndInfo it, DataPartState state)
|
2017-11-20 19:33:12 +00:00
|
|
|
{
|
2018-02-19 15:31:43 +00:00
|
|
|
if (!data_parts_by_state_and_info.modify(it, getStateModifier(state)))
|
2017-11-20 19:33:12 +00:00
|
|
|
throw Exception("Can't modify " + (*it)->getNameWithState(), ErrorCodes::LOGICAL_ERROR);
|
|
|
|
}
|
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
void modifyPartState(DataPartIteratorByInfo it, DataPartState state)
|
2017-11-20 19:33:12 +00:00
|
|
|
{
|
2018-02-19 15:31:43 +00:00
|
|
|
if (!data_parts_by_state_and_info.modify(data_parts_indexes.project<TagByStateAndInfo>(it), getStateModifier(state)))
|
2017-11-20 19:33:12 +00:00
|
|
|
throw Exception("Can't modify " + (*it)->getNameWithState(), ErrorCodes::LOGICAL_ERROR);
|
|
|
|
}
|
|
|
|
|
|
|
|
void modifyPartState(const DataPartPtr & part, DataPartState state)
|
|
|
|
{
|
2018-02-19 15:31:43 +00:00
|
|
|
auto it = data_parts_by_info.find(part->info);
|
|
|
|
if (it == data_parts_by_info.end() || (*it).get() != part.get())
|
|
|
|
throw Exception("Part " + part->name + " doesn't exist", ErrorCodes::LOGICAL_ERROR);
|
2017-11-20 19:33:12 +00:00
|
|
|
|
2018-02-19 15:31:43 +00:00
|
|
|
if (!data_parts_by_state_and_info.modify(data_parts_indexes.project<TagByStateAndInfo>(it), getStateModifier(state)))
|
2017-11-20 19:33:12 +00:00
|
|
|
throw Exception("Can't modify " + (*it)->getNameWithState(), ErrorCodes::LOGICAL_ERROR);
|
|
|
|
}
|
|
|
|
|
2017-04-01 07:20:54 +00:00
|
|
|
/// Used to serialize calls to grabOldParts.
|
|
|
|
std::mutex grab_old_parts_mutex;
|
|
|
|
/// The same for clearOldTemporaryDirectories.
|
|
|
|
std::mutex clear_old_temporary_directories_mutex;
|
|
|
|
|
2020-06-17 10:34:23 +00:00
|
|
|
void checkProperties(const StorageInMemoryMetadata & new_metadata, const StorageInMemoryMetadata & old_metadata, bool attach = false) const;
|
2020-06-10 11:16:31 +00:00
|
|
|
|
2020-06-17 10:34:23 +00:00
|
|
|
void setProperties(const StorageInMemoryMetadata & new_metadata, const StorageInMemoryMetadata & old_metadata, bool attach = false);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2020-06-19 15:21:48 +00:00
|
|
|
void checkPartitionKeyAndInitMinMax(const KeyDescription & new_partition_key);
|
2017-08-19 18:11:20 +00:00
|
|
|
|
2020-06-17 10:34:23 +00:00
|
|
|
void checkTTLExpressions(const StorageInMemoryMetadata & new_metadata, const StorageInMemoryMetadata & old_metadata) const;
|
2020-04-02 16:11:10 +00:00
|
|
|
|
2020-04-22 06:22:14 +00:00
|
|
|
void checkStoragePolicy(const StoragePolicyPtr & new_storage_policy) const;
|
2020-01-09 14:50:34 +00:00
|
|
|
|
2021-10-08 13:13:56 +00:00
|
|
|
/// Calculates column and secondary indexes sizes in compressed form for the current state of data_parts. Call with data_parts mutex locked.
|
|
|
|
void calculateColumnAndSecondaryIndexSizesImpl();
|
|
|
|
|
|
|
|
/// Adds or subtracts the contribution of the part to compressed column and secondary indexes sizes.
|
|
|
|
void addPartContributionToColumnAndSecondaryIndexSizes(const DataPartPtr & part);
|
|
|
|
void removePartContributionToColumnAndSecondaryIndexSizes(const DataPartPtr & part);
|
2017-09-11 17:55:41 +00:00
|
|
|
|
2017-09-11 22:40:51 +00:00
|
|
|
/// If there is no part in the partition with ID `partition_id`, returns empty ptr. Should be called under the lock.
|
2020-07-13 17:27:52 +00:00
|
|
|
DataPartPtr getAnyPartInPartition(const String & partition_id, DataPartsLock & data_parts_lock) const;
|
2018-02-19 15:31:43 +00:00
|
|
|
|
2021-12-30 14:27:22 +00:00
|
|
|
/// Return parts in the Active set that are covered by the new_part_info or the part that covers it.
|
2018-02-19 15:31:43 +00:00
|
|
|
/// Will check that the new part doesn't already exist and that it doesn't intersect existing part.
|
|
|
|
DataPartsVector getActivePartsToReplace(
|
|
|
|
const MergeTreePartInfo & new_part_info,
|
2018-02-19 16:12:16 +00:00
|
|
|
const String & new_part_name,
|
2018-02-19 15:31:43 +00:00
|
|
|
DataPartPtr & out_covering_part,
|
2018-05-21 13:49:54 +00:00
|
|
|
DataPartsLock & data_parts_lock) const;
|
2018-01-21 07:30:07 +00:00
|
|
|
|
2018-03-16 06:51:37 +00:00
|
|
|
/// Checks whether the column is in the primary key, possibly wrapped in a chain of functions with single argument.
|
2020-06-17 12:39:20 +00:00
|
|
|
bool isPrimaryOrMinMaxKeyColumnPossiblyWrappedInFunctions(const ASTPtr & node, const StorageMetadataPtr & metadata_snapshot) const;
|
2018-11-01 17:13:01 +00:00
|
|
|
|
|
|
|
/// Common part for |freezePartition()| and |freezeAll()|.
|
2021-03-02 20:28:42 +00:00
|
|
|
using MatcherFn = std::function<bool(const String &)>;
|
2021-04-10 23:33:54 +00:00
|
|
|
PartitionCommandsResultInfo freezePartitionsByMatcher(MatcherFn matcher, const StorageMetadataPtr & metadata_snapshot, const String & with_name, ContextPtr context);
|
|
|
|
PartitionCommandsResultInfo unfreezePartitionsByMatcher(MatcherFn matcher, const String & backup_name, ContextPtr context);
|
2019-06-19 16:16:13 +00:00
|
|
|
|
2020-11-11 13:34:07 +00:00
|
|
|
// Partition helpers
|
2020-03-09 01:50:33 +00:00
|
|
|
bool canReplacePartition(const DataPartPtr & src_part) const;
|
2019-09-03 11:32:25 +00:00
|
|
|
|
2021-04-20 02:31:08 +00:00
|
|
|
/// Tries to drop part in background without any waits or throwing exceptions in case of errors.
|
2021-05-17 14:26:36 +00:00
|
|
|
virtual void dropPartNoWaitNoThrow(const String & part_name) = 0;
|
|
|
|
|
|
|
|
virtual void dropPart(const String & part_name, bool detach, ContextPtr context) = 0;
|
|
|
|
virtual void dropPartition(const ASTPtr & partition, bool detach, ContextPtr context) = 0;
|
2021-04-10 23:33:54 +00:00
|
|
|
virtual PartitionCommandsResultInfo attachPartition(const ASTPtr & partition, const StorageMetadataPtr & metadata_snapshot, bool part, ContextPtr context) = 0;
|
|
|
|
virtual void replacePartitionFrom(const StoragePtr & source_table, const ASTPtr & partition, bool replace, ContextPtr context) = 0;
|
|
|
|
virtual void movePartitionToTable(const StoragePtr & dest_table, const ASTPtr & partition, ContextPtr context) = 0;
|
2020-11-11 18:17:41 +00:00
|
|
|
|
2021-04-13 04:40:33 +00:00
|
|
|
virtual void fetchPartition(
|
|
|
|
const ASTPtr & partition,
|
|
|
|
const StorageMetadataPtr & metadata_snapshot,
|
|
|
|
const String & from,
|
|
|
|
bool fetch_part,
|
|
|
|
ContextPtr query_context);
|
2020-11-11 13:34:07 +00:00
|
|
|
|
2021-04-20 12:26:05 +00:00
|
|
|
virtual void movePartitionToShard(const ASTPtr & partition, bool move_part, const String & to, ContextPtr query_context);
|
|
|
|
|
2019-09-03 11:32:25 +00:00
|
|
|
void writePartLog(
|
|
|
|
PartLogElement::Type type,
|
|
|
|
const ExecutionStatus & execution_status,
|
|
|
|
UInt64 elapsed_ns,
|
|
|
|
const String & new_part_name,
|
|
|
|
const DataPartPtr & result_part,
|
|
|
|
const DataPartsVector & source_parts,
|
|
|
|
const MergeListEntry * merge_entry);
|
2019-09-05 15:53:23 +00:00
|
|
|
|
2019-09-06 15:09:20 +00:00
|
|
|
/// If part is assigned to merge or mutation (possibly replicated)
|
2020-08-08 00:47:03 +00:00
|
|
|
/// Should be overridden by children, because they can have different
|
2019-09-10 11:21:59 +00:00
|
|
|
/// mechanisms for parts locking
|
2019-09-05 15:53:23 +00:00
|
|
|
virtual bool partIsAssignedToBackgroundOperation(const DataPartPtr & part) const = 0;
|
|
|
|
|
2020-04-03 10:40:46 +00:00
|
|
|
/// Return most recent mutations commands for part which weren't applied
|
|
|
|
/// Used to receive AlterConversions for part and apply them on fly. This
|
|
|
|
/// method has different implementations for replicated and non replicated
|
|
|
|
/// MergeTree because they store mutations in different way.
|
2020-11-28 08:17:20 +00:00
|
|
|
virtual MutationCommands getFirstAlterMutationCommandsForPart(const DataPartPtr & part) const = 0;
|
2019-09-10 11:21:59 +00:00
|
|
|
/// Moves part to specified space, used in ALTER ... MOVE ... queries
|
2019-11-27 09:39:44 +00:00
|
|
|
bool movePartsToSpace(const DataPartsVector & parts, SpacePtr space);
|
2019-09-05 15:53:23 +00:00
|
|
|
|
2019-11-14 11:10:17 +00:00
|
|
|
|
2019-09-05 15:53:23 +00:00
|
|
|
private:
|
2019-09-06 15:09:20 +00:00
|
|
|
/// RAII Wrapper for atomic work with currently moving parts
|
2020-06-27 19:05:00 +00:00
|
|
|
/// Acquire them in constructor and remove them in destructor
|
2019-09-10 11:21:59 +00:00
|
|
|
/// Uses data.currently_moving_parts_mutex
|
2019-09-05 15:53:23 +00:00
|
|
|
struct CurrentlyMovingPartsTagger
|
|
|
|
{
|
|
|
|
MergeTreeMovingParts parts_to_move;
|
|
|
|
MergeTreeData & data;
|
|
|
|
CurrentlyMovingPartsTagger(MergeTreeMovingParts && moving_parts_, MergeTreeData & data_);
|
|
|
|
|
|
|
|
~CurrentlyMovingPartsTagger();
|
|
|
|
};
|
|
|
|
|
2020-10-14 07:22:48 +00:00
|
|
|
using CurrentlyMovingPartsTaggerPtr = std::shared_ptr<CurrentlyMovingPartsTagger>;
|
|
|
|
|
2019-09-10 11:21:59 +00:00
|
|
|
/// Move selected parts to corresponding disks
|
2020-10-14 07:22:48 +00:00
|
|
|
bool moveParts(const CurrentlyMovingPartsTaggerPtr & moving_tagger);
|
2019-09-05 15:53:23 +00:00
|
|
|
|
2019-09-06 15:09:20 +00:00
|
|
|
/// Select parts for move and disks for them. Used in background moving processes.
|
2020-10-14 07:22:48 +00:00
|
|
|
CurrentlyMovingPartsTaggerPtr selectPartsForMove();
|
2019-09-06 15:09:20 +00:00
|
|
|
|
2019-09-10 11:21:59 +00:00
|
|
|
/// Check selected parts for movements. Used by ALTER ... MOVE queries.
|
2020-10-14 07:22:48 +00:00
|
|
|
CurrentlyMovingPartsTaggerPtr checkPartsForMove(const DataPartsVector & parts, SpacePtr space);
|
2020-01-13 16:28:29 +00:00
|
|
|
|
2020-06-15 17:41:44 +00:00
|
|
|
bool canUsePolymorphicParts(const MergeTreeSettings & settings, String * out_reason = nullptr) const;
|
2020-04-14 19:47:19 +00:00
|
|
|
|
2020-06-30 18:47:12 +00:00
|
|
|
std::mutex write_ahead_log_mutex;
|
2020-04-14 19:47:19 +00:00
|
|
|
WriteAheadLogPtr write_ahead_log;
|
2020-06-23 16:40:58 +00:00
|
|
|
|
|
|
|
virtual void startBackgroundMovesIfNeeded() = 0;
|
2020-07-12 12:58:17 +00:00
|
|
|
|
|
|
|
bool allow_nullable_key{};
|
2020-12-22 10:34:35 +00:00
|
|
|
|
|
|
|
void addPartContributionToDataVolume(const DataPartPtr & part);
|
|
|
|
void removePartContributionToDataVolume(const DataPartPtr & part);
|
|
|
|
|
Fix tiny race between count() and INSERT/merges/... in MergeTree
Before it was possible to return stale counter from
StorageMergeTree::totalRows() (that is used for
optimize_trivial_count_query) since the accounting is done in two steps:
- subtract old number of rows
<-- here the number can be zero, even though there are rows -->
- add new number of rows
This was found by CI [1] in 01615_random_one_shard_insertion test:
Here you can see that INSERT went to both tables:
<details>
2022.01.16 09:07:34.288252 [ 154369 ] {a1905be0-93da-460c-8c6f-9b5adace72a0} <Debug> DistributedBlockOutputStream: It took 0.035197041 sec. to insert 100
blocks, 2841.1479249065287 rows per second. Insertion status:
Wrote 54 blocks and 54 rows on shard 0 replica 0, localhost:9000 (average 0 ms per block, the slowest block 1 ms)
Wrote 46 blocks and 46 rows on shard 1 replica 0, localhost:9000 (average 0 ms per block, the slowest block 1 ms)
</details>
But the test fails, since select from shard1.tbl returns 0, and the
problem was concurrent merge:
<details>
2022.01.16 09:07:34.289470 [ 146495 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Debug> executeQuery: (from [::1]:42082) (comment: 01615_random_one_shard_insertion.sql) select count() != 0 from shard_0.tbl;
2022.01.16 09:07:34.289564 [ 375 ] {c7a885fa-4ef4-4dcf-a4de-1650d44fa0ab::all_1_54_9} <Debug> MergeTask::MergeProjectionsStage: Merge sorted 54 rows, containing 1 columns (1 merged, 0 gathered) in 0.00171193 sec., 31543.345814373253 rows/sec., 246.43 KiB>
2022.01.16 09:07:34.289810 [ 375 ] {c7a885fa-4ef4-4dcf-a4de-1650d44fa0ab::all_1_54_9} <Trace> shard_0.tbl (c7a885fa-4ef4-4dcf-a4de-1650d44fa0ab): Renaming temporary part tmp_merge_all_1_54_9 to all_1_54_9.
2022.01.16 09:07:34.289858 [ 146495 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Trace> ContextAccess (default): Access granted: SELECT(number) ON shard_0.tbl
2022.01.16 09:07:34.289897 [ 375 ] {c7a885fa-4ef4-4dcf-a4de-1650d44fa0ab::all_1_54_9} <Trace> shard_0.tbl (c7a885fa-4ef4-4dcf-a4de-1650d44fa0ab) (MergerMutator): Merged 6 parts: from all_1_49_8 to all_54_54_0
2022.01.16 09:07:34.289920 [ 146495 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Trace> InterpreterSelectQuery: WithMergeableState -> Complete
2022.01.16 09:07:34.289987 [ 375 ] {} <Debug> MemoryTracker: Peak memory usage Mutate/Merge: 3.12 MiB.
2022.01.16 09:07:34.290305 [ 154344 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Trace> MergingAggregatedTransform: Reading blocks of partially aggregated data.
2022.01.16 09:07:34.290332 [ 154344 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Debug> MergingAggregatedTransform: Read 1 blocks of partially aggregated data, total 1 rows.
2022.01.16 09:07:34.290343 [ 154344 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Trace> Aggregator: Merging partially aggregated single-level data.
2022.01.16 09:07:34.290358 [ 154344 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Trace> Aggregator: Merged partially aggregated single-level data.
2022.01.16 09:07:34.290366 [ 154344 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Trace> Aggregator: Converting aggregated data to blocks
2022.01.16 09:07:34.290391 [ 154344 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Debug> Aggregator: Converted aggregated data to blocks. 1 rows, 8.00 B in 1.0939e-05 sec. (91416.034 rows/sec., 714.19 KiB/sec.)
2022.01.16 09:07:34.290709 [ 146495 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Information> executeQuery: Read 1 rows, 4.01 KiB in 0.001187722 sec., 841 rows/sec., 3.30 MiB/sec.
2022.01.16 09:07:34.290774 [ 146495 ] {cd9d4cf2-7131-4179-b0b2-3aeec4045755} <Debug> MemoryTracker: Peak memory usage (for query): 0.00 B.
</details>
[1]: https://s3.amazonaws.com/clickhouse-test-reports/33675/7848ea7d609e4c720e8e4494eb6207c0751f5aea/stateless_tests__ubsan__actions_.html
This also fixes a race between DROP TABLE check and INSERT/merges.
v0: use Active parts instead.
v2: fix total counters accounting instead.
2022-01-18 20:19:46 +00:00
|
|
|
void increaseDataVolume(ssize_t bytes, ssize_t rows, ssize_t parts);
|
2020-12-22 10:34:35 +00:00
|
|
|
void setDataVolume(size_t bytes, size_t rows, size_t parts);
|
|
|
|
|
|
|
|
std::atomic<size_t> total_active_size_bytes = 0;
|
|
|
|
std::atomic<size_t> total_active_size_rows = 0;
|
|
|
|
std::atomic<size_t> total_active_size_parts = 0;
|
2021-01-25 05:01:39 +00:00
|
|
|
|
|
|
|
// Record all query ids which access the table. It's guarded by `query_id_set_mutex` and is always mutable.
|
|
|
|
mutable std::set<String> query_id_set;
|
|
|
|
mutable std::mutex query_id_set_mutex;
|
2021-03-02 20:28:42 +00:00
|
|
|
|
|
|
|
// Get partition matcher for FREEZE / UNFREEZE queries.
|
2021-04-10 23:33:54 +00:00
|
|
|
MatcherFn getPartitionMatcher(const ASTPtr & partition, ContextPtr context) const;
|
2021-07-05 12:44:58 +00:00
|
|
|
|
|
|
|
/// Returns default settings for storage with possible changes from global config.
|
|
|
|
virtual std::unique_ptr<MergeTreeSettings> getDefaultSettings() const = 0;
|
2021-08-30 15:29:09 +00:00
|
|
|
|
|
|
|
void loadDataPartsFromDisk(
|
|
|
|
DataPartsVector & broken_parts_to_detach,
|
|
|
|
DataPartsVector & duplicate_parts_to_remove,
|
|
|
|
ThreadPool & pool,
|
|
|
|
size_t num_parts,
|
|
|
|
std::queue<std::vector<std::pair<String, DiskPtr>>> & parts_queue,
|
|
|
|
bool skip_sanity_checks,
|
|
|
|
const MergeTreeSettingsPtr & settings);
|
|
|
|
|
|
|
|
void loadDataPartsFromWAL(
|
|
|
|
DataPartsVector & broken_parts_to_detach,
|
|
|
|
DataPartsVector & duplicate_parts_to_remove,
|
|
|
|
MutableDataPartsVector & parts_from_wal,
|
|
|
|
DataPartsLock & part_lock);
|
2022-01-17 11:52:51 +00:00
|
|
|
|
|
|
|
/// Create zero-copy exclusive lock for part and disk. Useful for coordination of
|
|
|
|
/// distributed operations which can lead to data duplication. Implemented only in ReplicatedMergeTree.
|
|
|
|
virtual std::optional<ZeroCopyLock> tryCreateZeroCopyExclusiveLock(const DataPartPtr &, const DiskPtr &) { return std::nullopt; }
|
2014-03-09 17:36:01 +00:00
|
|
|
};
|
|
|
|
|
2021-02-18 08:50:31 +00:00
|
|
|
/// RAII struct to record big parts that are submerging or emerging.
|
|
|
|
/// It's used to calculate the balanced statistics of JBOD array.
|
|
|
|
struct CurrentlySubmergingEmergingTagger
|
|
|
|
{
|
|
|
|
MergeTreeData & storage;
|
2021-03-08 09:38:07 +00:00
|
|
|
String emerging_part_name;
|
|
|
|
MergeTreeData::DataPartsVector submerging_parts;
|
2021-02-18 08:50:31 +00:00
|
|
|
Poco::Logger * log;
|
|
|
|
|
|
|
|
CurrentlySubmergingEmergingTagger(
|
|
|
|
MergeTreeData & storage_, const String & name_, MergeTreeData::DataPartsVector && parts_, Poco::Logger * log_)
|
2021-03-08 09:38:07 +00:00
|
|
|
: storage(storage_), emerging_part_name(name_), submerging_parts(std::move(parts_)), log(log_)
|
2021-02-18 08:50:31 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
~CurrentlySubmergingEmergingTagger();
|
2014-03-09 17:36:01 +00:00
|
|
|
};
|
|
|
|
|
2021-09-16 21:19:58 +00:00
|
|
|
|
|
|
|
/// TODO: move it somewhere
|
|
|
|
[[ maybe_unused ]] static bool needSyncPart(size_t input_rows, size_t input_bytes, const MergeTreeSettings & settings)
|
|
|
|
{
|
|
|
|
return ((settings.min_rows_to_fsync_after_merge && input_rows >= settings.min_rows_to_fsync_after_merge)
|
|
|
|
|| (settings.min_compressed_bytes_to_fsync_after_merge && input_bytes >= settings.min_compressed_bytes_to_fsync_after_merge));
|
|
|
|
}
|
|
|
|
|
2014-03-09 17:36:01 +00:00
|
|
|
}
|