1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
|
/**
* Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
* SPDX-License-Identifier: Apache-2.0.
*/
#include <aws/auth/signable.h>
#include <aws/common/string.h>
#include <aws/http/request_response.h>
/*
* This is a simple aws_signable wrapper implementation for the aws_http_message struct
*/
struct aws_signable_http_request_impl {
struct aws_http_message *request;
struct aws_array_list headers;
};
static int s_aws_signable_http_request_get_property(
const struct aws_signable *signable,
const struct aws_string *name,
struct aws_byte_cursor *out_value) {
struct aws_signable_http_request_impl *impl = signable->impl;
AWS_ZERO_STRUCT(*out_value);
/*
* uri and method can be queried directly from the wrapper request
*/
if (aws_string_eq(name, g_aws_http_uri_property_name)) {
aws_http_message_get_request_path(impl->request, out_value);
} else if (aws_string_eq(name, g_aws_http_method_property_name)) {
aws_http_message_get_request_method(impl->request, out_value);
} else {
return AWS_OP_ERR;
}
return AWS_OP_SUCCESS;
}
static int s_aws_signable_http_request_get_property_list(
const struct aws_signable *signable,
const struct aws_string *name,
struct aws_array_list **out_list) {
struct aws_signable_http_request_impl *impl = signable->impl;
*out_list = NULL;
if (aws_string_eq(name, g_aws_http_headers_property_list_name)) {
*out_list = &impl->headers;
} else {
return AWS_OP_ERR;
}
return AWS_OP_SUCCESS;
}
static int s_aws_signable_http_request_get_payload_stream(
const struct aws_signable *signable,
struct aws_input_stream **out_input_stream) {
struct aws_signable_http_request_impl *impl = signable->impl;
*out_input_stream = aws_http_message_get_body_stream(impl->request);
return AWS_OP_SUCCESS;
}
static void s_aws_signable_http_request_destroy(struct aws_signable *signable) {
if (signable == NULL) {
return;
}
struct aws_signable_http_request_impl *impl = signable->impl;
if (impl == NULL) {
return;
}
aws_array_list_clean_up(&impl->headers);
aws_mem_release(signable->allocator, signable);
}
static struct aws_signable_vtable s_signable_http_request_vtable = {
.get_property = s_aws_signable_http_request_get_property,
.get_property_list = s_aws_signable_http_request_get_property_list,
.get_payload_stream = s_aws_signable_http_request_get_payload_stream,
.destroy = s_aws_signable_http_request_destroy,
};
struct aws_signable *aws_signable_new_http_request(struct aws_allocator *allocator, struct aws_http_message *request) {
struct aws_signable *signable = NULL;
struct aws_signable_http_request_impl *impl = NULL;
aws_mem_acquire_many(
allocator, 2, &signable, sizeof(struct aws_signable), &impl, sizeof(struct aws_signable_http_request_impl));
AWS_ZERO_STRUCT(*signable);
AWS_ZERO_STRUCT(*impl);
signable->allocator = allocator;
signable->vtable = &s_signable_http_request_vtable;
signable->impl = impl;
/*
* Copy the headers since they're not different types
*/
size_t header_count = aws_http_message_get_header_count(request);
if (aws_array_list_init_dynamic(
&impl->headers, allocator, header_count, sizeof(struct aws_signable_property_list_pair))) {
goto on_error;
}
for (size_t i = 0; i < header_count; ++i) {
struct aws_http_header header;
aws_http_message_get_header(request, &header, i);
struct aws_signable_property_list_pair property = {.name = header.name, .value = header.value};
aws_array_list_push_back(&impl->headers, &property);
}
impl->request = request;
return signable;
on_error:
aws_signable_destroy(signable);
return NULL;
}
|