// Copyright (c) 2018, The Monero Project // All rights reserved. // // Redistribution and use in source and binary forms, with or without modification, are // permitted provided that the following conditions are met: // // 1. Redistributions of source code must retain the above copyright notice, this list of // conditions and the following disclaimer. // // 2. Redistributions in binary form must reproduce the above copyright notice, this list // of conditions and the following disclaimer in the documentation and/or other // materials provided with the distribution. // // 3. Neither the name of the copyright holder nor the names of its contributors may be // used to endorse or promote products derived from this software without specific // prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY // EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL // THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, // PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, // STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF // THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #include "data.h" #include #include #include "wire/crypto.h" #include "wire.h" namespace lws { namespace db { namespace { template void map_output_id(F& format, T& self) { wire::object(format, WIRE_FIELD(high), WIRE_FIELD(low)); } } WIRE_DEFINE_OBJECT(output_id, map_output_id); namespace { constexpr const char* map_account_status[] = {"active", "inactive", "hidden"}; constexpr const char* map_request[] = {"create", "import"}; } WIRE_DEFINE_ENUM(account_status, map_account_status); WIRE_DEFINE_ENUM(request, map_request); namespace { template void map_account_address(F& format, T& self) { wire::object(format, WIRE_FIELD(spend_public), WIRE_FIELD(view_public)); } } WIRE_DEFINE_OBJECT(account_address, map_account_address); void write_bytes(wire::writer& dest, const account& self, const bool show_key) { view_key const* const key = show_key ? std::addressof(self.key) : nullptr; const bool admin = (self.flags & admin_account); const bool generated_locally = (self.flags & account_generated_locally); wire::object(dest, WIRE_FIELD(id), wire::field("access_time", self.access), WIRE_FIELD(address), wire::optional_field("view_key", key), WIRE_FIELD(scan_height), WIRE_FIELD(start_height), wire::field("creation_time", self.creation), wire::field("admin", admin), wire::field("generated_locally", generated_locally) ); } namespace { template void map_block_info(F& format, T& self) { wire::object(format, WIRE_FIELD(id), WIRE_FIELD(hash)); } } WIRE_DEFINE_OBJECT(block_info, map_block_info); namespace { template void map_transaction_link(F& format, T& self) { wire::object(format, WIRE_FIELD(height), WIRE_FIELD(tx_hash)); } } WIRE_DEFINE_OBJECT(transaction_link, map_transaction_link); void write_bytes(wire::writer& dest, const output& self) { const std::pair unpacked = db::unpack(self.extra); const bool coinbase = (unpacked.first & lws::db::coinbase_output); const bool rct = (unpacked.first & lws::db::ringct_output); const auto rct_mask = rct ? std::addressof(self.ringct_mask) : nullptr; epee::span payment_bytes{}; if (unpacked.second == 32) payment_bytes = epee::as_byte_span(self.payment_id.long_); else if (unpacked.second == 8) payment_bytes = epee::as_byte_span(self.payment_id.short_); const auto payment_id = payment_bytes.empty() ? nullptr : std::addressof(payment_bytes); wire::object(dest, wire::field("id", std::cref(self.spend_meta.id)), wire::field("block", self.link.height), wire::field("index", self.spend_meta.index), wire::field("amount", self.spend_meta.amount), wire::field("timestamp", self.timestamp), wire::field("tx_hash", std::cref(self.link.tx_hash)), wire::field("tx_prefix_hash", std::cref(self.tx_prefix_hash)), wire::field("tx_public", std::cref(self.spend_meta.tx_public)), wire::optional_field("rct_mask", rct_mask), wire::optional_field("payment_id", payment_id), wire::field("unlock_time", self.unlock_time), wire::field("mixin_count", self.spend_meta.mixin_count), wire::field("coinbase", coinbase) ); } namespace { template void map_spend(F& format, T1& self, T2& payment_id) { wire::object(format, wire::field("height", std::ref(self.link.height)), wire::field("tx_hash", std::ref(self.link.tx_hash)), WIRE_FIELD(image), WIRE_FIELD(source), WIRE_FIELD(timestamp), WIRE_FIELD(unlock_time), WIRE_FIELD(mixin_count), wire::optional_field("payment_id", std::ref(payment_id)) ); } } void read_bytes(wire::reader& source, spend& dest) { boost::optional payment_id; map_spend(source, dest, payment_id); if (payment_id) { dest.length = sizeof(dest.payment_id); dest.payment_id = std::move(*payment_id); } else dest.length = 0; } void write_bytes(wire::writer& dest, const spend& source) { crypto::hash const* const payment_id = (source.length == sizeof(source.payment_id) ? std::addressof(source.payment_id) : nullptr); return map_spend(dest, source, payment_id); } namespace { template void map_key_image(F& format, T& self) { wire::object(format, wire::field("key_image", std::ref(self.value)), wire::field("tx_hash", std::ref(self.link.tx_hash)), wire::field("height", std::ref(self.link.height)) ); } } WIRE_DEFINE_OBJECT(key_image, map_key_image); void write_bytes(wire::writer& dest, const request_info& self, const bool show_key) { db::view_key const* const key = show_key ? std::addressof(self.key) : nullptr; const bool generated = (self.creation_flags & lws::db::account_generated_locally); wire::object(dest, WIRE_FIELD(address), wire::optional_field("view_key", key), WIRE_FIELD(start_height), wire::field("generated_locally", generated) ); } /*! TODO consider making an `operator<` for `crypto::tx_hash`. Not known to be needed elsewhere yet. */ bool operator<(transaction_link const& left, transaction_link const& right) noexcept { return left.height == right.height ? std::memcmp(std::addressof(left.tx_hash), std::addressof(right.tx_hash), sizeof(left.tx_hash)) < 0 : left.height < right.height; } bool operator<=(transaction_link const& left, transaction_link const& right) noexcept { return right.height == left.height ? std::memcmp(std::addressof(left.tx_hash), std::addressof(right.tx_hash), sizeof(left.tx_hash)) <= 0 : left.height < right.height; } } // db } // lws