197 lines
6.6 KiB
C++
197 lines
6.6 KiB
C++
/*
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
* or more contributor license agreements. See the NOTICE file
|
|
* distributed with this work for additional information
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
* to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance
|
|
* with the License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
/*
|
|
* Copyright (C) 2015 ScyllaDB
|
|
*
|
|
* Modified by ScyllaDB
|
|
*/
|
|
|
|
/*
|
|
* This file is part of Scylla.
|
|
*
|
|
* Scylla is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* Scylla is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with Scylla. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "gc_clock.hh"
|
|
#include "timestamp.hh"
|
|
#include "schema.hh"
|
|
#include "atomic_cell.hh"
|
|
#include "tombstone.hh"
|
|
#include "exceptions/exceptions.hh"
|
|
#include "cql3/query_options.hh"
|
|
|
|
#include <unordered_map>
|
|
|
|
namespace cql3 {
|
|
|
|
/**
|
|
* A simple container that simplify passing parameters for collections methods.
|
|
*/
|
|
class update_parameters final {
|
|
public:
|
|
// Holder for data needed by CQL list updates which depend on current state of the list.
|
|
struct prefetch_data {
|
|
using key = std::pair<partition_key, std::experimental::optional<clustering_key>>;
|
|
struct key_hashing {
|
|
partition_key::hashing pk_hash;
|
|
clustering_key::hashing ck_hash;
|
|
|
|
key_hashing(const schema& s)
|
|
: pk_hash(s)
|
|
, ck_hash(s)
|
|
{ }
|
|
|
|
size_t operator()(const key& k) const {
|
|
return pk_hash(k.first) ^ (k.second ? ck_hash(*k.second) : 0);
|
|
}
|
|
};
|
|
struct key_equality {
|
|
partition_key::equality pk_eq;
|
|
clustering_key::equality ck_eq;
|
|
|
|
key_equality(const schema& s)
|
|
: pk_eq(s)
|
|
, ck_eq(s)
|
|
{ }
|
|
|
|
bool operator()(const key& k1, const key& k2) const {
|
|
return pk_eq(k1.first, k2.first)
|
|
&& bool(k1.second) == bool(k2.second) && (!k1.second || ck_eq(*k1.second, *k2.second));
|
|
}
|
|
};
|
|
struct cell {
|
|
bytes key;
|
|
bytes value;
|
|
};
|
|
using cell_list = std::vector<cell>;
|
|
using row = std::unordered_map<column_id, cell_list>;
|
|
public:
|
|
std::unordered_map<key, row, key_hashing, key_equality> rows;
|
|
schema_ptr schema;
|
|
public:
|
|
prefetch_data(schema_ptr schema);
|
|
};
|
|
// Note: value (mutation) only required to contain the rows we are interested in
|
|
using prefetched_rows_type = std::experimental::optional<prefetch_data>;
|
|
private:
|
|
const gc_clock::duration _ttl;
|
|
const prefetched_rows_type _prefetched; // For operation that require a read-before-write
|
|
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, gc_clock::duration ttl, prefetched_rows_type prefetched)
|
|
: _ttl(ttl)
|
|
, _prefetched(std::move(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(sprint("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(bytes_view value) const {
|
|
auto ttl = _ttl;
|
|
|
|
if (ttl.count() <= 0) {
|
|
ttl = _schema->default_time_to_live();
|
|
}
|
|
|
|
if (ttl.count() > 0) {
|
|
return atomic_cell::make_live(_timestamp, value, _local_deletion_time + ttl, ttl);
|
|
} else {
|
|
return atomic_cell::make_live(_timestamp, value);
|
|
}
|
|
};
|
|
|
|
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};
|
|
}
|
|
|
|
#if 0
|
|
public RangeTombstone makeRangeTombstone(ColumnSlice slice) throws InvalidRequestException
|
|
{
|
|
QueryProcessor.validateComposite(slice.start, metadata.comparator);
|
|
QueryProcessor.validateComposite(slice.finish, metadata.comparator);
|
|
return new RangeTombstone(slice.start, slice.finish, timestamp, localDeletionTime);
|
|
}
|
|
|
|
public RangeTombstone makeTombstoneForOverwrite(ColumnSlice slice) throws InvalidRequestException
|
|
{
|
|
QueryProcessor.validateComposite(slice.start, metadata.comparator);
|
|
QueryProcessor.validateComposite(slice.finish, metadata.comparator);
|
|
return new RangeTombstone(slice.start, slice.finish, timestamp - 1, localDeletionTime);
|
|
}
|
|
#endif
|
|
|
|
gc_clock::duration ttl() const {
|
|
return _ttl.count() > 0 ? _ttl : _schema->default_time_to_live();
|
|
}
|
|
|
|
gc_clock::time_point expiry() const {
|
|
return ttl() + _local_deletion_time;
|
|
}
|
|
|
|
api::timestamp_type timestamp() const {
|
|
return _timestamp;
|
|
}
|
|
|
|
const prefetch_data::cell_list*
|
|
get_prefetched_list(
|
|
partition_key pkey,
|
|
std::experimental::optional<clustering_key> ckey,
|
|
const column_definition& column) const;
|
|
};
|
|
|
|
}
|