When a value_view needs to store a temporarily instantiated object, it can use the new variant field. The temporary value will live only as long as the view itself.
213 lines
6.3 KiB
C++
213 lines
6.3 KiB
C++
/*
|
|
* Copyright (C) 2017 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 "types.hh"
|
|
#include "bytes.hh"
|
|
|
|
#include <optional>
|
|
#include <variant>
|
|
|
|
#include <seastar/util/variant_utils.hh>
|
|
|
|
#include "utils/fragmented_temporary_buffer.hh"
|
|
|
|
namespace cql3 {
|
|
|
|
struct null_value {
|
|
};
|
|
|
|
struct unset_value {
|
|
};
|
|
|
|
class raw_value;
|
|
/// \brief View to a raw CQL protocol value.
|
|
///
|
|
/// \see raw_value
|
|
struct raw_value_view {
|
|
std::variant<fragmented_temporary_buffer::view, null_value, unset_value> _data;
|
|
// Temporary storage is only useful if a raw_value_view needs to be instantiated
|
|
// with a value which lifetime is bounded only to the view itself.
|
|
// This hack is introduced in order to avoid storing temporary storage
|
|
// in an external container, which may cause memory leaking problems.
|
|
// This pointer is disengaged for regular raw_value_view instances.
|
|
// Data is stored in a shared pointer for two reasons:
|
|
// - pointers are cheap to copy
|
|
// - it makes the view keep its semantics - it's safe to copy a view multiple times
|
|
// and all copies still refer to the same underlying data.
|
|
lw_shared_ptr<bytes> _temporary_storage = nullptr;
|
|
|
|
raw_value_view(null_value&& data)
|
|
: _data{std::move(data)}
|
|
{}
|
|
raw_value_view(unset_value&& data)
|
|
: _data{std::move(data)}
|
|
{}
|
|
raw_value_view(fragmented_temporary_buffer::view data)
|
|
: _data{data}
|
|
{}
|
|
// This constructor is only used by make_temporary() and it acquires ownership
|
|
// of the given buffer. The view created that way refers to its own temporary storage.
|
|
explicit raw_value_view(bytes&& temporary_storage);
|
|
public:
|
|
static raw_value_view make_null() {
|
|
return raw_value_view{std::move(null_value{})};
|
|
}
|
|
static raw_value_view make_unset_value() {
|
|
return raw_value_view{std::move(unset_value{})};
|
|
}
|
|
static raw_value_view make_value(fragmented_temporary_buffer::view view) {
|
|
return raw_value_view{view};
|
|
}
|
|
static raw_value_view make_temporary(raw_value&& value);
|
|
bool is_null() const {
|
|
return std::holds_alternative<null_value>(_data);
|
|
}
|
|
bool is_unset_value() const {
|
|
return std::holds_alternative<unset_value>(_data);
|
|
}
|
|
bool is_value() const {
|
|
return std::holds_alternative<fragmented_temporary_buffer::view>(_data);
|
|
}
|
|
std::optional<fragmented_temporary_buffer::view> data() const {
|
|
if (auto pdata = std::get_if<fragmented_temporary_buffer::view>(&_data)) {
|
|
return *pdata;
|
|
}
|
|
return {};
|
|
}
|
|
explicit operator bool() const {
|
|
return is_value();
|
|
}
|
|
const fragmented_temporary_buffer::view* operator->() const {
|
|
return &std::get<fragmented_temporary_buffer::view>(_data);
|
|
}
|
|
const fragmented_temporary_buffer::view& operator*() const {
|
|
return std::get<fragmented_temporary_buffer::view>(_data);
|
|
}
|
|
|
|
bool operator==(const raw_value_view& other) const {
|
|
if (_data.index() != other._data.index()) {
|
|
return false;
|
|
}
|
|
if (is_value() && **this != *other) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
bool operator!=(const raw_value_view& other) const {
|
|
return !(*this == other);
|
|
}
|
|
|
|
friend std::ostream& operator<<(std::ostream& os, const raw_value_view& value);
|
|
};
|
|
|
|
/// \brief Raw CQL protocol value.
|
|
///
|
|
/// The `raw_value` type represents an uninterpreted value from the CQL wire
|
|
/// protocol. A raw value can hold either a null value, an unset value, or a byte
|
|
/// blob that represents the value.
|
|
class raw_value {
|
|
std::variant<bytes, null_value, unset_value> _data;
|
|
|
|
raw_value(null_value&& data)
|
|
: _data{std::move(data)}
|
|
{}
|
|
raw_value(unset_value&& data)
|
|
: _data{std::move(data)}
|
|
{}
|
|
raw_value(bytes&& data)
|
|
: _data{std::move(data)}
|
|
{}
|
|
raw_value(const bytes& data)
|
|
: _data{data}
|
|
{}
|
|
public:
|
|
static raw_value make_null() {
|
|
return raw_value{std::move(null_value{})};
|
|
}
|
|
static raw_value make_unset_value() {
|
|
return raw_value{std::move(unset_value{})};
|
|
}
|
|
static raw_value make_value(const raw_value_view& view);
|
|
static raw_value make_value(bytes&& bytes) {
|
|
return raw_value{std::move(bytes)};
|
|
}
|
|
static raw_value make_value(const bytes& bytes) {
|
|
return raw_value{bytes};
|
|
}
|
|
static raw_value make_value(const bytes_opt& bytes) {
|
|
if (bytes) {
|
|
return make_value(*bytes);
|
|
}
|
|
return make_null();
|
|
}
|
|
bool is_null() const {
|
|
return std::holds_alternative<null_value>(_data);
|
|
}
|
|
bool is_unset_value() const {
|
|
return std::holds_alternative<unset_value>(_data);
|
|
}
|
|
bool is_value() const {
|
|
return std::holds_alternative<bytes>(_data);
|
|
}
|
|
bytes_opt data() const {
|
|
if (auto pdata = std::get_if<bytes>(&_data)) {
|
|
return *pdata;
|
|
}
|
|
return {};
|
|
}
|
|
explicit operator bool() const {
|
|
return is_value();
|
|
}
|
|
const bytes* operator->() const {
|
|
return &std::get<bytes>(_data);
|
|
}
|
|
const bytes& operator*() const {
|
|
return std::get<bytes>(_data);
|
|
}
|
|
bytes&& extract_value() && {
|
|
auto b = std::get_if<bytes>(&_data);
|
|
assert(b);
|
|
return std::move(*b);
|
|
}
|
|
raw_value_view to_view() const;
|
|
};
|
|
|
|
}
|
|
|
|
inline bytes to_bytes(const cql3::raw_value_view& view)
|
|
{
|
|
return linearized(*view);
|
|
}
|
|
|
|
inline bytes_opt to_bytes_opt(const cql3::raw_value_view& view) {
|
|
auto buffer_view = view.data();
|
|
if (buffer_view) {
|
|
return bytes_opt(linearized(*buffer_view));
|
|
}
|
|
return bytes_opt();
|
|
}
|
|
|
|
inline bytes_opt to_bytes_opt(const cql3::raw_value& value) {
|
|
return to_bytes_opt(value.to_view());
|
|
}
|