-
Notifications
You must be signed in to change notification settings - Fork 813
/
Copy pathemitter.py
283 lines (219 loc) · 10.2 KB
/
emitter.py
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
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
# (C) Datadog, Inc. 2010-2016
# All rights reserved
# Licensed under Simplified BSD License (see LICENSE)
# stdlib
from hashlib import md5
import logging
import re
import string
import zlib
import unicodedata
# 3p
import requests
import simplejson as json
# project
from config import get_version
from utils.proxy import set_no_proxy_settings
set_no_proxy_settings()
# urllib3 logs a bunch of stuff at the info level
requests_log = logging.getLogger("requests.packages.urllib3")
requests_log.setLevel(logging.WARN)
requests_log.propagate = True
# From http://stackoverflow.com/questions/92438/stripping-non-printable-characters-from-a-string-in-python
control_chars = ''.join(map(unichr, range(0, 32) + range(127, 160)))
control_char_re = re.compile('[%s]' % re.escape(control_chars))
# Only enforced for the metrics API on our end, for now
MAX_COMPRESSED_SIZE = 2 << 20 # 2MB, the backend should accept up to 3MB but let's be conservative here
# maximum depth of recursive calls to payload splitting function, a bit arbitrary (we don't want too much
# depth in the recursive calls, but we want to give up only when some metrics are clearly too large)
MAX_SPLIT_DEPTH = 2
def remove_control_chars(s, log):
if isinstance(s, str):
sanitized = control_char_re.sub('', s)
elif isinstance(s, unicode):
sanitized = ''.join(['' if unicodedata.category(c) in ['Cc','Cf'] else c
for c in u'{}'.format(s)])
if sanitized != s:
log.warning('Removed control chars from string: ' + s)
return sanitized
def remove_undecodable_chars(s, log):
sanitized = s
if isinstance(s, str):
try:
s.decode('utf8')
except UnicodeDecodeError:
sanitized = s.decode('utf8', errors='ignore')
log.warning(u'Removed undecodable chars from string: ' + s.decode('utf8', errors='replace'))
return sanitized
def sanitize_payload(item, log, sanitize_func):
if isinstance(item, dict):
newdict = {}
for k, v in item.iteritems():
newval = sanitize_payload(v, log, sanitize_func)
newkey = sanitize_func(k, log)
newdict[newkey] = newval
return newdict
if isinstance(item, list):
newlist = []
for listitem in item:
newlist.append(sanitize_payload(listitem, log, sanitize_func))
return newlist
if isinstance(item, tuple):
newlist = []
for listitem in item:
newlist.append(sanitize_payload(listitem, log, sanitize_func))
return tuple(newlist)
if isinstance(item, basestring):
return sanitize_func(item, log)
return item
def post_payload(url, message, serialize_func, agentConfig, log):
log.debug('http_emitter: attempting postback to ' + string.split(url, "api_key=")[0])
try:
payloads = serialize_func(message, MAX_COMPRESSED_SIZE, 0, log)
except UnicodeDecodeError:
log.exception('http_emitter: Unable to convert message to json')
# early return as we can't actually process the message
return
except RuntimeError:
log.exception('http_emitter: runtime error dumping message to json')
# early return as we can't actually process the message
return
except Exception:
log.exception('http_emitter: unknown exception processing message')
return
for payload in payloads:
try:
headers = get_post_headers(agentConfig, payload)
r = requests.post(url, data=payload, timeout=5, headers=headers)
r.raise_for_status()
if r.status_code >= 200 and r.status_code < 205:
log.debug("Payload accepted")
except Exception as e:
log.error("Unable to post payload: %s" % e.message)
def serialize_payload(message, log):
payload = ""
try:
payload = json.dumps(message)
except UnicodeDecodeError:
newmessage = sanitize_payload(message, log, remove_control_chars)
try:
payload = json.dumps(newmessage)
except UnicodeDecodeError:
log.info('Removing undecodable characters from payload')
newmessage = sanitize_payload(newmessage, log, remove_undecodable_chars)
payload = json.dumps(newmessage)
return payload
def serialize_and_compress_legacy_payload(legacy_payload, max_compressed_size, depth, log):
"""
Serialize and compress the legacy payload
"""
serialized_payload = serialize_payload(legacy_payload, log)
zipped = zlib.compress(serialized_payload)
log.debug("payload_size=%d, compressed_size=%d, compression_ratio=%.3f"
% (len(serialized_payload), len(zipped), float(len(serialized_payload))/float(len(zipped))))
compressed_payloads = [zipped]
if len(zipped) > max_compressed_size:
# let's just log a warning for now, splitting the legacy payload is tricky
log.warning("collector payload is above the limit of %dKB compressed", max_compressed_size/(1 << 10))
return compressed_payloads
def serialize_and_compress_metrics_payload(metrics_payload, max_compressed_size, depth, log):
"""
Serialize and compress the metrics payload
If the compressed payload is too big, we attempt to split it into smaller payloads and call this
function recursively on each smaller payload
"""
compressed_payloads = []
serialized_payload = serialize_payload(metrics_payload, log)
zipped = zlib.compress(serialized_payload)
compression_ratio = float(len(serialized_payload))/float(len(zipped))
log.debug("payload_size=%d, compressed_size=%d, compression_ratio=%.3f"
% (len(serialized_payload), len(zipped), compression_ratio))
if len(zipped) < max_compressed_size:
compressed_payloads.append(zipped)
else:
series = metrics_payload["series"]
if depth > MAX_SPLIT_DEPTH:
log.error("Maximum depth of payload splitting reached, dropping the %d metrics in this chunk", len(series))
return compressed_payloads
# Try to account for the compression when estimating the number of chunks needed to get small-enough chunks
n_chunks = len(zipped)/max_compressed_size + 1 + int(compression_ratio/2)
log.debug("payload is too big (%d bytes), splitting it in %d chunks", len(zipped), n_chunks)
series_per_chunk = len(series)/n_chunks + 1
for i in range(n_chunks):
# Create each chunk and make them go through this function recursively ; increment the `depth` of the recursive call
compressed_payloads.extend(
serialize_and_compress_metrics_payload(
{
"series": series[i*series_per_chunk:(i+1)*series_per_chunk]
},
max_compressed_size,
depth+1,
log
)
)
return compressed_payloads
def serialize_and_compress_checkruns_payload(checkruns_payload, max_compressed_size, depth, log):
"""
Serialize and compress the checkruns payload
"""
serialized_payload = serialize_payload(checkruns_payload, log)
zipped = zlib.compress(serialized_payload)
log.debug("payload_size=%d, compressed_size=%d, compression_ratio=%.3f"
% (len(serialized_payload), len(zipped), float(len(serialized_payload))/float(len(zipped))))
compressed_payloads = [zipped]
if len(zipped) > max_compressed_size:
# let's just log a warning for now, splitting the legacy payload is tricky
log.warning("checkruns payload is above the limit of %dKB compressed", max_compressed_size/(1 << 10))
return compressed_payloads
def split_payload(legacy_payload):
metrics_payload = {"series": []}
# See https://github.com/DataDog/dd-agent/blob/5.11.1/checks/__init__.py#L905-L926 for format
for ts in legacy_payload['metrics']:
sample = {
"metric": ts[0],
"points": [(ts[1], ts[2])],
"source_type_name": "System",
}
if len(ts) >= 4:
# Default to the metric hostname if present
if ts[3].get('hostname'):
sample['host'] = ts[3]['hostname']
else:
# If not use the general payload one
sample['host'] = legacy_payload['internalHostname']
if ts[3].get('type'):
sample['type'] = ts[3]['type']
if ts[3].get('tags'):
sample['tags'] = ts[3]['tags']
if ts[3].get('device_name'):
sample['device'] = ts[3]['device_name']
metrics_payload["series"].append(sample)
del legacy_payload['metrics']
checkruns_payload = legacy_payload["service_checks"]
del legacy_payload["service_checks"]
return legacy_payload, metrics_payload, checkruns_payload
def http_emitter(message, log, agentConfig, endpoint):
api_key = message.get('apiKey')
if not api_key:
raise Exception("The http emitter requires an api key")
# For perf reason. We now want to send the metrics to the api endpoint. So we are extracting them
# from the payload here, transform them into the expected format and send them (via the forwarder)
legacy_url = "{0}/intake/{1}?api_key={2}".format(agentConfig['dd_url'], endpoint, api_key)
metrics_endpoint = "{0}/api/v1/series?api_key={1}".format(agentConfig['dd_url'], api_key)
checkruns_endpoint = "{0}/api/v1/check_run?api_key={1}".format(agentConfig['dd_url'], api_key)
legacy_payload, metrics_payload, checkruns_payload = split_payload(message)
# Post legacy payload
post_payload(legacy_url, legacy_payload, serialize_and_compress_legacy_payload, agentConfig, log)
# Post metrics payload
post_payload(metrics_endpoint, metrics_payload, serialize_and_compress_metrics_payload, agentConfig, log)
# Post check runs payload
post_payload(checkruns_endpoint, checkruns_payload, serialize_and_compress_checkruns_payload, agentConfig, log)
def get_post_headers(agentConfig, payload):
return {
'User-Agent': 'Datadog Agent/%s' % agentConfig['version'],
'Content-Type': 'application/json',
'Content-Encoding': 'deflate',
'Accept': 'text/html, */*',
'Content-MD5': md5(payload).hexdigest(),
'DD-Collector-Version': get_version()
}