|
2 | 2 | #include "aws/s3/private/s3_util.h"
|
3 | 3 | #include <aws/cal/hash.h>
|
4 | 4 | #include <aws/checksums/crc.h>
|
| 5 | +#include <aws/http/request_response.h> |
5 | 6 |
|
6 | 7 | #define AWS_CRC32_LEN sizeof(uint32_t)
|
7 | 8 | #define AWS_CRC32C_LEN sizeof(uint32_t)
|
@@ -330,17 +331,109 @@ int aws_checksum_compute(
|
330 | 331 | }
|
331 | 332 | }
|
332 | 333 |
|
333 |
| -void aws_checksum_config_storage_init( |
| 334 | +static int s_init_and_verify_checksum_config_from_headers( |
| 335 | + struct checksum_config_storage *checksum_config, |
| 336 | + const struct aws_http_message *message, |
| 337 | + const void *log_id) { |
| 338 | + /* Check if the checksum header was set from the message */ |
| 339 | + struct aws_http_headers *headers = aws_http_message_get_headers(message); |
| 340 | + enum aws_s3_checksum_algorithm header_algo = AWS_SCA_NONE; |
| 341 | + struct aws_byte_cursor header_value; |
| 342 | + AWS_ZERO_STRUCT(header_value); |
| 343 | + |
| 344 | + for (size_t i = 0; i < AWS_ARRAY_SIZE(s_checksum_algo_priority_list); i++) { |
| 345 | + enum aws_s3_checksum_algorithm algorithm = s_checksum_algo_priority_list[i]; |
| 346 | + const struct aws_byte_cursor algorithm_header_name = |
| 347 | + aws_get_http_header_name_from_checksum_algorithm(algorithm); |
| 348 | + if (aws_http_headers_get(headers, algorithm_header_name, &header_value) == AWS_OP_SUCCESS) { |
| 349 | + if (header_algo == AWS_SCA_NONE) { |
| 350 | + header_algo = algorithm; |
| 351 | + } else { |
| 352 | + /* If there are multiple checksum headers set, it's malformed request */ |
| 353 | + AWS_LOGF_ERROR( |
| 354 | + AWS_LS_S3_META_REQUEST, |
| 355 | + "id=%p Could not create auto-ranged-put meta request; multiple checksum headers has been set", |
| 356 | + log_id); |
| 357 | + return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT); |
| 358 | + } |
| 359 | + } |
| 360 | + } |
| 361 | + if (header_algo == AWS_SCA_NONE) { |
| 362 | + /* No checksum header found, done */ |
| 363 | + return AWS_OP_SUCCESS; |
| 364 | + } |
| 365 | + |
| 366 | + if (checksum_config->has_full_object_checksum) { |
| 367 | + /* If the full object checksum has been set, it's malformed request */ |
| 368 | + AWS_LOGF_ERROR( |
| 369 | + AWS_LS_S3_META_REQUEST, |
| 370 | + "id=%p: Could not create auto-ranged-put meta request; full object checksum is set from multiple ways.", |
| 371 | + log_id); |
| 372 | + return aws_raise_error(AWS_ERROR_INVALID_ARGUMENT); |
| 373 | + } |
| 374 | + |
| 375 | + AWS_LOGF_DEBUG( |
| 376 | + AWS_LS_S3_META_REQUEST, |
| 377 | + "id=%p Setting the full-object checksum from header; algorithm: " PRInSTR ", value: " PRInSTR ".", |
| 378 | + log_id, |
| 379 | + AWS_BYTE_CURSOR_PRI(aws_get_checksum_algorithm_name(header_algo)), |
| 380 | + AWS_BYTE_CURSOR_PRI(header_value)); |
| 381 | + /* Set algo */ |
| 382 | + checksum_config->checksum_algorithm = header_algo; |
| 383 | + /** |
| 384 | + * Set the location to NONE to avoid adding extra checksums from client. |
| 385 | + * |
| 386 | + * Notes: The multipart upload will set the location to trailer to add parts level checksums. |
| 387 | + **/ |
| 388 | + checksum_config->location = AWS_SCL_NONE; |
| 389 | + |
| 390 | + /* Set full object checksum from the header value. */ |
| 391 | + aws_byte_buf_init_copy_from_cursor( |
| 392 | + &checksum_config->full_object_checksum, checksum_config->allocator, header_value); |
| 393 | + checksum_config->has_full_object_checksum = true; |
| 394 | + return AWS_OP_SUCCESS; |
| 395 | +} |
| 396 | + |
| 397 | +int aws_checksum_config_storage_init( |
334 | 398 | struct aws_allocator *allocator,
|
335 | 399 | struct checksum_config_storage *internal_config,
|
336 |
| - const struct aws_s3_checksum_config *config) { |
| 400 | + const struct aws_s3_checksum_config *config, |
| 401 | + const struct aws_http_message *message, |
| 402 | + const void *log_id) { |
337 | 403 | AWS_ZERO_STRUCT(*internal_config);
|
338 | 404 | /* Zero out the struct and set the allocator regardless. */
|
339 | 405 | internal_config->allocator = allocator;
|
340 | 406 |
|
341 | 407 | if (!config) {
|
342 |
| - return; |
| 408 | + return AWS_OP_SUCCESS; |
| 409 | + } |
| 410 | + |
| 411 | + struct aws_http_headers *headers = aws_http_message_get_headers(message); |
| 412 | + if (config->location == AWS_SCL_TRAILER) { |
| 413 | + struct aws_byte_cursor existing_encoding; |
| 414 | + AWS_ZERO_STRUCT(existing_encoding); |
| 415 | + if (aws_http_headers_get(headers, g_content_encoding_header_name, &existing_encoding) == AWS_OP_SUCCESS) { |
| 416 | + if (aws_byte_cursor_find_exact(&existing_encoding, &g_content_encoding_header_aws_chunked, NULL) == |
| 417 | + AWS_OP_SUCCESS) { |
| 418 | + AWS_LOGF_ERROR( |
| 419 | + AWS_LS_S3_META_REQUEST, |
| 420 | + "id=%p Cannot create meta s3 request; for trailer checksum, the original request cannot be " |
| 421 | + "aws-chunked encoding. The client will encode the request instead.", |
| 422 | + (void *)log_id); |
| 423 | + aws_raise_error(AWS_ERROR_INVALID_ARGUMENT); |
| 424 | + return AWS_OP_ERR; |
| 425 | + } |
| 426 | + } |
343 | 427 | }
|
| 428 | + if (config->location != AWS_SCL_NONE && config->checksum_algorithm == AWS_SCA_NONE) { |
| 429 | + AWS_LOGF_ERROR( |
| 430 | + AWS_LS_S3_META_REQUEST, |
| 431 | + "id=%p Cannot create meta s3 request; checksum location is set, but no checksum algorithm selected.", |
| 432 | + (void *)log_id); |
| 433 | + aws_raise_error(AWS_ERROR_INVALID_ARGUMENT); |
| 434 | + return AWS_OP_ERR; |
| 435 | + } |
| 436 | + |
344 | 437 | internal_config->checksum_algorithm = config->checksum_algorithm;
|
345 | 438 | internal_config->location = config->location;
|
346 | 439 | internal_config->validate_response_checksum = config->validate_response_checksum;
|
@@ -385,6 +478,14 @@ void aws_checksum_config_storage_init(
|
385 | 478 | internal_config->response_checksum_algorithms.sha1 = true;
|
386 | 479 | internal_config->response_checksum_algorithms.sha256 = true;
|
387 | 480 | }
|
| 481 | + |
| 482 | + /* After applying settings from config, check the message header to override the corresponding settings. */ |
| 483 | + if (s_init_and_verify_checksum_config_from_headers(internal_config, message, log_id)) { |
| 484 | + return AWS_OP_ERR; |
| 485 | + } |
| 486 | + /* Anything fail afterward will need to cleanup the storage. */ |
| 487 | + |
| 488 | + return AWS_OP_SUCCESS; |
388 | 489 | }
|
389 | 490 |
|
390 | 491 | void aws_checksum_config_storage_cleanup(struct checksum_config_storage *internal_config) {
|
|
0 commit comments