As requested in #22104, moved the files and fixed other includes and build system. Moved files: - combine.hh - collection_mutation.hh - collection_mutation.cc - converting_mutation_partition_applier.hh - converting_mutation_partition_applier.cc - counters.hh - counters.cc - timestamp.hh Fixes: #22104 This is a cleanup, no need to backport Closes scylladb/scylladb#25085
176 lines
6.7 KiB
C++
176 lines
6.7 KiB
C++
/*
|
|
* Copyright (C) 2015-present ScyllaDB
|
|
*
|
|
* Modified by ScyllaDB
|
|
*/
|
|
|
|
/*
|
|
* SPDX-License-Identifier: (LicenseRef-ScyllaDB-Source-Available-1.0 and Apache-2.0)
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "gc_clock.hh"
|
|
#include "mutation/timestamp.hh"
|
|
#include "schema/schema_fwd.hh"
|
|
#include "mutation/atomic_cell.hh"
|
|
#include "mutation/tombstone.hh"
|
|
#include "exceptions/exceptions.hh"
|
|
#include "cql3/query_options.hh"
|
|
#include "cql3/selection/selection.hh"
|
|
#include "query/query-request.hh"
|
|
#include "query/query-result.hh"
|
|
|
|
namespace cql3 {
|
|
|
|
/**
|
|
* A simple container that simplify passing parameters for collections methods.
|
|
*/
|
|
class update_parameters final {
|
|
public:
|
|
// Option set for partition_slice to be used when fetching prefetch_data
|
|
static constexpr query::partition_slice::option_set options = query::partition_slice::option_set::of<
|
|
query::partition_slice::option::send_partition_key,
|
|
query::partition_slice::option::send_clustering_key,
|
|
query::partition_slice::option::collections_as_maps>();
|
|
|
|
// Holder for data for
|
|
// 1) CQL list updates which depend on current state of the list
|
|
// 2) cells needed to check conditions of a CAS statement,
|
|
// 3) rows of CAS result set.
|
|
struct prefetch_data {
|
|
using key = std::pair<partition_key, clustering_key>;
|
|
using key_view = std::pair<const partition_key&, const clustering_key&>;
|
|
struct key_less {
|
|
partition_key::tri_compare pk_cmp;
|
|
clustering_key::tri_compare ck_cmp;
|
|
|
|
key_less(const schema& s)
|
|
: pk_cmp(s)
|
|
, ck_cmp(s)
|
|
{ }
|
|
std::strong_ordering tri_compare(const partition_key& pk1, const clustering_key& ck1,
|
|
const partition_key& pk2, const clustering_key& ck2) const {
|
|
|
|
std::strong_ordering rc = pk_cmp(pk1, pk2);
|
|
return rc != 0 ? rc : ck_cmp(ck1, ck2);
|
|
}
|
|
// Allow mixing std::pair<partition_key, clustering_key> and
|
|
// std::pair<const partition_key&, const clustering_key&> during lookup
|
|
template <typename Pair1, typename Pair2>
|
|
bool operator()(const Pair1& k1, const Pair2& k2) const {
|
|
return tri_compare(k1.first, k1.second, k2.first, k2.second) < 0;
|
|
}
|
|
};
|
|
public:
|
|
struct row {
|
|
bool has_static;
|
|
// indexes are determined by prefetch_data::selection
|
|
std::vector<managed_bytes_opt> cells;
|
|
// Return true if this row has at least one static column set.
|
|
bool has_static_columns(const schema& schema) const {
|
|
return has_static;
|
|
}
|
|
};
|
|
// Use an ordered map since CAS result set must be naturally ordered
|
|
// when returned to the client.
|
|
std::map<key, row, key_less> rows;
|
|
schema_ptr schema;
|
|
shared_ptr<cql3::selection::selection> selection;
|
|
public:
|
|
prefetch_data(schema_ptr schema);
|
|
// Find a row object for either static or regular subset of cells, depending
|
|
// on whether clustering key is empty or not.
|
|
// A particular cell within the row can then be found using a column id.
|
|
const row* find_row(const partition_key& pkey, const clustering_key& ckey) const;
|
|
};
|
|
// Note: value (mutation) only required to contain the rows we are interested in
|
|
private:
|
|
const std::optional<gc_clock::duration> _ttl;
|
|
// For operations that require a read-before-write, stores prefetched cell values.
|
|
// For CAS statements, stores values of conditioned columns.
|
|
// Is a reference to an outside prefetch_data container since a CAS BATCH statement
|
|
// prefetches all rows at once, for all its nested modification statements.
|
|
const prefetch_data& _prefetched;
|
|
public:
|
|
const api::timestamp_type _timestamp;
|
|
const gc_clock::time_point _local_deletion_time;
|
|
const schema_ptr _schema;
|
|
const query_options& _options;
|
|
|
|
update_parameters(const schema_ptr schema_, const query_options& options,
|
|
api::timestamp_type timestamp, std::optional<gc_clock::duration> ttl, const prefetch_data& prefetched)
|
|
: _ttl(ttl)
|
|
, _prefetched(prefetched)
|
|
, _timestamp(timestamp)
|
|
, _local_deletion_time(gc_clock::now())
|
|
, _schema(std::move(schema_))
|
|
, _options(options)
|
|
{
|
|
// We use MIN_VALUE internally to mean the absence of of timestamp (in Selection, in sstable stats, ...), so exclude
|
|
// it to avoid potential confusion.
|
|
if (timestamp < api::min_timestamp || timestamp > api::max_timestamp) {
|
|
throw exceptions::invalid_request_exception(format("Out of bound timestamp, must be in [{:d}, {:d}]",
|
|
api::min_timestamp, api::max_timestamp));
|
|
}
|
|
}
|
|
|
|
atomic_cell make_dead_cell() const {
|
|
return atomic_cell::make_dead(_timestamp, _local_deletion_time);
|
|
}
|
|
|
|
atomic_cell make_cell(const abstract_type& type, const raw_value_view& value, atomic_cell::collection_member cm = atomic_cell::collection_member::no) const {
|
|
auto ttl = this->ttl();
|
|
|
|
return value.with_value([&] (const FragmentedView auto& v) {
|
|
if (ttl.count() > 0) {
|
|
return atomic_cell::make_live(type, _timestamp, v, _local_deletion_time + ttl, ttl, cm);
|
|
} else {
|
|
return atomic_cell::make_live(type, _timestamp, v, cm);
|
|
}
|
|
});
|
|
};
|
|
|
|
atomic_cell make_cell(const abstract_type& type, const managed_bytes_view& value, atomic_cell::collection_member cm = atomic_cell::collection_member::no) const {
|
|
auto ttl = this->ttl();
|
|
|
|
if (ttl.count() > 0) {
|
|
return atomic_cell::make_live(type, _timestamp, value, _local_deletion_time + ttl, ttl, cm);
|
|
} else {
|
|
return atomic_cell::make_live(type, _timestamp, value, cm);
|
|
}
|
|
};
|
|
|
|
atomic_cell make_counter_update_cell(int64_t delta) const {
|
|
return atomic_cell::make_live_counter_update(_timestamp, delta);
|
|
}
|
|
|
|
tombstone make_tombstone() const {
|
|
return {_timestamp, _local_deletion_time};
|
|
}
|
|
|
|
tombstone make_tombstone_just_before() const {
|
|
return {_timestamp - 1, _local_deletion_time};
|
|
}
|
|
|
|
gc_clock::duration ttl() const {
|
|
return _ttl.value_or(_schema->default_time_to_live());
|
|
}
|
|
|
|
gc_clock::time_point expiry() const {
|
|
return ttl() + _local_deletion_time;
|
|
}
|
|
|
|
api::timestamp_type timestamp() const {
|
|
return _timestamp;
|
|
}
|
|
|
|
std::optional<std::vector<std::pair<data_value, data_value>>>
|
|
get_prefetched_list(const partition_key& pkey, const clustering_key& ckey, const column_definition& column) const;
|
|
|
|
static prefetch_data build_prefetch_data(schema_ptr schema, const query::result& query_result,
|
|
const query::partition_slice& slice);
|
|
};
|
|
|
|
}
|