1#include <userver/s3api/models/multipart_upload/responses.hpp>
6#include <userver/s3api/models/errors.hpp>
7#include <userver/utils/datetime.hpp>
8#include <userver/utils/from_string.hpp>
10USERVER_NAMESPACE_BEGIN
12namespace s3api::multipart_upload {
15pugi::xml_node GetRequiredChildNode(
const pugi::xml_document& doc,
const char* child_name) {
16 auto child = doc.child(child_name);
23std::optional<std::string_view> ExtractChildValue(
24 const pugi::xml_node& node,
25 const char* child_name,
26 bool is_empty_allowed =
true
28 const auto child = node.child(child_name);
29 auto result = (!child.empty() ? std::make_optional<std::string_view>(child.child_value()) : std::nullopt);
30 if (!is_empty_allowed && result && result->empty()) {
31 throw ResponseParsingError(fmt::format(
"node '{}.{}' is not expected to be empty", node.name(), child_name));
36std::string_view ExtractRequiredChildValue(
const pugi::xml_node& node,
const char* child_name) {
37 const auto result = ExtractChildValue(node, child_name,
false);
40 fmt::format(
"node '{}' is missing the required child node '{}'", node.name(), child_name)
46bool ToBoolean(
const std::optional<std::string_view>& maybe_str) {
47 if (maybe_str.value_or(
"false") !=
"true") {
54std::enable_if_t<std::is_integral_v<T>, std::optional<T>> ExtractChildValueAsIntegral(
55 const pugi::xml_node& node,
56 const char* child_name
59 const auto maybe_str = ExtractChildValue(node, child_name,
false);
64 return USERVER_NAMESPACE::utils::FromString<T>(*maybe_str);
66}
catch (
const std::exception& exc) {
68 fmt::format(
"failed to parse node '{}.{}' value as integral type - {}", node.name(), child_name, exc.what())
73std::enable_if_t<std::is_integral_v<T>, T> ExtractRequiredChildValueAsIntegral(
74 const pugi::xml_node& node,
75 const char* child_name
77 const auto maybe_value = ExtractChildValueAsIntegral<T>(node, child_name);
80 fmt::format(
"node '{}' is missing the required child node '{}'", node.name(), child_name)
86constexpr auto kExtractRequiredChildValueAsUInt = ExtractRequiredChildValueAsIntegral<
unsigned>;
87constexpr auto kExtractChildValueAsUInt = ExtractChildValueAsIntegral<
unsigned>;
88constexpr auto kExtractChildValueAsULong = ExtractChildValueAsIntegral<
unsigned long>;
94 pugi::xml_document xml;
95 const pugi::xml_parse_result
96 parse_result = xml.load_string(http_s3_respose_body.c_str(), pugi::parse_default | pugi::parse_escapes);
97 if (parse_result.status != pugi::status_ok) {
100 const auto xml_root_item = GetRequiredChildNode(xml,
"InitiateMultipartUploadResult");
101 result.bucket = ExtractRequiredChildValue(xml_root_item,
"Bucket");
102 result.upload_id = ExtractRequiredChildValue(xml_root_item,
"UploadId");
103 result.key = ExtractRequiredChildValue(xml_root_item,
"Key");
109 pugi::xml_document xml;
110 const pugi::xml_parse_result
111 parse_result = xml.load_string(http_s3_respose_body.c_str(), pugi::parse_default | pugi::parse_escapes);
112 if (parse_result.status != pugi::status_ok) {
115 const auto xml_root_item = GetRequiredChildNode(xml,
"CompleteMultipartUploadResult");
116 result.location = ExtractRequiredChildValue(xml_root_item,
"Location");
117 result.bucket = ExtractRequiredChildValue(xml_root_item,
"Bucket");
118 result.key = ExtractRequiredChildValue(xml_root_item,
"Key");
124 pugi::xml_document xml;
125 const pugi::xml_parse_result
126 parse_result = xml.load_string(http_s3_respose_body.c_str(), pugi::parse_default | pugi::parse_escapes);
127 if (parse_result.status != pugi::status_ok) {
131 const auto xml_root_item = GetRequiredChildNode(xml,
"ListMultipartUploadsResult");
132 result.bucket = ExtractRequiredChildValue(xml_root_item,
"Bucket");
133 result.key_marker = ExtractChildValue(xml_root_item,
"KeyMarker");
134 result.upload_id_marker = ExtractChildValue(xml_root_item,
"UploadIdMarker");
135 result.next_key_marker = ExtractChildValue(xml_root_item,
"NextKeyMarker");
136 result.next_upload_id_marker = ExtractChildValue(xml_root_item,
"NextUploadIdMarker");
137 result.delimiter = ExtractChildValue(xml_root_item,
"Delimiter");
138 result.is_truncated = ToBoolean(ExtractChildValue(xml_root_item,
"IsTruncated"));
139 result.max_uploads = kExtractChildValueAsUInt(xml_root_item,
"MaxUploads");
141 for (
auto node = xml_root_item.child(
"Upload"); node; node = node.next_sibling(
"Upload")) {
143 multipart_upload.key = ExtractRequiredChildValue(node,
"Key");
144 multipart_upload.upload_id = ExtractRequiredChildValue(node,
"UploadId");
145 if (
const auto maybe_initiated_str = ExtractChildValue(node,
"Initiated")) {
148 multipart_upload.initiated_ts =
151 result.uploads.push_back(std::move(multipart_upload));
154 for (
auto node = xml_root_item.child(
"CommonPrefixes"); node; node = node.next_sibling(
"CommonPrefixes")) {
155 result.common_prefixes.emplace_back(ExtractRequiredChildValue(node,
"Prefix"));
163 pugi::xml_document xml;
164 const pugi::xml_parse_result
165 parse_result = xml.load_string(http_s3_respose_body.c_str(), pugi::parse_default | pugi::parse_escapes);
166 if (parse_result.status != pugi::status_ok) {
170 const auto xml_root_item = GetRequiredChildNode(xml,
"ListPartsResult");
171 result.bucket = ExtractRequiredChildValue(xml_root_item,
"Bucket");
172 result.upload_id = ExtractRequiredChildValue(xml_root_item,
"UploadId");
173 result.max_parts = kExtractChildValueAsUInt(xml_root_item,
"MaxParts");
174 result.part_number_marker = kExtractChildValueAsUInt(xml_root_item,
"PartNumberMarker");
175 result.next_part_number_marker = kExtractChildValueAsUInt(xml_root_item,
"NextPartNumberMarker");
176 result.key = ExtractRequiredChildValue(xml_root_item,
"Key");
177 result.is_truncated = ToBoolean(ExtractChildValue(xml_root_item,
"IsTruncated"));
179 for (
auto part_node = xml_root_item.child(
"Part"); part_node; part_node = part_node.next_sibling(
"Part")) {
181 part.etag = ExtractRequiredChildValue(part_node,
"ETag");
182 part.part_number = kExtractRequiredChildValueAsUInt(part_node,
"PartNumber");
183 part.byte_size = kExtractChildValueAsULong(part_node,
"Size");
185 if (
const auto maybe_last_modified_ts_str = ExtractChildValue(part_node,
"LastModified")) {
188 part.last_modified_ts =
192 result.parts.push_back(std::move(part));