1 | /* |
2 | * Copyright 2010-2018 Amazon.com, Inc. or its affiliates. All Rights Reserved. |
3 | * |
4 | * Licensed under the Apache License, Version 2.0 (the "License"). |
5 | * You may not use this file except in compliance with the License. |
6 | * A copy of the License is located at |
7 | * |
8 | * http://aws.amazon.com/apache2.0 |
9 | * |
10 | * or in the "license" file accompanying this file. This file is distributed |
11 | * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either |
12 | * express or implied. See the License for the specific language governing |
13 | * permissions and limitations under the License. |
14 | */ |
15 | |
16 | #include <aws/core/utils/memory/AWSMemory.h> |
17 | #include <aws/core/monitoring/DefaultMonitoring.h> |
18 | #include <aws/core/utils/DateTime.h> |
19 | #include <aws/core/utils/json/JsonSerializer.h> |
20 | #include <aws/core/utils/Outcome.h> |
21 | #include <aws/core/client/AWSClient.h> |
22 | #include <aws/core/auth/AWSCredentialsProvider.h> |
23 | #include <aws/core/platform/Environment.h> |
24 | #include <aws/core/config/AWSProfileConfigLoader.h> |
25 | #include <aws/core/utils/logging/LogMacros.h> |
26 | using namespace Aws::Utils; |
27 | |
28 | namespace Aws |
29 | { |
30 | namespace Monitoring |
31 | { |
32 | static const char DEFAULT_MONITORING_ALLOC_TAG[] = "DefaultMonitoringAllocTag" ; |
33 | static const int CLIENT_ID_LENGTH_LIMIT = 256; |
34 | static const int USER_AGENT_LENGHT_LIMIT = 256; |
35 | static const int ERROR_MESSAGE_LENGTH_LIMIT = 512; |
36 | |
37 | const char DEFAULT_MONITORING_CLIENT_ID[] = "" ; // default to empty; |
38 | const char DEFAULT_MONITORING_HOST[] = "127.0.0.1" ; // default to loopback ip address instead of "localhost" based on design specification. |
39 | unsigned short DEFAULT_MONITORING_PORT = 31000; //default to 31000; |
40 | bool DEFAULT_MONITORING_ENABLE = false; //default to false; |
41 | |
42 | const int DefaultMonitoring::DEFAULT_MONITORING_VERSION = 1; |
43 | const char DefaultMonitoring::DEFAULT_CSM_CONFIG_ENABLED[] = "csm_enabled" ; |
44 | const char DefaultMonitoring::DEFAULT_CSM_CONFIG_CLIENT_ID[] = "csm_client_id" ; |
45 | const char DefaultMonitoring::DEFAULT_CSM_CONFIG_HOST[] = "csm_host" ; |
46 | const char DefaultMonitoring::DEFAULT_CSM_CONFIG_PORT[] = "csm_port" ; |
47 | const char DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_ENABLED[] = "AWS_CSM_ENABLED" ; |
48 | const char DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_CLIENT_ID[] = "AWS_CSM_CLIENT_ID" ; |
49 | const char DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_HOST[] = "AWS_CSM_HOST" ; |
50 | const char DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_PORT[] = "AWS_CSM_PORT" ; |
51 | |
52 | |
53 | struct DefaultContext |
54 | { |
55 | Aws::Utils::DateTime apiCallStartTime; |
56 | Aws::Utils::DateTime attemptStartTime; |
57 | int retryCount = 0; |
58 | bool lastAttemptSucceeded = false; |
59 | bool lastErrorRetriable = false; //dosen't apply if last attempt succeeded. |
60 | const Aws::Client::HttpResponseOutcome* outcome = nullptr; |
61 | }; |
62 | |
63 | static inline void FillRequiredFieldsToJson(Json::JsonValue& json, |
64 | const Aws::String& type, |
65 | const Aws::String& service, |
66 | const Aws::String& api, |
67 | const Aws::String& clientId, |
68 | const DateTime& timestamp, |
69 | int version, |
70 | const Aws::String& userAgent) |
71 | { |
72 | json.WithString("Type" , type) |
73 | .WithString("Service" , service) |
74 | .WithString("Api" , api) |
75 | .WithString("ClientId" , clientId.substr(0, CLIENT_ID_LENGTH_LIMIT)) |
76 | .WithInt64("Timestamp" , timestamp.Millis()) |
77 | .WithInteger("Version" , version) |
78 | .WithString("UserAgent" , userAgent.substr(0, USER_AGENT_LENGHT_LIMIT)); |
79 | } |
80 | |
81 | static inline void FillRequiredApiCallFieldsToJson(Json::JsonValue& json, |
82 | int attemptCount, |
83 | int64_t apiCallLatency, |
84 | bool maxRetriesExceeded) |
85 | { |
86 | json.WithInteger("AttemptCount" , attemptCount) |
87 | .WithInt64("Latency" , apiCallLatency) |
88 | .WithInteger("MaxRetriesExceeded" , maxRetriesExceeded ? 1 : 0); |
89 | } |
90 | |
91 | static inline void FillRequiredApiAttemptFieldsToJson(Json::JsonValue& json, |
92 | const Aws::String& domainName, |
93 | int64_t attemptLatency) |
94 | { |
95 | json.WithString("Fqdn" , domainName) |
96 | .WithInt64("AttemptLatency" , attemptLatency); |
97 | } |
98 | |
99 | static inline void (Json::JsonValue& json, const Aws::Http::HeaderValueCollection& , |
100 | const Aws::String& , const Aws::String& targetName) |
101 | { |
102 | auto iter = headers.find(headerName); |
103 | if (iter != headers.end()) |
104 | { |
105 | json.WithString(targetName, iter->second); |
106 | } |
107 | } |
108 | |
109 | static inline void ExportHttpMetricsToJson(Json::JsonValue& json, const Aws::Monitoring::HttpClientMetricsCollection& httpMetrics, Aws::Monitoring::HttpClientMetricsType type) |
110 | { |
111 | auto iter = httpMetrics.find(GetHttpClientMetricNameByType(type)); |
112 | if (iter != httpMetrics.end()) |
113 | { |
114 | json.WithInt64(GetHttpClientMetricNameByType(type), iter->second); |
115 | } |
116 | } |
117 | |
118 | static inline void FillOptionalApiCallFieldsToJson(Json::JsonValue& json, |
119 | const Aws::Http::HttpRequest* request, |
120 | const Aws::Client::HttpResponseOutcome& outcome) |
121 | { |
122 | if (!request->GetSigningRegion().empty()) |
123 | { |
124 | json.WithString("Region" , request->GetSigningRegion()); |
125 | } |
126 | if (!outcome.IsSuccess()) |
127 | { |
128 | if (outcome.GetError().GetExceptionName().empty()) // Not Aws Excecption |
129 | { |
130 | json.WithString("FinalSdkExceptionMessage" , outcome.GetError().GetMessage().substr(0, ERROR_MESSAGE_LENGTH_LIMIT)); |
131 | } |
132 | else // Aws Exception |
133 | { |
134 | json.WithString("FinalAwsException" , outcome.GetError().GetExceptionName()) |
135 | .WithString("FinalAwsExceptionMessage" , outcome.GetError().GetMessage().substr(0, ERROR_MESSAGE_LENGTH_LIMIT)); |
136 | } |
137 | json.WithInteger("FinalHttpStatusCode" , static_cast<int>(outcome.GetError().GetResponseCode())); |
138 | } |
139 | else |
140 | { |
141 | json.WithInteger("FinalHttpStatusCode" , static_cast<int>(outcome.GetResult()->GetResponseCode())); |
142 | } |
143 | } |
144 | |
145 | static inline void FillOptionalApiAttemptFieldsToJson(Json::JsonValue& json, |
146 | const Aws::Http::HttpRequest* request, |
147 | const Aws::Client::HttpResponseOutcome& outcome, |
148 | const CoreMetricsCollection& metricsFromCore) |
149 | { |
150 | /** |
151 | *No matter request succeeded or not, these fields should be included as long as their requirements |
152 | *are met. We should be able to access response (so as to access original requeset) if the response has error. |
153 | */ |
154 | if (request->HasAwsSessionToken() && !request->GetAwsSessionToken().empty()) |
155 | { |
156 | json.WithString("SessionToken" , request->GetAwsSessionToken()); |
157 | } |
158 | if (!request->GetSigningRegion().empty()) |
159 | { |
160 | json.WithString("Region" , request->GetSigningRegion()); |
161 | } |
162 | if (!request->GetSigningAccessKey().empty()) |
163 | { |
164 | json.WithString("AccessKey" , request->GetSigningAccessKey()); |
165 | } |
166 | |
167 | const auto& = outcome.IsSuccess() ? outcome.GetResult()->GetHeaders() : outcome.GetError().GetResponseHeaders(); |
168 | |
169 | ExportResponseHeaderToJson(json, headers, StringUtils::ToLower("x-amzn-RequestId" ), "XAmznRequestId" ); |
170 | ExportResponseHeaderToJson(json, headers, StringUtils::ToLower("x-amz-request-id" ), "XAmzRequestId" ); |
171 | ExportResponseHeaderToJson(json, headers, StringUtils::ToLower("x-amz-id-2" ), "XAmzId2" ); |
172 | |
173 | if (!outcome.IsSuccess()) |
174 | { |
175 | if (outcome.GetError().GetExceptionName().empty()) // Not Aws Excecption |
176 | { |
177 | json.WithString("SdkExceptionMessage" , outcome.GetError().GetMessage().substr(0, ERROR_MESSAGE_LENGTH_LIMIT)); |
178 | } |
179 | else // Aws Exception |
180 | { |
181 | json.WithString("AwsException" , outcome.GetError().GetExceptionName()) |
182 | .WithString("AwsExceptionMessage" , outcome.GetError().GetMessage().substr(0, ERROR_MESSAGE_LENGTH_LIMIT)); |
183 | } |
184 | json.WithInteger("HttpStatusCode" , static_cast<int>(outcome.GetError().GetResponseCode())); |
185 | } |
186 | else |
187 | { |
188 | json.WithInteger("HttpStatusCode" , static_cast<int>(outcome.GetResult()->GetResponseCode())); |
189 | } |
190 | |
191 | // Optional MetricsCollectedFromCore |
192 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::AcquireConnectionLatency); |
193 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::ConnectionReused); |
194 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::ConnectLatency); |
195 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::DestinationIp); |
196 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::DnsLatency); |
197 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::RequestLatency); |
198 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::SslLatency); |
199 | ExportHttpMetricsToJson(json, metricsFromCore.httpClientMetrics, HttpClientMetricsType::TcpLatency); |
200 | } |
201 | |
202 | DefaultMonitoring::DefaultMonitoring(const Aws::String& clientId, const Aws::String& host, unsigned short port): |
203 | m_udp(host.c_str(), port), m_clientId(clientId) |
204 | { |
205 | } |
206 | |
207 | void* DefaultMonitoring::OnRequestStarted(const Aws::String& serviceName, const Aws::String& requestName, const std::shared_ptr<const Aws::Http::HttpRequest>& request) const |
208 | { |
209 | AWS_UNREFERENCED_PARAM(request); |
210 | |
211 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "OnRequestStart Service: " << serviceName << "Request: " << requestName); |
212 | auto context = Aws::New<DefaultContext>(DEFAULT_MONITORING_ALLOC_TAG); |
213 | context->apiCallStartTime = Aws::Utils::DateTime::Now(); |
214 | context->attemptStartTime = context->apiCallStartTime; |
215 | context->retryCount = 0; |
216 | return context; |
217 | } |
218 | |
219 | |
220 | void DefaultMonitoring::OnRequestSucceeded(const Aws::String& serviceName, const Aws::String& requestName, const std::shared_ptr<const Aws::Http::HttpRequest>& request, |
221 | const Aws::Client::HttpResponseOutcome& outcome, const CoreMetricsCollection& metricsFromCore, void* context) const |
222 | { |
223 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "OnRequestSucceeded Service: " << serviceName << "Request: " << requestName); |
224 | CollectAndSendAttemptData(serviceName, requestName, request, outcome, metricsFromCore, context); |
225 | } |
226 | |
227 | void DefaultMonitoring::OnRequestFailed(const Aws::String& serviceName, const Aws::String& requestName, const std::shared_ptr<const Aws::Http::HttpRequest>& request, |
228 | const Aws::Client::HttpResponseOutcome& outcome, const CoreMetricsCollection& metricsFromCore, void* context) const |
229 | { |
230 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "OnRequestFailed Service: " << serviceName << "Request: " << requestName); |
231 | CollectAndSendAttemptData(serviceName, requestName, request, outcome, metricsFromCore, context); |
232 | } |
233 | |
234 | void DefaultMonitoring::OnRequestRetry(const Aws::String& serviceName, const Aws::String& requestName, |
235 | const std::shared_ptr<const Aws::Http::HttpRequest>& request, void* context) const |
236 | { |
237 | AWS_UNREFERENCED_PARAM(request); |
238 | |
239 | DefaultContext* defaultContext = static_cast<DefaultContext*>(context); |
240 | defaultContext->retryCount++; |
241 | defaultContext->attemptStartTime = Aws::Utils::DateTime::Now(); |
242 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "OnRequestRetry Service: " << serviceName << "Request: " << requestName << " RetryCnt:" << defaultContext->retryCount); |
243 | } |
244 | |
245 | void DefaultMonitoring::OnFinish(const Aws::String& serviceName, const Aws::String& requestName, |
246 | const std::shared_ptr<const Aws::Http::HttpRequest>& request, void* context) const |
247 | { |
248 | AWS_UNREFERENCED_PARAM(request); |
249 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "OnRequestFinish Service: " << serviceName << "Request: " << requestName); |
250 | |
251 | DefaultContext* defaultContext = static_cast<DefaultContext*>(context); |
252 | Aws::Utils::Json::JsonValue json; |
253 | FillRequiredFieldsToJson(json, "ApiCall" , serviceName, requestName, m_clientId, defaultContext->apiCallStartTime, DEFAULT_MONITORING_VERSION, request->GetUserAgent()); |
254 | FillRequiredApiCallFieldsToJson(json, defaultContext->retryCount + 1, (DateTime::Now() - defaultContext->apiCallStartTime).count(), (!defaultContext->lastAttemptSucceeded && defaultContext->lastErrorRetriable)); |
255 | FillOptionalApiCallFieldsToJson(json, request.get(), *(defaultContext->outcome)); |
256 | Aws::String compactData = json.View().WriteCompact(); |
257 | m_udp.SendData(reinterpret_cast<const uint8_t*>(compactData.c_str()), static_cast<int>(compactData.size())); |
258 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Send API Metrics: \n" << json.View().WriteReadable()); |
259 | Aws::Delete(defaultContext); |
260 | } |
261 | |
262 | void DefaultMonitoring::CollectAndSendAttemptData(const Aws::String& serviceName, const Aws::String& requestName, |
263 | const std::shared_ptr<const Aws::Http::HttpRequest>& request, const Aws::Client::HttpResponseOutcome& outcome, |
264 | const CoreMetricsCollection& metricsFromCore, void* context) const |
265 | { |
266 | DefaultContext* defaultContext = static_cast<DefaultContext*>(context); |
267 | defaultContext->outcome = &outcome; |
268 | defaultContext->lastAttemptSucceeded = outcome.IsSuccess() ? true : false; |
269 | defaultContext->lastErrorRetriable = (!outcome.IsSuccess() && outcome.GetError().ShouldRetry()) ? true : false; |
270 | Aws::Utils::Json::JsonValue json; |
271 | FillRequiredFieldsToJson(json, "ApiCallAttempt" , serviceName, requestName, m_clientId, defaultContext->attemptStartTime, DEFAULT_MONITORING_VERSION, request->GetUserAgent()); |
272 | FillRequiredApiAttemptFieldsToJson(json, request->GetUri().GetAuthority(), (DateTime::Now() - defaultContext->attemptStartTime).count()); |
273 | FillOptionalApiAttemptFieldsToJson(json, request.get(), outcome, metricsFromCore); |
274 | Aws::String compactData = json.View().WriteCompact(); |
275 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Send Attempt Metrics: \n" << json.View().WriteReadable()); |
276 | m_udp.SendData(reinterpret_cast<const uint8_t*>(compactData.c_str()), static_cast<int>(compactData.size())); |
277 | } |
278 | |
279 | Aws::UniquePtr<MonitoringInterface> DefaultMonitoringFactory::CreateMonitoringInstance() const |
280 | { |
281 | Aws::String clientId(DEFAULT_MONITORING_CLIENT_ID); // default to empty |
282 | Aws::String host(DEFAULT_MONITORING_HOST); // default to 127.0.0.1 |
283 | unsigned short port = DEFAULT_MONITORING_PORT; // default to 31000 |
284 | bool enable = DEFAULT_MONITORING_ENABLE; //default to false; |
285 | |
286 | //check profile_config |
287 | Aws::String tmpEnable = Aws::Config::GetCachedConfigValue(DefaultMonitoring::DEFAULT_CSM_CONFIG_ENABLED); |
288 | Aws::String tmpClientId = Aws::Config::GetCachedConfigValue(DefaultMonitoring::DEFAULT_CSM_CONFIG_CLIENT_ID); |
289 | Aws::String tmpHost = Aws::Config::GetCachedConfigValue(DefaultMonitoring::DEFAULT_CSM_CONFIG_HOST); |
290 | Aws::String tmpPort = Aws::Config::GetCachedConfigValue(DefaultMonitoring::DEFAULT_CSM_CONFIG_PORT); |
291 | |
292 | if (!tmpEnable.empty()) |
293 | { |
294 | enable = StringUtils::CaselessCompare(tmpEnable.c_str(), "true" ) ? true : false; |
295 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved csm_enabled from profile_config to be " << enable); |
296 | } |
297 | if (!tmpClientId.empty()) |
298 | { |
299 | clientId = tmpClientId; |
300 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved csm_client_id from profile_config to be " << clientId); |
301 | } |
302 | |
303 | if (!tmpHost.empty()) |
304 | { |
305 | host = tmpHost; |
306 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved csm_host from profile_config to be " << host); |
307 | } |
308 | |
309 | if (!tmpPort.empty()) |
310 | { |
311 | port = static_cast<short>(StringUtils::ConvertToInt32(tmpPort.c_str())); |
312 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved csm_port from profile_config to be " << port); |
313 | } |
314 | |
315 | // check environment variables |
316 | tmpEnable = Aws::Environment::GetEnv(DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_ENABLED); |
317 | tmpClientId = Aws::Environment::GetEnv(DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_CLIENT_ID); |
318 | tmpHost = Aws::Environment::GetEnv(DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_HOST); |
319 | tmpPort = Aws::Environment::GetEnv(DefaultMonitoring::DEFAULT_CSM_ENVIRONMENT_VAR_PORT); |
320 | if (!tmpEnable.empty()) |
321 | { |
322 | enable = StringUtils::CaselessCompare(tmpEnable.c_str(), "true" ) ? true : false; |
323 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved AWS_CSM_ENABLED from Environment variable to be " << enable); |
324 | } |
325 | if (!tmpClientId.empty()) |
326 | { |
327 | clientId = tmpClientId; |
328 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved AWS_CSM_CLIENT_ID from Environment variable to be " << clientId); |
329 | |
330 | } |
331 | if (!tmpHost.empty()) |
332 | { |
333 | host = tmpHost; |
334 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved AWS_CSM_HOST from Environment variable to be " << host); |
335 | } |
336 | if (!tmpPort.empty()) |
337 | { |
338 | port = static_cast<unsigned short>(StringUtils::ConvertToInt32(tmpPort.c_str())); |
339 | AWS_LOGSTREAM_DEBUG(DEFAULT_MONITORING_ALLOC_TAG, "Resolved AWS_CSM_PORT from Environment variable to be " << port); |
340 | } |
341 | |
342 | if (!enable) |
343 | { |
344 | return nullptr; |
345 | } |
346 | return Aws::MakeUnique<DefaultMonitoring>(DEFAULT_MONITORING_ALLOC_TAG, clientId, host, port); |
347 | } |
348 | |
349 | } |
350 | } |
351 | |