userver: userver/storages/postgres/io/field_buffer.hpp Source File
Loading...
Searching...
No Matches
field_buffer.hpp
1#pragma once
2
3#include <userver/storages/postgres/exceptions.hpp>
4#include <userver/storages/postgres/io/buffer_io.hpp>
5#include <userver/storages/postgres/io/integral_types.hpp>
6#include <userver/storages/postgres/io/nullable_traits.hpp>
7
8USERVER_NAMESPACE_BEGIN
9
10namespace storages::postgres::io {
11
12inline constexpr FieldBuffer FieldBuffer::GetSubBuffer(std::size_t offset, std::size_t size, BufferCategory cat) const {
13 const auto* new_buffer_start = buffer + offset;
14 if (offset > length) {
15 throw InvalidInputBufferSize(fmt::format("Offset {} requested for a buffer of size {}", offset, length));
16 }
17 size = size == npos ? length - offset : size;
18 if (offset + size > length) {
20 fmt::format("Attempt to read {} bytes more than was sent by server", offset + size - length)
21 );
22 }
24 cat = category;
25 }
26 return {is_null, cat, size, new_buffer_start};
27}
28
29template <typename T>
30std::size_t FieldBuffer::Read(T&& value, BufferCategory cat, std::size_t len) {
31 io::ReadBuffer(GetSubBuffer(0, len, cat), std::forward<T>(value));
32 length -= len;
33 buffer += len;
34 return len;
35}
36
37template <typename T>
38std::size_t FieldBuffer::Read(T&& value, const TypeBufferCategory& categories, std::size_t len, BufferCategory cat) {
39 io::ReadBuffer(GetSubBuffer(0, len, cat), std::forward<T>(value), categories);
40 length -= len;
41 buffer += len;
42 return len;
43}
44
45template <typename T>
46std::size_t FieldBuffer::ReadRaw(T&& value, const TypeBufferCategory& categories, BufferCategory cat) {
47 using ValueType = std::remove_cvref_t<T>;
48 Integer field_length{0};
49 auto consumed = Read(field_length, BufferCategory::kPlainBuffer);
50 if (field_length == kPgNullBufferSize) {
51 // NULL value
52 traits::GetSetNull<ValueType>::SetNull(value);
53 return consumed;
54 } else if (field_length < 0) {
55 // invalid length value
56 throw InvalidInputBufferSize(fmt::format("Negative buffer size value {}", field_length));
57 } else if (field_length == 0) {
58 traits::GetSetNull<ValueType>::SetDefault(value);
59 return consumed;
60 } else {
61 return consumed + Read(std::forward<T>(value), categories, field_length, cat);
62 }
63}
64
65template <typename T>
66std::size_t ReadRawBinary(FieldBuffer buffer, T& value, const TypeBufferCategory& categories) {
67 return buffer.ReadRaw(value, categories);
68}
69
70namespace detail {
71
72template <typename T, typename Buffer>
73inline constexpr bool kFormatterAcceptsReplacementOid = requires(T& f, const UserTypes& types, Buffer& buf, Oid oid) {
74 f(types, buf, oid);
75};
76
77} // namespace detail
78
79template <typename T, typename Buffer>
80void WriteRawBinary(
81 const UserTypes& types,
82 Buffer& buffer,
83 const T& value,
84 [[maybe_unused]] Oid replace_oid = kInvalidOid
85) {
86 traits::CheckFormatter<T>();
87 static constexpr auto size_len = sizeof(Integer);
88 if (traits::GetSetNull<T>::IsNull(value)) {
89 io::WriteBuffer(types, buffer, kPgNullBufferSize);
90 } else {
91 using BufferFormatter = typename traits::IO<T>::FormatterType;
92 auto len_start = buffer.size();
93 buffer.resize(buffer.size() + size_len);
94 auto size_before = buffer.size();
95 if constexpr (detail::kFormatterAcceptsReplacementOid<BufferFormatter, Buffer>) {
96 BufferFormatter{value}(types, buffer, replace_oid);
97 } else {
98 io::WriteBuffer(types, buffer, value);
99 }
100 const Integer bytes = buffer.size() - size_before;
101 BufferWriter(bytes)(buffer.begin() + len_start);
102 }
103}
104
105} // namespace storages::postgres::io
106
107USERVER_NAMESPACE_END