aws-sdk-dynamodb 0.1.0__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- aws_sdk_dynamodb/__init__.py +17 -0
- aws_sdk_dynamodb/_async.py +25 -0
- aws_sdk_dynamodb/_auth/_identity.py +16 -0
- aws_sdk_dynamodb/_auth/_providers.py +162 -0
- aws_sdk_dynamodb/_auth/_signers.py +83 -0
- aws_sdk_dynamodb/_auth/_sigv4.py +364 -0
- aws_sdk_dynamodb/_auth/_zapros_handler.py +62 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/batch_execute_statement.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/batch_get_item.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/batch_write_item.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/create_backup.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/create_global_table.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/create_table.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/delete_backup.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/delete_item.py +25 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/delete_resource_policy.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/delete_table.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_backup.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_continuous_backups.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_contributor_insights.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_endpoints.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_export.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_global_table.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_global_table_settings.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_import.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_kinesis_streaming_destination.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_limits.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_table.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_table_replica_auto_scaling.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/describe_time_to_live.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/disable_kinesis_streaming_destination.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/enable_kinesis_streaming_destination.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/execute_statement.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/execute_transaction.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/export_table_to_point_in_time.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/get_item.py +24 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/get_resource_policy.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/import_table.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_backups.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_contributor_insights.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_exports.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_global_tables.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_imports.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_tables.py +25 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/list_tags_of_resource.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/put_item.py +24 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/put_resource_policy.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/query.py +23 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/restore_table_from_backup.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/restore_table_to_point_in_time.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/scan.py +23 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/tag_resource.py +24 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/transact_get_items.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/transact_write_items.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/untag_resource.py +24 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_continuous_backups.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_contributor_insights.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_global_table.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_global_table_settings.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_item.py +25 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_kinesis_streaming_destination.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_table.py +29 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_table_replica_auto_scaling.py +31 -0
- aws_sdk_dynamodb/_operations/dynamo_db_20120810/update_time_to_live.py +31 -0
- aws_sdk_dynamodb/_pagination.py +21 -0
- aws_sdk_dynamodb/_protocol/__init__.py +1 -0
- aws_sdk_dynamodb/_protocol/errors.py +58 -0
- aws_sdk_dynamodb/_protocol/xml.py +27 -0
- aws_sdk_dynamodb/_rule_engine/__init__.py +0 -0
- aws_sdk_dynamodb/_rule_engine/_aws_partition.py +160 -0
- aws_sdk_dynamodb/_rule_engine/_endpoint_rule_set.py +707 -0
- aws_sdk_dynamodb/_rule_engine/_endpoint_runtime.py +389 -0
- aws_sdk_dynamodb/_services/_pipeline.py +202 -0
- aws_sdk_dynamodb/_services/async_dynamo_db_20120810.py +3979 -0
- aws_sdk_dynamodb/_services/dynamo_db_20120810.py +3911 -0
- aws_sdk_dynamodb/errors/__init__.py +94 -0
- aws_sdk_dynamodb/errors/_base.py +94 -0
- aws_sdk_dynamodb/errors/backup_in_use_exception.py +27 -0
- aws_sdk_dynamodb/errors/backup_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/conditional_check_failed_exception.py +31 -0
- aws_sdk_dynamodb/errors/continuous_backups_unavailable_exception.py +27 -0
- aws_sdk_dynamodb/errors/duplicate_item_exception.py +27 -0
- aws_sdk_dynamodb/errors/export_conflict_exception.py +27 -0
- aws_sdk_dynamodb/errors/export_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/global_table_already_exists_exception.py +27 -0
- aws_sdk_dynamodb/errors/global_table_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/idempotent_parameter_mismatch_exception.py +27 -0
- aws_sdk_dynamodb/errors/import_conflict_exception.py +27 -0
- aws_sdk_dynamodb/errors/import_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/index_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/internal_server_error.py +28 -0
- aws_sdk_dynamodb/errors/invalid_endpoint_exception.py +27 -0
- aws_sdk_dynamodb/errors/invalid_export_time_exception.py +27 -0
- aws_sdk_dynamodb/errors/invalid_restore_time_exception.py +27 -0
- aws_sdk_dynamodb/errors/item_collection_size_limit_exceeded_exception.py +28 -0
- aws_sdk_dynamodb/errors/limit_exceeded_exception.py +28 -0
- aws_sdk_dynamodb/errors/point_in_time_recovery_unavailable_exception.py +27 -0
- aws_sdk_dynamodb/errors/policy_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/provisioned_throughput_exceeded_exception.py +33 -0
- aws_sdk_dynamodb/errors/replica_already_exists_exception.py +27 -0
- aws_sdk_dynamodb/errors/replica_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/replicated_write_conflict_exception.py +27 -0
- aws_sdk_dynamodb/errors/request_limit_exceeded.py +32 -0
- aws_sdk_dynamodb/errors/resource_in_use_exception.py +28 -0
- aws_sdk_dynamodb/errors/resource_not_found_exception.py +28 -0
- aws_sdk_dynamodb/errors/table_already_exists_exception.py +27 -0
- aws_sdk_dynamodb/errors/table_in_use_exception.py +27 -0
- aws_sdk_dynamodb/errors/table_not_found_exception.py +27 -0
- aws_sdk_dynamodb/errors/throttling_exception.py +34 -0
- aws_sdk_dynamodb/errors/transaction_canceled_exception.py +32 -0
- aws_sdk_dynamodb/errors/transaction_conflict_exception.py +27 -0
- aws_sdk_dynamodb/errors/transaction_in_progress_exception.py +27 -0
- aws_sdk_dynamodb/py.typed +0 -0
- aws_sdk_dynamodb/types/approximate_creation_date_time_precision.py +8 -0
- aws_sdk_dynamodb/types/archival_reason.py +5 -0
- aws_sdk_dynamodb/types/archival_summary.py +20 -0
- aws_sdk_dynamodb/types/attribute_action.py +9 -0
- aws_sdk_dynamodb/types/attribute_definition.py +16 -0
- aws_sdk_dynamodb/types/attribute_definitions.py +10 -0
- aws_sdk_dynamodb/types/attribute_map.py +12 -0
- aws_sdk_dynamodb/types/attribute_name.py +5 -0
- aws_sdk_dynamodb/types/attribute_name_list.py +10 -0
- aws_sdk_dynamodb/types/attribute_updates.py +12 -0
- aws_sdk_dynamodb/types/attribute_value.py +69 -0
- aws_sdk_dynamodb/types/attribute_value_list.py +10 -0
- aws_sdk_dynamodb/types/attribute_value_update.py +15 -0
- aws_sdk_dynamodb/types/auto_scaling_policy_description.py +19 -0
- aws_sdk_dynamodb/types/auto_scaling_policy_description_list.py +10 -0
- aws_sdk_dynamodb/types/auto_scaling_policy_name.py +5 -0
- aws_sdk_dynamodb/types/auto_scaling_policy_update.py +17 -0
- aws_sdk_dynamodb/types/auto_scaling_role_arn.py +5 -0
- aws_sdk_dynamodb/types/auto_scaling_settings_description.py +31 -0
- aws_sdk_dynamodb/types/auto_scaling_settings_update.py +33 -0
- aws_sdk_dynamodb/types/auto_scaling_target_tracking_scaling_policy_configuration_description.py +24 -0
- aws_sdk_dynamodb/types/auto_scaling_target_tracking_scaling_policy_configuration_update.py +24 -0
- aws_sdk_dynamodb/types/availability_error_message.py +5 -0
- aws_sdk_dynamodb/types/backfilling.py +5 -0
- aws_sdk_dynamodb/types/backup_arn.py +5 -0
- aws_sdk_dynamodb/types/backup_creation_date_time.py +6 -0
- aws_sdk_dynamodb/types/backup_description.py +22 -0
- aws_sdk_dynamodb/types/backup_details.py +34 -0
- aws_sdk_dynamodb/types/backup_name.py +5 -0
- aws_sdk_dynamodb/types/backup_size_bytes.py +5 -0
- aws_sdk_dynamodb/types/backup_status.py +9 -0
- aws_sdk_dynamodb/types/backup_summaries.py +8 -0
- aws_sdk_dynamodb/types/backup_summary.py +43 -0
- aws_sdk_dynamodb/types/backup_type.py +9 -0
- aws_sdk_dynamodb/types/backup_type_filter.py +10 -0
- aws_sdk_dynamodb/types/backups_input_limit.py +5 -0
- aws_sdk_dynamodb/types/batch_execute_statement_input.py +16 -0
- aws_sdk_dynamodb/types/batch_execute_statement_output.py +19 -0
- aws_sdk_dynamodb/types/batch_get_item_input.py +16 -0
- aws_sdk_dynamodb/types/batch_get_item_output.py +24 -0
- aws_sdk_dynamodb/types/batch_get_request_map.py +12 -0
- aws_sdk_dynamodb/types/batch_get_response_map.py +12 -0
- aws_sdk_dynamodb/types/batch_statement_error.py +20 -0
- aws_sdk_dynamodb/types/batch_statement_error_code_enum.py +17 -0
- aws_sdk_dynamodb/types/batch_statement_request.py +27 -0
- aws_sdk_dynamodb/types/batch_statement_response.py +20 -0
- aws_sdk_dynamodb/types/batch_write_item_input.py +23 -0
- aws_sdk_dynamodb/types/batch_write_item_output.py +24 -0
- aws_sdk_dynamodb/types/batch_write_item_request_map.py +12 -0
- aws_sdk_dynamodb/types/billed_size_bytes.py +5 -0
- aws_sdk_dynamodb/types/billing_mode.py +8 -0
- aws_sdk_dynamodb/types/billing_mode_summary.py +17 -0
- aws_sdk_dynamodb/types/binary_attribute_value.py +5 -0
- aws_sdk_dynamodb/types/binary_set_attribute_value.py +10 -0
- aws_sdk_dynamodb/types/boolean_attribute_value.py +5 -0
- aws_sdk_dynamodb/types/boolean_object.py +5 -0
- aws_sdk_dynamodb/types/cancellation_reason.py +18 -0
- aws_sdk_dynamodb/types/cancellation_reason_list.py +10 -0
- aws_sdk_dynamodb/types/capacity.py +22 -0
- aws_sdk_dynamodb/types/client_request_token.py +5 -0
- aws_sdk_dynamodb/types/client_token.py +5 -0
- aws_sdk_dynamodb/types/cloud_watch_log_group_arn.py +5 -0
- aws_sdk_dynamodb/types/code.py +5 -0
- aws_sdk_dynamodb/types/comparison_operator.py +19 -0
- aws_sdk_dynamodb/types/condition.py +17 -0
- aws_sdk_dynamodb/types/condition_check.py +35 -0
- aws_sdk_dynamodb/types/condition_expression.py +5 -0
- aws_sdk_dynamodb/types/conditional_operator.py +8 -0
- aws_sdk_dynamodb/types/confirm_remove_self_resource_access.py +7 -0
- aws_sdk_dynamodb/types/consistent_read.py +5 -0
- aws_sdk_dynamodb/types/consumed_capacity.py +37 -0
- aws_sdk_dynamodb/types/consumed_capacity_multiple.py +10 -0
- aws_sdk_dynamodb/types/consumed_capacity_units.py +5 -0
- aws_sdk_dynamodb/types/continuous_backups_description.py +19 -0
- aws_sdk_dynamodb/types/continuous_backups_status.py +8 -0
- aws_sdk_dynamodb/types/contributor_insights_action.py +8 -0
- aws_sdk_dynamodb/types/contributor_insights_mode.py +8 -0
- aws_sdk_dynamodb/types/contributor_insights_rule.py +5 -0
- aws_sdk_dynamodb/types/contributor_insights_rule_list.py +10 -0
- aws_sdk_dynamodb/types/contributor_insights_status.py +11 -0
- aws_sdk_dynamodb/types/contributor_insights_summaries.py +10 -0
- aws_sdk_dynamodb/types/contributor_insights_summary.py +25 -0
- aws_sdk_dynamodb/types/create_backup_input.py +14 -0
- aws_sdk_dynamodb/types/create_backup_output.py +12 -0
- aws_sdk_dynamodb/types/create_global_secondary_index_action.py +33 -0
- aws_sdk_dynamodb/types/create_global_table_input.py +14 -0
- aws_sdk_dynamodb/types/create_global_table_output.py +14 -0
- aws_sdk_dynamodb/types/create_global_table_witness_group_member_action.py +11 -0
- aws_sdk_dynamodb/types/create_replica_action.py +11 -0
- aws_sdk_dynamodb/types/create_replication_group_member_action.py +35 -0
- aws_sdk_dynamodb/types/create_table_input.py +81 -0
- aws_sdk_dynamodb/types/create_table_output.py +14 -0
- aws_sdk_dynamodb/types/csv_delimiter.py +5 -0
- aws_sdk_dynamodb/types/csv_header.py +5 -0
- aws_sdk_dynamodb/types/csv_header_list.py +8 -0
- aws_sdk_dynamodb/types/csv_options.py +15 -0
- aws_sdk_dynamodb/types/date.py +6 -0
- aws_sdk_dynamodb/types/delete.py +35 -0
- aws_sdk_dynamodb/types/delete_backup_input.py +11 -0
- aws_sdk_dynamodb/types/delete_backup_output.py +14 -0
- aws_sdk_dynamodb/types/delete_global_secondary_index_action.py +11 -0
- aws_sdk_dynamodb/types/delete_global_table_witness_group_member_action.py +11 -0
- aws_sdk_dynamodb/types/delete_item_input.py +57 -0
- aws_sdk_dynamodb/types/delete_item_output.py +22 -0
- aws_sdk_dynamodb/types/delete_replica_action.py +11 -0
- aws_sdk_dynamodb/types/delete_replication_group_member_action.py +11 -0
- aws_sdk_dynamodb/types/delete_request.py +11 -0
- aws_sdk_dynamodb/types/delete_resource_policy_input.py +17 -0
- aws_sdk_dynamodb/types/delete_resource_policy_output.py +14 -0
- aws_sdk_dynamodb/types/delete_table_input.py +11 -0
- aws_sdk_dynamodb/types/delete_table_output.py +14 -0
- aws_sdk_dynamodb/types/deletion_protection_enabled.py +5 -0
- aws_sdk_dynamodb/types/describe_backup_input.py +11 -0
- aws_sdk_dynamodb/types/describe_backup_output.py +14 -0
- aws_sdk_dynamodb/types/describe_continuous_backups_input.py +11 -0
- aws_sdk_dynamodb/types/describe_continuous_backups_output.py +14 -0
- aws_sdk_dynamodb/types/describe_contributor_insights_input.py +15 -0
- aws_sdk_dynamodb/types/describe_contributor_insights_output.py +40 -0
- aws_sdk_dynamodb/types/describe_endpoints_request.py +7 -0
- aws_sdk_dynamodb/types/describe_endpoints_response.py +11 -0
- aws_sdk_dynamodb/types/describe_export_input.py +11 -0
- aws_sdk_dynamodb/types/describe_export_output.py +14 -0
- aws_sdk_dynamodb/types/describe_global_table_input.py +11 -0
- aws_sdk_dynamodb/types/describe_global_table_output.py +14 -0
- aws_sdk_dynamodb/types/describe_global_table_settings_input.py +11 -0
- aws_sdk_dynamodb/types/describe_global_table_settings_output.py +17 -0
- aws_sdk_dynamodb/types/describe_import_input.py +11 -0
- aws_sdk_dynamodb/types/describe_import_output.py +13 -0
- aws_sdk_dynamodb/types/describe_kinesis_streaming_destination_input.py +11 -0
- aws_sdk_dynamodb/types/describe_kinesis_streaming_destination_output.py +17 -0
- aws_sdk_dynamodb/types/describe_limits_input.py +7 -0
- aws_sdk_dynamodb/types/describe_limits_output.py +26 -0
- aws_sdk_dynamodb/types/describe_table_input.py +11 -0
- aws_sdk_dynamodb/types/describe_table_output.py +12 -0
- aws_sdk_dynamodb/types/describe_table_replica_auto_scaling_input.py +11 -0
- aws_sdk_dynamodb/types/describe_table_replica_auto_scaling_output.py +14 -0
- aws_sdk_dynamodb/types/describe_time_to_live_input.py +11 -0
- aws_sdk_dynamodb/types/describe_time_to_live_output.py +14 -0
- aws_sdk_dynamodb/types/destination_status.py +12 -0
- aws_sdk_dynamodb/types/double_object.py +5 -0
- aws_sdk_dynamodb/types/enable_kinesis_streaming_configuration.py +14 -0
- aws_sdk_dynamodb/types/endpoint.py +14 -0
- aws_sdk_dynamodb/types/endpoints.py +8 -0
- aws_sdk_dynamodb/types/error_count.py +7 -0
- aws_sdk_dynamodb/types/error_message.py +5 -0
- aws_sdk_dynamodb/types/exception_description.py +5 -0
- aws_sdk_dynamodb/types/exception_name.py +5 -0
- aws_sdk_dynamodb/types/execute_statement_input.py +41 -0
- aws_sdk_dynamodb/types/execute_statement_output.py +24 -0
- aws_sdk_dynamodb/types/execute_transaction_input.py +24 -0
- aws_sdk_dynamodb/types/execute_transaction_output.py +17 -0
- aws_sdk_dynamodb/types/expected_attribute_map.py +12 -0
- aws_sdk_dynamodb/types/expected_attribute_value.py +25 -0
- aws_sdk_dynamodb/types/export_arn.py +5 -0
- aws_sdk_dynamodb/types/export_description.py +84 -0
- aws_sdk_dynamodb/types/export_end_time.py +6 -0
- aws_sdk_dynamodb/types/export_format.py +8 -0
- aws_sdk_dynamodb/types/export_from_time.py +6 -0
- aws_sdk_dynamodb/types/export_manifest.py +5 -0
- aws_sdk_dynamodb/types/export_next_token.py +5 -0
- aws_sdk_dynamodb/types/export_start_time.py +6 -0
- aws_sdk_dynamodb/types/export_status.py +9 -0
- aws_sdk_dynamodb/types/export_summaries.py +8 -0
- aws_sdk_dynamodb/types/export_summary.py +18 -0
- aws_sdk_dynamodb/types/export_table_to_point_in_time_input.py +48 -0
- aws_sdk_dynamodb/types/export_table_to_point_in_time_output.py +14 -0
- aws_sdk_dynamodb/types/export_time.py +6 -0
- aws_sdk_dynamodb/types/export_to_time.py +6 -0
- aws_sdk_dynamodb/types/export_type.py +8 -0
- aws_sdk_dynamodb/types/export_view_type.py +8 -0
- aws_sdk_dynamodb/types/expression_attribute_name_map.py +12 -0
- aws_sdk_dynamodb/types/expression_attribute_name_variable.py +5 -0
- aws_sdk_dynamodb/types/expression_attribute_value_map.py +12 -0
- aws_sdk_dynamodb/types/expression_attribute_value_variable.py +5 -0
- aws_sdk_dynamodb/types/failure_code.py +5 -0
- aws_sdk_dynamodb/types/failure_exception.py +17 -0
- aws_sdk_dynamodb/types/failure_message.py +5 -0
- aws_sdk_dynamodb/types/filter_condition_map.py +12 -0
- aws_sdk_dynamodb/types/get.py +25 -0
- aws_sdk_dynamodb/types/get_item_input.py +39 -0
- aws_sdk_dynamodb/types/get_item_output.py +17 -0
- aws_sdk_dynamodb/types/get_resource_policy_input.py +11 -0
- aws_sdk_dynamodb/types/get_resource_policy_output.py +17 -0
- aws_sdk_dynamodb/types/global_secondary_index.py +33 -0
- aws_sdk_dynamodb/types/global_secondary_index_auto_scaling_update.py +16 -0
- aws_sdk_dynamodb/types/global_secondary_index_auto_scaling_update_list.py +10 -0
- aws_sdk_dynamodb/types/global_secondary_index_description.py +47 -0
- aws_sdk_dynamodb/types/global_secondary_index_description_list.py +10 -0
- aws_sdk_dynamodb/types/global_secondary_index_info.py +27 -0
- aws_sdk_dynamodb/types/global_secondary_index_list.py +10 -0
- aws_sdk_dynamodb/types/global_secondary_index_update.py +24 -0
- aws_sdk_dynamodb/types/global_secondary_index_update_list.py +10 -0
- aws_sdk_dynamodb/types/global_secondary_index_warm_throughput_description.py +21 -0
- aws_sdk_dynamodb/types/global_secondary_indexes.py +10 -0
- aws_sdk_dynamodb/types/global_table.py +15 -0
- aws_sdk_dynamodb/types/global_table_arn_string.py +5 -0
- aws_sdk_dynamodb/types/global_table_description.py +30 -0
- aws_sdk_dynamodb/types/global_table_global_secondary_index_settings_update.py +22 -0
- aws_sdk_dynamodb/types/global_table_global_secondary_index_settings_update_list.py +10 -0
- aws_sdk_dynamodb/types/global_table_list.py +8 -0
- aws_sdk_dynamodb/types/global_table_settings_replication_mode.py +9 -0
- aws_sdk_dynamodb/types/global_table_status.py +10 -0
- aws_sdk_dynamodb/types/global_table_witness_description.py +15 -0
- aws_sdk_dynamodb/types/global_table_witness_description_list.py +10 -0
- aws_sdk_dynamodb/types/global_table_witness_group_update.py +19 -0
- aws_sdk_dynamodb/types/global_table_witness_group_update_list.py +10 -0
- aws_sdk_dynamodb/types/import_arn.py +5 -0
- aws_sdk_dynamodb/types/import_end_time.py +6 -0
- aws_sdk_dynamodb/types/import_next_token.py +5 -0
- aws_sdk_dynamodb/types/import_start_time.py +6 -0
- aws_sdk_dynamodb/types/import_status.py +11 -0
- aws_sdk_dynamodb/types/import_summary.py +37 -0
- aws_sdk_dynamodb/types/import_summary_list.py +10 -0
- aws_sdk_dynamodb/types/import_table_description.py +80 -0
- aws_sdk_dynamodb/types/import_table_input.py +33 -0
- aws_sdk_dynamodb/types/import_table_output.py +13 -0
- aws_sdk_dynamodb/types/imported_item_count.py +7 -0
- aws_sdk_dynamodb/types/incremental_export_specification.py +22 -0
- aws_sdk_dynamodb/types/index_name.py +5 -0
- aws_sdk_dynamodb/types/index_status.py +10 -0
- aws_sdk_dynamodb/types/input_compression_type.py +9 -0
- aws_sdk_dynamodb/types/input_format.py +9 -0
- aws_sdk_dynamodb/types/input_format_options.py +12 -0
- aws_sdk_dynamodb/types/integer.py +7 -0
- aws_sdk_dynamodb/types/integer_object.py +5 -0
- aws_sdk_dynamodb/types/item_collection_key_attribute_map.py +12 -0
- aws_sdk_dynamodb/types/item_collection_metrics.py +19 -0
- aws_sdk_dynamodb/types/item_collection_metrics_multiple.py +10 -0
- aws_sdk_dynamodb/types/item_collection_metrics_per_table.py +12 -0
- aws_sdk_dynamodb/types/item_collection_size_estimate_bound.py +5 -0
- aws_sdk_dynamodb/types/item_collection_size_estimate_range.py +10 -0
- aws_sdk_dynamodb/types/item_count.py +5 -0
- aws_sdk_dynamodb/types/item_list.py +8 -0
- aws_sdk_dynamodb/types/item_response.py +12 -0
- aws_sdk_dynamodb/types/item_response_list.py +8 -0
- aws_sdk_dynamodb/types/key.py +12 -0
- aws_sdk_dynamodb/types/key_conditions.py +12 -0
- aws_sdk_dynamodb/types/key_expression.py +5 -0
- aws_sdk_dynamodb/types/key_list.py +8 -0
- aws_sdk_dynamodb/types/key_schema.py +10 -0
- aws_sdk_dynamodb/types/key_schema_attribute_name.py +5 -0
- aws_sdk_dynamodb/types/key_schema_element.py +16 -0
- aws_sdk_dynamodb/types/key_type.py +8 -0
- aws_sdk_dynamodb/types/keys_and_attributes.py +32 -0
- aws_sdk_dynamodb/types/kinesis_data_stream_destination.py +25 -0
- aws_sdk_dynamodb/types/kinesis_data_stream_destinations.py +10 -0
- aws_sdk_dynamodb/types/kinesis_streaming_destination_input.py +20 -0
- aws_sdk_dynamodb/types/kinesis_streaming_destination_output.py +25 -0
- aws_sdk_dynamodb/types/kms_master_key_arn.py +5 -0
- aws_sdk_dynamodb/types/kms_master_key_id.py +5 -0
- aws_sdk_dynamodb/types/last_update_date_time.py +6 -0
- aws_sdk_dynamodb/types/list_attribute_value.py +10 -0
- aws_sdk_dynamodb/types/list_backups_input.py +35 -0
- aws_sdk_dynamodb/types/list_backups_output.py +19 -0
- aws_sdk_dynamodb/types/list_contributor_insights_input.py +18 -0
- aws_sdk_dynamodb/types/list_contributor_insights_limit.py +7 -0
- aws_sdk_dynamodb/types/list_contributor_insights_output.py +17 -0
- aws_sdk_dynamodb/types/list_exports_input.py +20 -0
- aws_sdk_dynamodb/types/list_exports_max_limit.py +5 -0
- aws_sdk_dynamodb/types/list_exports_output.py +17 -0
- aws_sdk_dynamodb/types/list_global_tables_input.py +22 -0
- aws_sdk_dynamodb/types/list_global_tables_output.py +19 -0
- aws_sdk_dynamodb/types/list_imports_input.py +20 -0
- aws_sdk_dynamodb/types/list_imports_max_limit.py +5 -0
- aws_sdk_dynamodb/types/list_imports_output.py +17 -0
- aws_sdk_dynamodb/types/list_tables_input.py +19 -0
- aws_sdk_dynamodb/types/list_tables_input_limit.py +5 -0
- aws_sdk_dynamodb/types/list_tables_output.py +17 -0
- aws_sdk_dynamodb/types/list_tags_of_resource_input.py +15 -0
- aws_sdk_dynamodb/types/list_tags_of_resource_output.py +15 -0
- aws_sdk_dynamodb/types/local_secondary_index.py +17 -0
- aws_sdk_dynamodb/types/local_secondary_index_description.py +26 -0
- aws_sdk_dynamodb/types/local_secondary_index_description_list.py +10 -0
- aws_sdk_dynamodb/types/local_secondary_index_info.py +18 -0
- aws_sdk_dynamodb/types/local_secondary_index_list.py +10 -0
- aws_sdk_dynamodb/types/local_secondary_indexes.py +10 -0
- aws_sdk_dynamodb/types/long.py +7 -0
- aws_sdk_dynamodb/types/long_object.py +5 -0
- aws_sdk_dynamodb/types/map_attribute_value.py +12 -0
- aws_sdk_dynamodb/types/multi_region_consistency.py +8 -0
- aws_sdk_dynamodb/types/next_token_string.py +5 -0
- aws_sdk_dynamodb/types/non_key_attribute_name.py +5 -0
- aws_sdk_dynamodb/types/non_key_attribute_name_list.py +10 -0
- aws_sdk_dynamodb/types/non_negative_long_object.py +5 -0
- aws_sdk_dynamodb/types/null_attribute_value.py +5 -0
- aws_sdk_dynamodb/types/number_attribute_value.py +5 -0
- aws_sdk_dynamodb/types/number_set_attribute_value.py +10 -0
- aws_sdk_dynamodb/types/on_demand_throughput.py +16 -0
- aws_sdk_dynamodb/types/on_demand_throughput_override.py +12 -0
- aws_sdk_dynamodb/types/parameterized_statement.py +22 -0
- aws_sdk_dynamodb/types/parameterized_statements.py +10 -0
- aws_sdk_dynamodb/types/parti_ql_batch_request.py +10 -0
- aws_sdk_dynamodb/types/parti_ql_batch_response.py +10 -0
- aws_sdk_dynamodb/types/parti_ql_next_token.py +5 -0
- aws_sdk_dynamodb/types/parti_ql_statement.py +5 -0
- aws_sdk_dynamodb/types/point_in_time_recovery_description.py +24 -0
- aws_sdk_dynamodb/types/point_in_time_recovery_specification.py +19 -0
- aws_sdk_dynamodb/types/point_in_time_recovery_status.py +8 -0
- aws_sdk_dynamodb/types/policy_revision_id.py +5 -0
- aws_sdk_dynamodb/types/positive_integer_object.py +5 -0
- aws_sdk_dynamodb/types/positive_long_object.py +5 -0
- aws_sdk_dynamodb/types/prepared_statement_parameters.py +10 -0
- aws_sdk_dynamodb/types/processed_item_count.py +7 -0
- aws_sdk_dynamodb/types/projection.py +19 -0
- aws_sdk_dynamodb/types/projection_expression.py +5 -0
- aws_sdk_dynamodb/types/projection_type.py +9 -0
- aws_sdk_dynamodb/types/provisioned_throughput.py +17 -0
- aws_sdk_dynamodb/types/provisioned_throughput_description.py +28 -0
- aws_sdk_dynamodb/types/provisioned_throughput_override.py +14 -0
- aws_sdk_dynamodb/types/put.py +35 -0
- aws_sdk_dynamodb/types/put_item_input.py +57 -0
- aws_sdk_dynamodb/types/put_item_input_attribute_map.py +12 -0
- aws_sdk_dynamodb/types/put_item_output.py +22 -0
- aws_sdk_dynamodb/types/put_request.py +11 -0
- aws_sdk_dynamodb/types/put_resource_policy_input.py +23 -0
- aws_sdk_dynamodb/types/put_resource_policy_output.py +14 -0
- aws_sdk_dynamodb/types/query_input.py +83 -0
- aws_sdk_dynamodb/types/query_output.py +25 -0
- aws_sdk_dynamodb/types/reason.py +5 -0
- aws_sdk_dynamodb/types/recovery_period_in_days.py +5 -0
- aws_sdk_dynamodb/types/region_name.py +5 -0
- aws_sdk_dynamodb/types/replica.py +12 -0
- aws_sdk_dynamodb/types/replica_auto_scaling_description.py +27 -0
- aws_sdk_dynamodb/types/replica_auto_scaling_description_list.py +10 -0
- aws_sdk_dynamodb/types/replica_auto_scaling_update.py +21 -0
- aws_sdk_dynamodb/types/replica_auto_scaling_update_list.py +10 -0
- aws_sdk_dynamodb/types/replica_description.py +65 -0
- aws_sdk_dynamodb/types/replica_description_list.py +10 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index.py +22 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_auto_scaling_description.py +22 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_auto_scaling_description_list.py +10 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_auto_scaling_update.py +16 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_auto_scaling_update_list.py +10 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_description.py +27 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_description_list.py +10 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_list.py +10 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_settings_description.py +33 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_settings_description_list.py +10 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_settings_update.py +22 -0
- aws_sdk_dynamodb/types/replica_global_secondary_index_settings_update_list.py +10 -0
- aws_sdk_dynamodb/types/replica_list.py +8 -0
- aws_sdk_dynamodb/types/replica_settings_description.py +47 -0
- aws_sdk_dynamodb/types/replica_settings_description_list.py +10 -0
- aws_sdk_dynamodb/types/replica_settings_update.py +30 -0
- aws_sdk_dynamodb/types/replica_settings_update_list.py +10 -0
- aws_sdk_dynamodb/types/replica_status.py +16 -0
- aws_sdk_dynamodb/types/replica_status_description.py +5 -0
- aws_sdk_dynamodb/types/replica_status_percent_progress.py +5 -0
- aws_sdk_dynamodb/types/replica_update.py +19 -0
- aws_sdk_dynamodb/types/replica_update_list.py +10 -0
- aws_sdk_dynamodb/types/replication_group_update.py +24 -0
- aws_sdk_dynamodb/types/replication_group_update_list.py +10 -0
- aws_sdk_dynamodb/types/resource.py +5 -0
- aws_sdk_dynamodb/types/resource_arn_string.py +5 -0
- aws_sdk_dynamodb/types/resource_policy.py +5 -0
- aws_sdk_dynamodb/types/restore_in_progress.py +5 -0
- aws_sdk_dynamodb/types/restore_summary.py +21 -0
- aws_sdk_dynamodb/types/restore_table_from_backup_input.py +44 -0
- aws_sdk_dynamodb/types/restore_table_from_backup_output.py +14 -0
- aws_sdk_dynamodb/types/restore_table_to_point_in_time_input.py +54 -0
- aws_sdk_dynamodb/types/restore_table_to_point_in_time_output.py +14 -0
- aws_sdk_dynamodb/types/return_consumed_capacity.py +10 -0
- aws_sdk_dynamodb/types/return_item_collection_metrics.py +8 -0
- aws_sdk_dynamodb/types/return_value.py +11 -0
- aws_sdk_dynamodb/types/return_values_on_condition_check_failure.py +8 -0
- aws_sdk_dynamodb/types/s3_bucket.py +5 -0
- aws_sdk_dynamodb/types/s3_bucket_owner.py +5 -0
- aws_sdk_dynamodb/types/s3_bucket_source.py +18 -0
- aws_sdk_dynamodb/types/s3_prefix.py +5 -0
- aws_sdk_dynamodb/types/s3_sse_algorithm.py +8 -0
- aws_sdk_dynamodb/types/s3_sse_kms_key_id.py +5 -0
- aws_sdk_dynamodb/types/scalar_attribute_type.py +9 -0
- aws_sdk_dynamodb/types/scan_input.py +78 -0
- aws_sdk_dynamodb/types/scan_output.py +25 -0
- aws_sdk_dynamodb/types/scan_segment.py +5 -0
- aws_sdk_dynamodb/types/scan_total_segments.py +5 -0
- aws_sdk_dynamodb/types/secondary_indexes_capacity_map.py +12 -0
- aws_sdk_dynamodb/types/select.py +10 -0
- aws_sdk_dynamodb/types/source_table_details.py +44 -0
- aws_sdk_dynamodb/types/source_table_feature_details.py +34 -0
- aws_sdk_dynamodb/types/sse_description.py +23 -0
- aws_sdk_dynamodb/types/sse_enabled.py +5 -0
- aws_sdk_dynamodb/types/sse_specification.py +20 -0
- aws_sdk_dynamodb/types/sse_status.py +11 -0
- aws_sdk_dynamodb/types/sse_type.py +8 -0
- aws_sdk_dynamodb/types/stream_arn.py +5 -0
- aws_sdk_dynamodb/types/stream_enabled.py +5 -0
- aws_sdk_dynamodb/types/stream_specification.py +17 -0
- aws_sdk_dynamodb/types/stream_view_type.py +10 -0
- aws_sdk_dynamodb/types/string.py +5 -0
- aws_sdk_dynamodb/types/string_attribute_value.py +5 -0
- aws_sdk_dynamodb/types/string_set_attribute_value.py +10 -0
- aws_sdk_dynamodb/types/table_arn.py +5 -0
- aws_sdk_dynamodb/types/table_auto_scaling_description.py +20 -0
- aws_sdk_dynamodb/types/table_class.py +8 -0
- aws_sdk_dynamodb/types/table_class_summary.py +15 -0
- aws_sdk_dynamodb/types/table_creation_date_time.py +6 -0
- aws_sdk_dynamodb/types/table_creation_parameters.py +40 -0
- aws_sdk_dynamodb/types/table_description.py +124 -0
- aws_sdk_dynamodb/types/table_id.py +5 -0
- aws_sdk_dynamodb/types/table_name.py +5 -0
- aws_sdk_dynamodb/types/table_name_list.py +8 -0
- aws_sdk_dynamodb/types/table_status.py +14 -0
- aws_sdk_dynamodb/types/table_warm_throughput_description.py +21 -0
- aws_sdk_dynamodb/types/tag.py +14 -0
- aws_sdk_dynamodb/types/tag_key_list.py +8 -0
- aws_sdk_dynamodb/types/tag_key_string.py +5 -0
- aws_sdk_dynamodb/types/tag_list.py +8 -0
- aws_sdk_dynamodb/types/tag_resource_input.py +14 -0
- aws_sdk_dynamodb/types/tag_value_string.py +5 -0
- aws_sdk_dynamodb/types/throttling_reason.py +15 -0
- aws_sdk_dynamodb/types/throttling_reason_list.py +10 -0
- aws_sdk_dynamodb/types/time_range_lower_bound.py +6 -0
- aws_sdk_dynamodb/types/time_range_upper_bound.py +6 -0
- aws_sdk_dynamodb/types/time_to_live_attribute_name.py +5 -0
- aws_sdk_dynamodb/types/time_to_live_description.py +19 -0
- aws_sdk_dynamodb/types/time_to_live_enabled.py +5 -0
- aws_sdk_dynamodb/types/time_to_live_specification.py +16 -0
- aws_sdk_dynamodb/types/time_to_live_status.py +10 -0
- aws_sdk_dynamodb/types/transact_get_item.py +11 -0
- aws_sdk_dynamodb/types/transact_get_item_list.py +10 -0
- aws_sdk_dynamodb/types/transact_get_items_input.py +17 -0
- aws_sdk_dynamodb/types/transact_get_items_output.py +17 -0
- aws_sdk_dynamodb/types/transact_write_item.py +23 -0
- aws_sdk_dynamodb/types/transact_write_item_list.py +10 -0
- aws_sdk_dynamodb/types/transact_write_items_input.py +28 -0
- aws_sdk_dynamodb/types/transact_write_items_output.py +19 -0
- aws_sdk_dynamodb/types/untag_resource_input.py +14 -0
- aws_sdk_dynamodb/types/update.py +38 -0
- aws_sdk_dynamodb/types/update_continuous_backups_input.py +14 -0
- aws_sdk_dynamodb/types/update_continuous_backups_output.py +14 -0
- aws_sdk_dynamodb/types/update_contributor_insights_input.py +25 -0
- aws_sdk_dynamodb/types/update_contributor_insights_output.py +25 -0
- aws_sdk_dynamodb/types/update_expression.py +5 -0
- aws_sdk_dynamodb/types/update_global_secondary_index_action.py +27 -0
- aws_sdk_dynamodb/types/update_global_table_input.py +14 -0
- aws_sdk_dynamodb/types/update_global_table_output.py +14 -0
- aws_sdk_dynamodb/types/update_global_table_settings_input.py +37 -0
- aws_sdk_dynamodb/types/update_global_table_settings_output.py +17 -0
- aws_sdk_dynamodb/types/update_item_input.py +67 -0
- aws_sdk_dynamodb/types/update_item_output.py +22 -0
- aws_sdk_dynamodb/types/update_kinesis_streaming_configuration.py +14 -0
- aws_sdk_dynamodb/types/update_kinesis_streaming_destination_input.py +20 -0
- aws_sdk_dynamodb/types/update_kinesis_streaming_destination_output.py +25 -0
- aws_sdk_dynamodb/types/update_replication_group_member_action.py +35 -0
- aws_sdk_dynamodb/types/update_table_input.py +78 -0
- aws_sdk_dynamodb/types/update_table_output.py +14 -0
- aws_sdk_dynamodb/types/update_table_replica_auto_scaling_input.py +26 -0
- aws_sdk_dynamodb/types/update_table_replica_auto_scaling_output.py +14 -0
- aws_sdk_dynamodb/types/update_time_to_live_input.py +16 -0
- aws_sdk_dynamodb/types/update_time_to_live_output.py +14 -0
- aws_sdk_dynamodb/types/warm_throughput.py +14 -0
- aws_sdk_dynamodb/types/witness_status.py +9 -0
- aws_sdk_dynamodb/types/write_request.py +15 -0
- aws_sdk_dynamodb/types/write_requests.py +8 -0
- aws_sdk_dynamodb-0.1.0.dist-info/METADATA +111 -0
- aws_sdk_dynamodb-0.1.0.dist-info/RECORD +573 -0
- aws_sdk_dynamodb-0.1.0.dist-info/WHEEL +5 -0
- aws_sdk_dynamodb-0.1.0.dist-info/licenses/LICENSE +21 -0
- aws_sdk_dynamodb-0.1.0.dist-info/top_level.txt +1 -0
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
from ._auth._identity import Identity as Identity, Credentials as Credentials
|
|
3
|
+
from ._auth._providers import (
|
|
4
|
+
IdentityNotFound as IdentityNotFound,
|
|
5
|
+
IdentityProvider as IdentityProvider,
|
|
6
|
+
ChainedProvider as ChainedProvider,
|
|
7
|
+
CachedProvider as CachedProvider,
|
|
8
|
+
CredentialsProvider as CredentialsProvider,
|
|
9
|
+
StaticAwsCredentialsProvider as StaticAwsCredentialsProvider,
|
|
10
|
+
EnvCredentialsProvider as EnvCredentialsProvider,
|
|
11
|
+
ProfileCredentialsProvider as ProfileCredentialsProvider,
|
|
12
|
+
)
|
|
13
|
+
from ._auth._signers import Signer as Signer, SigV4Signer as SigV4Signer
|
|
14
|
+
from ._services.dynamo_db_20120810 import DynamoDBClient as DynamoDBClient
|
|
15
|
+
from ._services.async_dynamo_db_20120810 import (
|
|
16
|
+
AsyncDynamoDBClient as AsyncDynamoDBClient,
|
|
17
|
+
)
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
import asyncio
|
|
4
|
+
from typing import TYPE_CHECKING
|
|
5
|
+
|
|
6
|
+
if TYPE_CHECKING:
|
|
7
|
+
import trio
|
|
8
|
+
else:
|
|
9
|
+
try:
|
|
10
|
+
import trio
|
|
11
|
+
except ImportError:
|
|
12
|
+
trio = None
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
def in_trio_run() -> bool:
|
|
16
|
+
if trio is None:
|
|
17
|
+
return False
|
|
18
|
+
return trio.lowlevel.in_trio_run()
|
|
19
|
+
|
|
20
|
+
|
|
21
|
+
async def anysleep(delay: float) -> None:
|
|
22
|
+
if in_trio_run():
|
|
23
|
+
await trio.sleep(delay)
|
|
24
|
+
else:
|
|
25
|
+
await asyncio.sleep(delay)
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
from datetime import datetime
|
|
4
|
+
from typing import TypedDict
|
|
5
|
+
|
|
6
|
+
from typing_extensions import NotRequired
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
class Identity(TypedDict):
|
|
10
|
+
expiration: NotRequired[datetime | None]
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class Credentials(Identity):
|
|
14
|
+
access_key: str
|
|
15
|
+
secret_key: str
|
|
16
|
+
session_token: NotRequired[str | None]
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
import configparser
|
|
4
|
+
import os
|
|
5
|
+
from abc import abstractmethod
|
|
6
|
+
from datetime import datetime, timezone
|
|
7
|
+
from pathlib import Path
|
|
8
|
+
from typing import Generic, TypeVar
|
|
9
|
+
|
|
10
|
+
from aws_sdk_dynamodb._auth._identity import (
|
|
11
|
+
Credentials,
|
|
12
|
+
Identity,
|
|
13
|
+
)
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
class IdentityNotFound(Exception):
|
|
17
|
+
"""Raised when a provider cannot resolve an identity. Chain continues."""
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
IdentityT = TypeVar("IdentityT", bound="Identity")
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
class IdentityProvider(Generic[IdentityT]):
|
|
24
|
+
@abstractmethod
|
|
25
|
+
def resolve_identity(self) -> IdentityT:
|
|
26
|
+
raise NotImplementedError
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
class ChainedProvider(IdentityProvider[IdentityT]):
|
|
30
|
+
"""Try each provider in order; first non-`IdentityNotFound` wins."""
|
|
31
|
+
|
|
32
|
+
def __init__(self, *providers: IdentityProvider[IdentityT]) -> None:
|
|
33
|
+
if not providers:
|
|
34
|
+
raise ValueError("ChainedProvider requires at least one provider")
|
|
35
|
+
self._providers = providers
|
|
36
|
+
|
|
37
|
+
def resolve_identity(self) -> IdentityT:
|
|
38
|
+
errors: list[str] = []
|
|
39
|
+
for p in self._providers:
|
|
40
|
+
try:
|
|
41
|
+
return p.resolve_identity()
|
|
42
|
+
except IdentityNotFound as e:
|
|
43
|
+
errors.append(f"{type(p).__name__}: {e}")
|
|
44
|
+
raise IdentityNotFound("no provider succeeded: " + "; ".join(errors))
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
class CachedProvider(IdentityProvider[IdentityT]):
|
|
48
|
+
"""Cache an identity until its `expiration` (minus skew) elapses."""
|
|
49
|
+
|
|
50
|
+
_SKEW_SECONDS = 60
|
|
51
|
+
|
|
52
|
+
def __init__(self, inner: IdentityProvider[IdentityT]) -> None:
|
|
53
|
+
self._inner = inner
|
|
54
|
+
self._cached: IdentityT | None = None
|
|
55
|
+
|
|
56
|
+
def resolve_identity(self) -> IdentityT:
|
|
57
|
+
if self._cached is not None and not self._expired(self._cached):
|
|
58
|
+
return self._cached
|
|
59
|
+
self._cached = self._inner.resolve_identity()
|
|
60
|
+
return self._cached
|
|
61
|
+
|
|
62
|
+
@classmethod
|
|
63
|
+
def _expired(cls, ident: Identity) -> bool:
|
|
64
|
+
exp = ident.get("expiration")
|
|
65
|
+
if exp is None:
|
|
66
|
+
return False
|
|
67
|
+
return (exp - datetime.now(timezone.utc)).total_seconds() <= cls._SKEW_SECONDS
|
|
68
|
+
|
|
69
|
+
|
|
70
|
+
class CredentialsProvider(IdentityProvider[Credentials]):
|
|
71
|
+
"""Base class for providers that resolve AWS `Credentials`."""
|
|
72
|
+
|
|
73
|
+
@abstractmethod
|
|
74
|
+
def resolve_identity(self) -> Credentials:
|
|
75
|
+
raise NotImplementedError
|
|
76
|
+
|
|
77
|
+
|
|
78
|
+
class StaticAwsCredentialsProvider(CredentialsProvider):
|
|
79
|
+
def __init__(self, credentials: Credentials) -> None:
|
|
80
|
+
self._credentials = credentials
|
|
81
|
+
|
|
82
|
+
def resolve_identity(self) -> Credentials:
|
|
83
|
+
return self._credentials
|
|
84
|
+
|
|
85
|
+
|
|
86
|
+
class EnvCredentialsProvider(CredentialsProvider):
|
|
87
|
+
"""Read AWS_ACCESS_KEY_ID / AWS_SECRET_ACCESS_KEY / AWS_SESSION_TOKEN."""
|
|
88
|
+
|
|
89
|
+
def resolve_identity(self) -> Credentials:
|
|
90
|
+
ak = os.environ.get("AWS_ACCESS_KEY_ID")
|
|
91
|
+
sk = os.environ.get("AWS_SECRET_ACCESS_KEY")
|
|
92
|
+
if not ak or not sk:
|
|
93
|
+
raise IdentityNotFound("AWS_ACCESS_KEY_ID / AWS_SECRET_ACCESS_KEY unset")
|
|
94
|
+
out: Credentials = {"access_key": ak, "secret_key": sk}
|
|
95
|
+
token = os.environ.get("AWS_SESSION_TOKEN")
|
|
96
|
+
if token:
|
|
97
|
+
out["session_token"] = token
|
|
98
|
+
return out
|
|
99
|
+
|
|
100
|
+
|
|
101
|
+
class ProfileCredentialsProvider(CredentialsProvider):
|
|
102
|
+
"""Read ~/.aws/credentials and ~/.aws/config for the active profile."""
|
|
103
|
+
|
|
104
|
+
def __init__(
|
|
105
|
+
self,
|
|
106
|
+
profile: str | None = None,
|
|
107
|
+
credentials_file: Path | None = None,
|
|
108
|
+
config_file: Path | None = None,
|
|
109
|
+
) -> None:
|
|
110
|
+
self._profile = profile or os.environ.get("AWS_PROFILE", "default")
|
|
111
|
+
self._cred_file = credentials_file or Path(
|
|
112
|
+
os.environ.get("AWS_SHARED_CREDENTIALS_FILE")
|
|
113
|
+
or Path.home() / ".aws" / "credentials"
|
|
114
|
+
)
|
|
115
|
+
self._cfg_file = config_file or Path(
|
|
116
|
+
os.environ.get("AWS_CONFIG_FILE") or Path.home() / ".aws" / "config"
|
|
117
|
+
)
|
|
118
|
+
|
|
119
|
+
def resolve_identity(self) -> Credentials:
|
|
120
|
+
section = self._load_profile()
|
|
121
|
+
ak = section.get("aws_access_key_id")
|
|
122
|
+
sk = section.get("aws_secret_access_key")
|
|
123
|
+
if not ak or not sk:
|
|
124
|
+
raise IdentityNotFound(
|
|
125
|
+
f"profile {self._profile!r}: missing aws_access_key_id/aws_secret_access_key"
|
|
126
|
+
)
|
|
127
|
+
out: Credentials = {"access_key": ak, "secret_key": sk}
|
|
128
|
+
token = section.get("aws_session_token")
|
|
129
|
+
if token:
|
|
130
|
+
out["session_token"] = token
|
|
131
|
+
return out
|
|
132
|
+
|
|
133
|
+
def _load_profile(self) -> dict[str, str]:
|
|
134
|
+
merged: dict[str, str] = {}
|
|
135
|
+
if self._cfg_file.is_file():
|
|
136
|
+
cfg = configparser.ConfigParser()
|
|
137
|
+
cfg.read(self._cfg_file)
|
|
138
|
+
# config file profiles look like `[profile foo]`, except default
|
|
139
|
+
key = (
|
|
140
|
+
"default" if self._profile == "default" else f"profile {self._profile}"
|
|
141
|
+
)
|
|
142
|
+
if cfg.has_section(key):
|
|
143
|
+
merged.update(dict(cfg.items(key)))
|
|
144
|
+
if self._cred_file.is_file():
|
|
145
|
+
cfg = configparser.ConfigParser()
|
|
146
|
+
cfg.read(self._cred_file)
|
|
147
|
+
if cfg.has_section(self._profile):
|
|
148
|
+
merged.update(dict(cfg.items(self._profile)))
|
|
149
|
+
if not merged:
|
|
150
|
+
raise IdentityNotFound(
|
|
151
|
+
f"profile {self._profile!r} not found in {self._cred_file} or {self._cfg_file}"
|
|
152
|
+
)
|
|
153
|
+
return merged
|
|
154
|
+
|
|
155
|
+
|
|
156
|
+
def default_aws_credentials_chain() -> IdentityProvider[Credentials]:
|
|
157
|
+
return CachedProvider(
|
|
158
|
+
ChainedProvider(
|
|
159
|
+
EnvCredentialsProvider(),
|
|
160
|
+
ProfileCredentialsProvider(),
|
|
161
|
+
)
|
|
162
|
+
)
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
from __future__ import annotations
|
|
2
|
+
|
|
3
|
+
from abc import ABC, abstractmethod
|
|
4
|
+
from typing import Generic, TypeVar
|
|
5
|
+
from typing import Any
|
|
6
|
+
from zapros import Request
|
|
7
|
+
from aws_sdk_dynamodb._auth._sigv4 import SigV4AuthContext, sign_sigv4
|
|
8
|
+
from aws_sdk_dynamodb._auth._identity import Credentials, Identity
|
|
9
|
+
from aws_sdk_dynamodb._auth._providers import IdentityProvider
|
|
10
|
+
|
|
11
|
+
IdentityT = TypeVar("IdentityT", bound="Identity")
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class Signer(ABC, Generic[IdentityT]):
|
|
15
|
+
"""Per-request request signer. Holds an IdentityProvider plus static config."""
|
|
16
|
+
|
|
17
|
+
def __init__(self, provider: IdentityProvider[IdentityT]) -> None:
|
|
18
|
+
self.provider = provider
|
|
19
|
+
|
|
20
|
+
@abstractmethod
|
|
21
|
+
async def asign(self, req: Request) -> Request: ...
|
|
22
|
+
@abstractmethod
|
|
23
|
+
def sign(self, req: Request) -> Request: ...
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
class SigV4Signer(Signer[Credentials]):
|
|
27
|
+
"""aws.auth#sigv4 — AWS Signature Version 4.
|
|
28
|
+
|
|
29
|
+
The full auth scheme (``name`` variant, ``signingName``, ``signingRegion``,
|
|
30
|
+
encoding/normalization flags) is provided by the caller — either from the
|
|
31
|
+
endpoint rule-set's ``authSchemes`` property or built by the generated
|
|
32
|
+
``get_signer`` from operation defaults.
|
|
33
|
+
"""
|
|
34
|
+
|
|
35
|
+
def __init__(
|
|
36
|
+
self, provider: IdentityProvider[Credentials], *, auth_scheme: dict[str, Any]
|
|
37
|
+
) -> None:
|
|
38
|
+
super().__init__(provider)
|
|
39
|
+
self._auth_scheme = auth_scheme
|
|
40
|
+
|
|
41
|
+
async def asign(self, req: Request) -> Request:
|
|
42
|
+
creds = self.provider.resolve_identity()
|
|
43
|
+
ctx: SigV4AuthContext = {
|
|
44
|
+
"type": "sig_v4",
|
|
45
|
+
"access_key_id": creds["access_key"],
|
|
46
|
+
"secret_access_key": creds["secret_key"],
|
|
47
|
+
"session_token": creds.get("session_token"),
|
|
48
|
+
"signing_region": self._auth_scheme["signingRegion"],
|
|
49
|
+
"signing_name": self._auth_scheme["signingName"],
|
|
50
|
+
}
|
|
51
|
+
if req.body is None:
|
|
52
|
+
body: bytes | None = b""
|
|
53
|
+
elif isinstance(req.body, bytes):
|
|
54
|
+
body = req.body
|
|
55
|
+
else:
|
|
56
|
+
body = None
|
|
57
|
+
# Strip Accept-Encoding so transports/intermediaries can't transcode
|
|
58
|
+
# the response and so the value never enters the canonical request.
|
|
59
|
+
if "accept-encoding" in req.headers:
|
|
60
|
+
del req.headers["Accept-Encoding"]
|
|
61
|
+
return sign_sigv4(req, ctx, body)
|
|
62
|
+
|
|
63
|
+
def sign(self, req: Request) -> Request:
|
|
64
|
+
creds = self.provider.resolve_identity()
|
|
65
|
+
ctx: SigV4AuthContext = {
|
|
66
|
+
"type": "sig_v4",
|
|
67
|
+
"access_key_id": creds["access_key"],
|
|
68
|
+
"secret_access_key": creds["secret_key"],
|
|
69
|
+
"session_token": creds.get("session_token"),
|
|
70
|
+
"signing_region": self._auth_scheme["signingRegion"],
|
|
71
|
+
"signing_name": self._auth_scheme["signingName"],
|
|
72
|
+
}
|
|
73
|
+
if req.body is None:
|
|
74
|
+
body: bytes | None = b""
|
|
75
|
+
elif isinstance(req.body, bytes):
|
|
76
|
+
body = req.body
|
|
77
|
+
else:
|
|
78
|
+
body = None
|
|
79
|
+
# Strip Accept-Encoding so transports/intermediaries can't transcode
|
|
80
|
+
# the response and so the value never enters the canonical request.
|
|
81
|
+
if "accept-encoding" in req.headers:
|
|
82
|
+
del req.headers["Accept-Encoding"]
|
|
83
|
+
return sign_sigv4(req, ctx, body)
|
|
@@ -0,0 +1,364 @@
|
|
|
1
|
+
"""AWS Signature Version 4 — single-chunk signing.
|
|
2
|
+
|
|
3
|
+
Reference:
|
|
4
|
+
https://docs.aws.amazon.com/IAM/latest/UserGuide/reference_sigv-create-signed-request.html
|
|
5
|
+
|
|
6
|
+
Verified byte-for-byte against ``botocore.auth.S3SigV4Auth`` / ``SigV4Auth``
|
|
7
|
+
across S3 GET/PUT/POST, query-string, session-token, and non-S3 (IAM)
|
|
8
|
+
canonicalization paths.
|
|
9
|
+
"""
|
|
10
|
+
|
|
11
|
+
from __future__ import annotations
|
|
12
|
+
|
|
13
|
+
import datetime as _dt
|
|
14
|
+
import hashlib
|
|
15
|
+
import hmac
|
|
16
|
+
import re
|
|
17
|
+
from typing import Literal, TypedDict
|
|
18
|
+
from urllib.parse import quote, unquote
|
|
19
|
+
|
|
20
|
+
from pywhatwgurl import URLSearchParams
|
|
21
|
+
from zapros import Headers, Request
|
|
22
|
+
import zapros
|
|
23
|
+
from zapros._utils import get_host_header_value
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
class SigV4AuthContext(TypedDict):
|
|
27
|
+
type: Literal["sig_v4"]
|
|
28
|
+
access_key_id: str
|
|
29
|
+
secret_access_key: str
|
|
30
|
+
session_token: str | None
|
|
31
|
+
signing_region: str
|
|
32
|
+
signing_name: str
|
|
33
|
+
|
|
34
|
+
|
|
35
|
+
_SIGV4_ALGORITHM = "AWS4-HMAC-SHA256"
|
|
36
|
+
_EMPTY_PAYLOAD_SHA256 = hashlib.sha256(b"").hexdigest()
|
|
37
|
+
|
|
38
|
+
# Headers excluded from the signed-headers set. Mirrors botocore's denylist:
|
|
39
|
+
# these are hop-by-hop / mutable in transit, so signing them would break
|
|
40
|
+
# proxies or duplicate values already added by the transport layer.
|
|
41
|
+
_UNSIGNED_HEADERS = frozenset(
|
|
42
|
+
{
|
|
43
|
+
"authorization",
|
|
44
|
+
"cache-control",
|
|
45
|
+
"connection",
|
|
46
|
+
"expect",
|
|
47
|
+
"from",
|
|
48
|
+
"keep-alive",
|
|
49
|
+
"max-forwards",
|
|
50
|
+
"pragma",
|
|
51
|
+
"referer",
|
|
52
|
+
"te",
|
|
53
|
+
"trailer",
|
|
54
|
+
"transfer-encoding",
|
|
55
|
+
"upgrade",
|
|
56
|
+
"user-agent",
|
|
57
|
+
"x-amzn-trace-id",
|
|
58
|
+
"content-length",
|
|
59
|
+
"accept",
|
|
60
|
+
"accept-encoding",
|
|
61
|
+
}
|
|
62
|
+
)
|
|
63
|
+
|
|
64
|
+
_MULTI_SPACE = re.compile(r" +")
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
def _uri_encode(value: str) -> str:
|
|
68
|
+
"""RFC 3986 percent-encoding using only the unreserved set as safe."""
|
|
69
|
+
return quote(value, safe="-_.~")
|
|
70
|
+
|
|
71
|
+
|
|
72
|
+
def _canonical_path(path: str, *, service: str) -> str:
|
|
73
|
+
"""Build CanonicalURI.
|
|
74
|
+
|
|
75
|
+
Per the SigV4 spec, every segment is URI-encoded; for services other
|
|
76
|
+
than S3 each segment is URI-encoded **twice**. S3 keeps the path
|
|
77
|
+
exactly as provided (no normalization, no double-encoding).
|
|
78
|
+
"""
|
|
79
|
+
if not path:
|
|
80
|
+
return "/"
|
|
81
|
+
if service == "s3":
|
|
82
|
+
return path if path.startswith("/") else "/" + path
|
|
83
|
+
decoded = unquote(path)
|
|
84
|
+
first = quote(decoded, safe="/~")
|
|
85
|
+
return quote(first, safe="/~")
|
|
86
|
+
|
|
87
|
+
|
|
88
|
+
def _canonical_query(query: str) -> str:
|
|
89
|
+
"""Build CanonicalQueryString from a raw query string (with or without ``?``)."""
|
|
90
|
+
if not query:
|
|
91
|
+
return ""
|
|
92
|
+
if query.startswith("?"):
|
|
93
|
+
query = query[1:]
|
|
94
|
+
if not query:
|
|
95
|
+
return ""
|
|
96
|
+
sp = URLSearchParams(query)
|
|
97
|
+
encoded = sorted((_uri_encode(k), _uri_encode(v)) for k, v in sp.entries())
|
|
98
|
+
return "&".join(f"{k}={v}" for k, v in encoded)
|
|
99
|
+
|
|
100
|
+
|
|
101
|
+
def _trim_header_value(value: str) -> str:
|
|
102
|
+
"""Trim leading/trailing whitespace and collapse internal whitespace runs.
|
|
103
|
+
|
|
104
|
+
Spec note: the canonical form treats sequential whitespace inside an
|
|
105
|
+
unquoted value as a single space. We do not parse quoted-string syntax;
|
|
106
|
+
the conservative collapse is correct for every header AWS actually signs.
|
|
107
|
+
"""
|
|
108
|
+
return _MULTI_SPACE.sub(" ", value.strip())
|
|
109
|
+
|
|
110
|
+
|
|
111
|
+
def _canonical_headers(headers: Headers) -> tuple[str, str]:
|
|
112
|
+
"""Return ``(canonical_headers, signed_headers)``."""
|
|
113
|
+
grouped: dict[str, list[str]] = {}
|
|
114
|
+
for name in headers:
|
|
115
|
+
lname = name.lower()
|
|
116
|
+
if lname in _UNSIGNED_HEADERS:
|
|
117
|
+
continue
|
|
118
|
+
grouped[lname] = [_trim_header_value(v) for v in headers.getall(name)]
|
|
119
|
+
|
|
120
|
+
signed = sorted(grouped)
|
|
121
|
+
canonical = "".join(f"{name}:{','.join(grouped[name])}\n" for name in signed)
|
|
122
|
+
return canonical, ";".join(signed)
|
|
123
|
+
|
|
124
|
+
|
|
125
|
+
def _build_canonical_request(
|
|
126
|
+
*,
|
|
127
|
+
method: str,
|
|
128
|
+
path: str,
|
|
129
|
+
query: str,
|
|
130
|
+
headers: Headers,
|
|
131
|
+
payload_hash: str,
|
|
132
|
+
service: str,
|
|
133
|
+
) -> tuple[str, str]:
|
|
134
|
+
canonical_uri = _canonical_path(path, service=service)
|
|
135
|
+
canonical_query = _canonical_query(query)
|
|
136
|
+
canonical_headers, signed_headers = _canonical_headers(headers)
|
|
137
|
+
canonical_request = (
|
|
138
|
+
f"{method}\n"
|
|
139
|
+
f"{canonical_uri}\n"
|
|
140
|
+
f"{canonical_query}\n"
|
|
141
|
+
f"{canonical_headers}\n"
|
|
142
|
+
f"{signed_headers}\n"
|
|
143
|
+
f"{payload_hash}"
|
|
144
|
+
)
|
|
145
|
+
return canonical_request, signed_headers
|
|
146
|
+
|
|
147
|
+
|
|
148
|
+
def _derive_signing_key(secret: str, date: str, region: str, service: str) -> bytes:
|
|
149
|
+
k_date = hmac.new(
|
|
150
|
+
b"AWS4" + secret.encode("utf-8"), date.encode("ascii"), hashlib.sha256
|
|
151
|
+
).digest()
|
|
152
|
+
k_region = hmac.new(k_date, region.encode("utf-8"), hashlib.sha256).digest()
|
|
153
|
+
k_service = hmac.new(k_region, service.encode("utf-8"), hashlib.sha256).digest()
|
|
154
|
+
return hmac.new(k_service, b"aws4_request", hashlib.sha256).digest()
|
|
155
|
+
|
|
156
|
+
|
|
157
|
+
def _amz_now() -> _dt.datetime:
|
|
158
|
+
return _dt.datetime.now(_dt.timezone.utc)
|
|
159
|
+
|
|
160
|
+
|
|
161
|
+
def _canonical_query_from_pairs(pairs: list[tuple[str, str]]) -> str:
|
|
162
|
+
"""CanonicalQueryString from raw (unencoded) key/value pairs."""
|
|
163
|
+
encoded = sorted((_uri_encode(k), _uri_encode(v)) for k, v in pairs)
|
|
164
|
+
return "&".join(f"{k}={v}" for k, v in encoded)
|
|
165
|
+
|
|
166
|
+
|
|
167
|
+
def sign_sigv4(
|
|
168
|
+
request: Request,
|
|
169
|
+
ctx: SigV4AuthContext,
|
|
170
|
+
body: bytes | None,
|
|
171
|
+
) -> Request:
|
|
172
|
+
"""Return a new ``Request`` carrying SigV4 single-chunk auth headers.
|
|
173
|
+
|
|
174
|
+
Pass ``body=None`` to sign with ``UNSIGNED-PAYLOAD`` (streaming requests).
|
|
175
|
+
The original ``request.body`` is forwarded unchanged in that case.
|
|
176
|
+
"""
|
|
177
|
+
service = ctx["signing_name"]
|
|
178
|
+
region = ctx["signing_region"]
|
|
179
|
+
|
|
180
|
+
headers = request.headers.copy()
|
|
181
|
+
|
|
182
|
+
# X-Amz-Date — honor caller-supplied value (allows deterministic tests).
|
|
183
|
+
existing_date = headers.get("X-Amz-Date")
|
|
184
|
+
if existing_date:
|
|
185
|
+
amz_date = existing_date
|
|
186
|
+
date_stamp = amz_date[:8]
|
|
187
|
+
else:
|
|
188
|
+
now = _amz_now()
|
|
189
|
+
amz_date = now.strftime("%Y%m%dT%H%M%SZ")
|
|
190
|
+
date_stamp = now.strftime("%Y%m%d")
|
|
191
|
+
headers["X-Amz-Date"] = amz_date
|
|
192
|
+
|
|
193
|
+
# Payload hash. For S3, x-amz-content-sha256 is mandatory and must be set
|
|
194
|
+
# BEFORE computing the canonical request (it gets signed).
|
|
195
|
+
payload_hash = headers.get("X-Amz-Content-SHA256")
|
|
196
|
+
if payload_hash is None:
|
|
197
|
+
if body is None:
|
|
198
|
+
payload_hash = "UNSIGNED-PAYLOAD"
|
|
199
|
+
else:
|
|
200
|
+
payload_hash = (
|
|
201
|
+
hashlib.sha256(body).hexdigest() if body else _EMPTY_PAYLOAD_SHA256
|
|
202
|
+
)
|
|
203
|
+
if service == "s3":
|
|
204
|
+
headers["X-Amz-Content-SHA256"] = payload_hash
|
|
205
|
+
|
|
206
|
+
# Session token (STS / assumed-role credentials).
|
|
207
|
+
session_token = ctx.get("session_token")
|
|
208
|
+
if session_token:
|
|
209
|
+
headers["X-Amz-Security-Token"] = session_token
|
|
210
|
+
|
|
211
|
+
# Host header is added by Request.__init__ from the URL; defensive fallback.
|
|
212
|
+
if "host" not in headers and request.url.hostname:
|
|
213
|
+
headers["Host"] = get_host_header_value(request.url)
|
|
214
|
+
|
|
215
|
+
canonical_request, signed_headers = _build_canonical_request(
|
|
216
|
+
method=request.method.upper(),
|
|
217
|
+
path=request.url.pathname,
|
|
218
|
+
query=request.url.search,
|
|
219
|
+
headers=headers,
|
|
220
|
+
payload_hash=payload_hash,
|
|
221
|
+
service=service,
|
|
222
|
+
)
|
|
223
|
+
|
|
224
|
+
credential_scope = f"{date_stamp}/{region}/{service}/aws4_request"
|
|
225
|
+
string_to_sign = (
|
|
226
|
+
f"{_SIGV4_ALGORITHM}\n"
|
|
227
|
+
f"{amz_date}\n"
|
|
228
|
+
f"{credential_scope}\n"
|
|
229
|
+
f"{hashlib.sha256(canonical_request.encode('utf-8')).hexdigest()}"
|
|
230
|
+
)
|
|
231
|
+
|
|
232
|
+
signing_key = _derive_signing_key(
|
|
233
|
+
ctx["secret_access_key"], date_stamp, region, service
|
|
234
|
+
)
|
|
235
|
+
signature = hmac.new(
|
|
236
|
+
signing_key, string_to_sign.encode("utf-8"), hashlib.sha256
|
|
237
|
+
).hexdigest()
|
|
238
|
+
|
|
239
|
+
headers["Authorization"] = (
|
|
240
|
+
f"{_SIGV4_ALGORITHM} "
|
|
241
|
+
f"Credential={ctx['access_key_id']}/{credential_scope},"
|
|
242
|
+
f"SignedHeaders={signed_headers},"
|
|
243
|
+
f"Signature={signature}"
|
|
244
|
+
)
|
|
245
|
+
|
|
246
|
+
effective_body = body if body is not None else request.body
|
|
247
|
+
if effective_body is not None:
|
|
248
|
+
return Request(
|
|
249
|
+
request.url,
|
|
250
|
+
request.method,
|
|
251
|
+
headers,
|
|
252
|
+
body=effective_body,
|
|
253
|
+
context=request.context,
|
|
254
|
+
)
|
|
255
|
+
return Request(request.url, request.method, headers, context=request.context)
|
|
256
|
+
|
|
257
|
+
|
|
258
|
+
def presign_sigv4(
|
|
259
|
+
request: Request,
|
|
260
|
+
ctx: SigV4AuthContext,
|
|
261
|
+
*,
|
|
262
|
+
expires_in: int = 3600,
|
|
263
|
+
now: _dt.datetime | None = None,
|
|
264
|
+
) -> Request:
|
|
265
|
+
"""Return a new ``Request`` whose URL carries SigV4 query-string auth.
|
|
266
|
+
|
|
267
|
+
The signature travels in the URL (``X-Amz-*`` query params), so the result
|
|
268
|
+
is usable standalone (browser, curl). Payload is signed as
|
|
269
|
+
``UNSIGNED-PAYLOAD``, so the body is not bound by the signature.
|
|
270
|
+
|
|
271
|
+
``expires_in`` is the validity window in seconds; range 1..604800 (7 days),
|
|
272
|
+
bounded by the max lifetime of the derived signing key.
|
|
273
|
+
"""
|
|
274
|
+
if not 1 <= expires_in <= 604800:
|
|
275
|
+
raise ValueError(f"expires_in must be in [1, 604800], got {expires_in}")
|
|
276
|
+
|
|
277
|
+
sign_time = now or _amz_now()
|
|
278
|
+
if sign_time.tzinfo is None:
|
|
279
|
+
raise ValueError("now must be timezone-aware (UTC)")
|
|
280
|
+
|
|
281
|
+
service = ctx["signing_name"]
|
|
282
|
+
region = ctx["signing_region"]
|
|
283
|
+
|
|
284
|
+
amz_date = sign_time.strftime("%Y%m%dT%H%M%SZ")
|
|
285
|
+
date_stamp = sign_time.strftime("%Y%m%d")
|
|
286
|
+
credential_scope = f"{date_stamp}/{region}/{service}/aws4_request"
|
|
287
|
+
|
|
288
|
+
# Canonical headers: host is mandatory; anything else already on the request
|
|
289
|
+
# is signed too (and must then be sent alongside the URL). Strip the headers
|
|
290
|
+
# that belong in the query string to avoid header/query value conflicts
|
|
291
|
+
# (which AWS rejects as InvalidRequest).
|
|
292
|
+
headers = request.headers.copy()
|
|
293
|
+
for h in (
|
|
294
|
+
"Authorization",
|
|
295
|
+
"X-Amz-Date",
|
|
296
|
+
"X-Amz-Content-SHA256",
|
|
297
|
+
"X-Amz-Security-Token",
|
|
298
|
+
):
|
|
299
|
+
if h in headers:
|
|
300
|
+
del headers[h]
|
|
301
|
+
if "host" not in headers and request.url.hostname:
|
|
302
|
+
headers["Host"] = get_host_header_value(request.url)
|
|
303
|
+
|
|
304
|
+
canonical_headers, signed_headers = _canonical_headers(headers)
|
|
305
|
+
|
|
306
|
+
# Signed query params (raw values). X-Amz-Signature is appended afterwards.
|
|
307
|
+
amz_params: list[tuple[str, str]] = [
|
|
308
|
+
("X-Amz-Algorithm", _SIGV4_ALGORITHM),
|
|
309
|
+
("X-Amz-Credential", f"{ctx['access_key_id']}/{credential_scope}"),
|
|
310
|
+
("X-Amz-Date", amz_date),
|
|
311
|
+
("X-Amz-Expires", str(expires_in)),
|
|
312
|
+
("X-Amz-SignedHeaders", signed_headers),
|
|
313
|
+
]
|
|
314
|
+
session_token = ctx.get("session_token")
|
|
315
|
+
if session_token:
|
|
316
|
+
# S3 and most services require the token inside the canonical query.
|
|
317
|
+
# (A few — e.g. iotdevicegateway — want it appended post-signature
|
|
318
|
+
# instead; handle those as a special case if you ever target them.)
|
|
319
|
+
amz_params.append(("X-Amz-Security-Token", session_token))
|
|
320
|
+
|
|
321
|
+
existing = list(URLSearchParams(request.url.search).entries())
|
|
322
|
+
canonical_query = _canonical_query_from_pairs(existing + amz_params)
|
|
323
|
+
|
|
324
|
+
canonical_request = (
|
|
325
|
+
f"{request.method.upper()}\n"
|
|
326
|
+
f"{_canonical_path(request.url.pathname, service=service)}\n"
|
|
327
|
+
f"{canonical_query}\n"
|
|
328
|
+
f"{canonical_headers}\n"
|
|
329
|
+
f"{signed_headers}\n"
|
|
330
|
+
f"UNSIGNED-PAYLOAD"
|
|
331
|
+
)
|
|
332
|
+
|
|
333
|
+
string_to_sign = (
|
|
334
|
+
f"{_SIGV4_ALGORITHM}\n"
|
|
335
|
+
f"{amz_date}\n"
|
|
336
|
+
f"{credential_scope}\n"
|
|
337
|
+
f"{hashlib.sha256(canonical_request.encode('utf-8')).hexdigest()}"
|
|
338
|
+
)
|
|
339
|
+
signing_key = _derive_signing_key(
|
|
340
|
+
ctx["secret_access_key"], date_stamp, region, service
|
|
341
|
+
)
|
|
342
|
+
signature = hmac.new(
|
|
343
|
+
signing_key, string_to_sign.encode("utf-8"), hashlib.sha256
|
|
344
|
+
).hexdigest()
|
|
345
|
+
|
|
346
|
+
# X-Amz-Signature is hex (no encoding needed) and is NOT part of the
|
|
347
|
+
# canonical query. The sorted canonical query doubles as the URL query.
|
|
348
|
+
final_query = f"{canonical_query}&X-Amz-Signature={signature}"
|
|
349
|
+
|
|
350
|
+
url = request.url
|
|
351
|
+
fragment = url.hash or ""
|
|
352
|
+
presigned_href = zapros.URL(
|
|
353
|
+
f"{url.protocol}//{url.host}{url.pathname}?{final_query}{fragment}"
|
|
354
|
+
)
|
|
355
|
+
|
|
356
|
+
if request.body is not None:
|
|
357
|
+
return Request(
|
|
358
|
+
presigned_href,
|
|
359
|
+
request.method,
|
|
360
|
+
headers,
|
|
361
|
+
body=request.body,
|
|
362
|
+
context=request.context,
|
|
363
|
+
)
|
|
364
|
+
return Request(presigned_href, request.method, headers, context=request.context)
|