6#include <unordered_map>
8#include <userver/http/common_headers.hpp>
9#include <userver/http/url.hpp>
11USERVER_NAMESPACE_BEGIN
13namespace s3api::api_methods {
16class StringWriter :
public pugi::xml_writer {
18 explicit StringWriter(std::string& out) : out_(out) {}
20 void write(
const void* data, size_t size)
override { out_.append(
static_cast<
const char*>(data), size); }
27constexpr std::string_view kDelimeter =
"delimiter";
28constexpr std::string_view kEncodingType =
"encoding-type";
29constexpr std::string_view kMarker =
"marker";
30constexpr std::string_view kMaxKeys =
"max-keys";
31constexpr std::string_view kMaxParts =
"max-parts";
32constexpr std::string_view kMaxUploads =
"max-uploads";
33constexpr std::string_view kPartNumber =
"partNumber";
34constexpr std::string_view kPartNumberMarker =
"part-number-marker";
35constexpr std::string_view kPrefix =
"prefix";
36constexpr std::string_view kUploadId =
"uploadId";
37constexpr std::string_view kUploadIdMarker =
"upload-id-marker";
40namespace content_types {
41constexpr std::string_view kDefault =
"application/octet-stream";
42constexpr std::string_view kApplicationXml =
"application/xml; charset=UTF-8";
45constexpr std::string_view kEncodingTypeValueUrl =
"url";
47constexpr std::string_view kS3AwsXmlNamespace =
"http://s3.amazonaws.com/doc/2006-03-01/";
50const unsigned kMaxUploadsListLimit = 1000u;
55 std::string_view bucket,
56 std::string_view path,
58 std::string_view content_type,
59 const std::optional<std::string_view>& content_disposition
66 req.headers
[USERVER_NAMESPACE::
http::
headers::kContentLength
] = std::to_string(data.size());
67 req.headers
[USERVER_NAMESPACE::
http::
headers::kContentType
] = content_type;
69 if (content_disposition.has_value()) {
70 req.headers
[USERVER_NAMESPACE::
http::
headers::kContentDisposition
] = *content_disposition;
73 req.body = std::move(data);
77Request DeleteObject(std::string_view bucket, std::string_view path) {
79 req.method = clients::http::
HttpMethod::kDelete;
85Request GetObject(std::string_view bucket, std::string_view path, std::optional<std::string_view> version) {
98Request GetObjectHead(std::string_view bucket, std::string_view path) {
100 req.method = clients::http::
HttpMethod::kHead;
106void SetRange(
Request& req, size_t begin, size_t end) {
108 "bytes=" + std::to_string(begin) +
'-' + std::to_string(end);
116 std::string_view bucket,
117 std::string_view path,
119 std::string_view marker,
120 std::string_view delimiter
126 std::unordered_map<std::string, std::string> params;
127 params.emplace(
"prefix", path);
129 params.emplace(query_args::kMaxKeys, std::to_string(max_keys));
132 if (!marker.empty()) {
133 params.emplace(query_args::kMarker, marker);
136 if (!delimiter.empty()) {
137 params.emplace(query_args::kDelimeter, delimiter);
144 std::string_view source_bucket,
145 std::string_view source_key,
146 std::string_view dest_bucket,
147 std::string_view dest_key,
148 std::string_view content_type
152 req.bucket = dest_bucket;
155 req.headers
[headers::kAmzCopySource
] = fmt::format(
"/{}/{}", source_bucket, source_key);
156 req.headers
[USERVER_NAMESPACE::
http::
headers::kContentType
] = content_type;
164 result.method = clients::http::
HttpMethod::kPost;
165 result.bucket = bucket;
166 result.req = fmt::format(
"{}?uploads", request.key);
168 if (request.content_type) {
169 result.headers
[http::
headers::kContentType
] = *request.content_type;
172 if (request.content_disposition) {
173 result.headers
[http::
headers::kContentDisposition
] = *request.content_disposition;
176 if (request.content_encoding) {
177 result.headers
[http::
headers::kContentEncoding
] = *request.content_encoding;
180 if (request.content_language) {
181 result.headers
[http::
headers::kContentLanguage
] = *request.content_language;
190 result.method = clients::http::
HttpMethod::kDelete;
191 result.bucket = bucket;
192 result.req = fmt::format(
"{}?{}", request.key,
http::MakeQuery({{query_args::kUploadId, request.upload_id}}
));
199 result.method = clients::http::
HttpMethod::kPost;
200 result.bucket = bucket;
201 result.req = fmt::format(
"{}?{}", request.key,
http::MakeQuery({{query_args::kUploadId, request.upload_id}}
));
203 pugi::xml_document doc;
204 auto multipart_upload_node = doc.append_child(
"CompleteMultipartUpload");
205 multipart_upload_node.append_attribute(
"xmlns").set_value(kS3AwsXmlNamespace.data());
206 for (
const auto& part : request.completed_parts) {
207 auto part_node = multipart_upload_node.append_child(
"Part");
208 part_node.append_child(
"PartNumber")
209 .append_child(pugi::node_pcdata)
210 .set_value(std::to_string(part.part_number).c_str());
211 part_node.append_child(
"ETag").append_child(pugi::node_pcdata).set_value(part.etag.c_str());
213 StringWriter writer{result.body};
214 doc.save(writer,
"", pugi::format_raw);
215 result.headers
= clients
::http
::Headers{
216 {
http::
headers::kContentType, content_types::kApplicationXml},
217 {
http::
headers::kContentLength, std::to_string(result.body.size())}};
224 result.method = clients::http::
HttpMethod::kPut;
229 result.headers
= clients
::http
::Headers{
230 {
http::
headers::kContentLength, std::to_string(request.data.size())},
231 {
http::
headers::kContentType, std::string{content_types::kDefault}}};
233 result.bucket = bucket;
234 result.req = fmt::format(
238 {{query_args::kUploadId, request.upload_id}, {query_args::kPartNumber, std::to_string(request.part_number)}}
241 result.body = std::move(request.data);
245Request CreateInternalApiRequest(
const std::string& bucket,
const multipart_upload::
ListPartsRequest& request) {
247 result.method = clients::http::
HttpMethod::kGet;
248 result.bucket = bucket;
249 result.req = request.key;
251 http::Args params{{std::string{query_args::kUploadId}, request.upload_id}};
253 if (request.max_parts > 0 && request.max_parts < kMaxUploadsListLimit) {
254 params.emplace(query_args::kMaxParts, std::to_string(request.max_parts));
257 if (
auto part_number_marker = request.part_number_marker.value_or(0u)) {
258 params.emplace(query_args::kPartNumberMarker, std::to_string(part_number_marker));
271 result.method = clients::http::
HttpMethod::kGet;
272 result.bucket = bucket;
275 if (request.delimiter) {
276 params.emplace(query_args::kDelimeter, *request.delimiter);
279 if (request.prefix) {
280 params.emplace(query_args::kPrefix, *request.prefix);
283 if (request.max_uploads > 0 && request.max_uploads < kMaxUploadsListLimit) {
284 params.emplace(query_args::kMaxUploads, std::to_string(request.max_uploads));
287 if (request.key_marker) {
288 params.emplace(query_args::kMarker, *request.key_marker);
291 if (request.upload_id_marker) {
292 params.emplace(query_args::kUploadIdMarker, *request.upload_id_marker);
295 if (request.encoding_type == EncodingType::kUrl) {
296 params.emplace(query_args::kEncodingType, kEncodingTypeValueUrl);
299 result.req =
"?uploads";
300 if (!params.empty()) {