Files
scylladb/cql3/values.hh

220 lines
6.0 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 <boost/variant.hpp>
#include <experimental/optional>
#include <seastar/util/variant_utils.hh>
#include "utils/fragmented_temporary_buffer.hh"
namespace cql3 {
struct null_value {
};
struct unset_value {
};
/// \brief View to a raw CQL protocol value.
///
/// \see raw_value
struct raw_value_view {
boost::variant<fragmented_temporary_buffer::view, null_value, unset_value> _data;
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}
{}
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};
}
bool is_null() const {
return _data.which() == 1;
}
bool is_unset_value() const {
return _data.which() == 2;
}
bool is_value() const {
return _data.which() == 0;
}
std::optional<fragmented_temporary_buffer::view> data() const {
if (_data.which() == 0) {
return boost::get<fragmented_temporary_buffer::view>(_data);
}
return {};
}
explicit operator bool() const {
return _data.which() == 0;
}
const fragmented_temporary_buffer::view* operator->() const {
return &boost::get<fragmented_temporary_buffer::view>(_data);
}
const fragmented_temporary_buffer::view& operator*() const {
return boost::get<fragmented_temporary_buffer::view>(_data);
}
bool operator==(const raw_value_view& other) const {
if (_data.which() != other._data.which()) {
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) {
seastar::visit(value._data, [&] (fragmented_temporary_buffer::view v) {
os << "{ value: ";
using boost::range::for_each;
for_each(v, [&os] (bytes_view bv) { os << bv; });
os << " }";
}, [&] (null_value) {
os << "{ null }";
}, [&] (unset_value) {
os << "{ unset }";
});
return os;
}
};
/// \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 {
boost::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) {
if (view.is_null()) {
return make_null();
}
if (view.is_unset_value()) {
return make_unset_value();
}
return make_value(linearized(*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 _data.which() == 1;
}
bool is_unset_value() const {
return _data.which() == 2;
}
bool is_value() const {
return _data.which() == 0;
}
bytes_opt data() const {
if (_data.which() == 0) {
return boost::get<bytes>(_data);
}
return {};
}
explicit operator bool() const {
return _data.which() == 0;
}
const bytes* operator->() const {
return &boost::get<bytes>(_data);
}
const bytes& operator*() const {
return boost::get<bytes>(_data);
}
raw_value_view to_view() const {
switch (_data.which()) {
case 0: return raw_value_view::make_value(fragmented_temporary_buffer::view(bytes_view{boost::get<bytes>(_data)}));
case 1: return raw_value_view::make_null();
default: return raw_value_view::make_unset_value();
}
}
};
}
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());
}