/* -*- Mode: C++; tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*- */ /* * Copyright 2020 Couchbase, Inc. * * Licensed 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. */ #pragma once #include #include #include #include #include #include #include namespace couchbase::protocol { class replace_response_body { public: static const inline client_opcode opcode = client_opcode::replace; private: mutation_token token_; public: mutation_token& token() { return token_; } bool parse(protocol::status status, const header_buffer& header, std::uint8_t framing_extras_size, std::uint16_t, std::uint8_t extras_size, const std::vector& body, const cmd_info&) { Expects(header[1] == static_cast(opcode)); if (status == protocol::status::success) { std::vector::difference_type offset = framing_extras_size; if (extras_size == 16) { memcpy(&token_.partition_uuid, body.data() + offset, sizeof(token_.partition_uuid)); token_.partition_uuid = utils::byte_swap_64(token_.partition_uuid); offset += 8; memcpy(&token_.sequence_number, body.data() + offset, sizeof(token_.sequence_number)); token_.sequence_number = utils::byte_swap_64(token_.sequence_number); return true; } } return false; } }; class replace_request_body { public: using response_body_type = replace_response_body; static const inline client_opcode opcode = client_opcode::replace; private: std::string key_{}; std::vector extras_{}; std::vector content_{}; std::uint32_t flags_{}; std::uint32_t expiration_{}; std::vector framing_extras_{}; public: void id(const document_id& id) { key_ = id.key; if (id.collection_uid) { unsigned_leb128 encoded(*id.collection_uid); key_.insert(0, encoded.get()); } } void durability(protocol::durability_level level, std::optional timeout) { if (level == protocol::durability_level::none) { return; } auto frame_id = static_cast(protocol::request_frame_info_id::durability_requirement); if (timeout) { framing_extras_.resize(4); framing_extras_[0] = static_cast((static_cast(frame_id) << 4U) | 3U); framing_extras_[1] = static_cast(level); uint16_t val = htons(*timeout); memcpy(framing_extras_.data() + 2, &val, sizeof(val)); } else { framing_extras_.resize(2); framing_extras_[0] = static_cast(static_cast(frame_id) << 4U | 1U); framing_extras_[1] = static_cast(level); } } void content(const std::string& content) { content_ = { content.begin(), content.end() }; } void flags(uint32_t flags) { flags_ = flags; } void expiration(uint32_t value) { expiration_ = value; } const std::string& key() { return key_; } const std::vector& framing_extras() { return framing_extras_; } const std::vector& extras() { if (extras_.empty()) { fill_extention(); } return extras_; } const std::vector& value() { return content_; } std::size_t size() { if (extras_.empty()) { fill_extention(); } return framing_extras_.size() + extras_.size() + key_.size() + content_.size(); } private: void fill_extention() { extras_.resize(sizeof(flags_) + sizeof(expiration_)); uint32_t field = htonl(flags_); memcpy(extras_.data(), &field, sizeof(field)); field = htonl(expiration_); memcpy(extras_.data() + sizeof(flags_), &field, sizeof(field)); } }; } // namespace couchbase::protocol