This repository has been archived by the owner on May 3, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathelb_to_logentries.py
106 lines (93 loc) · 3.8 KB
/
elb_to_logentries.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
#!/usr/bin/env python
import socket
import boto3
import csv
import json
import zlib
s3_client = boto3.resource('s3')
with open("tokens.json", 'r') as f:
tokens = json.load(f)
def find_token(key):
"""Finds the appropriate token for the key (filename)"""
filename = key.split('/')[-1]
for prefix, t in tokens.items():
if filename.startswith(prefix):
return t
raise ValueError("No token found for " + key)
def lambda_handler(event, context):
"""AWS lambda entry point"""
for record in event['Records']:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s3_out(record['s3']['bucket']['name'], record['s3']['object']['key'])
def s3_out(bucket, key, token=None, dummy=False):
"""Download s3 file and output to logentries"""
if not token:
token = find_token(key)
if dummy:
s = Dummy()
else:
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect(('data.logentries.com', 80))
object = s3_client.Object(bucket, key)
if(key.endswith("log.gz")):
compressed = object.get()["Body"].read()
body = zlib.decompress(compressed, 15+32).decode('utf-8')
elbV2 = True
else:
body = object.get()["Body"].read()
elbV2 = False
rows = csv.reader((line.replace('\0','') for line in body.splitlines()), delimiter=' ', quotechar='"')
for line in rows:
# log line format is:
# timestamp elb client:port backend:port request_processing_time backend_processing_time response_processing_time elb_status_code backend_status_code received_bytes sent_bytes "request" "user_agent" ssl_cipher ssl_protocol
if elbV2:
if len(line) > 23:
request = line[12].split(' ')
idx = request[1].find('/', 9)
url = request[1][idx:]
parsed = {
'ip': line[3].split(':')[0],
'method': request[0],
'url': url,
'user_agent': line[13]
}
msg = "\"{1}\" ip=\"{ip}\" request_time=\"{6}\" elb_status=\"{8}\" backend_status=\"{9}\"" \
" bytes_received=\"{10}\" bytes_sent=\"{11}\" method=\"{method}\" url=\"{url}\"" \
" user_agent=\"{user_agent}\"\n" \
.format(*line, **parsed)
else:
s.send(token + "ERROR line too short: " + ' '.join(line) + "\n")
else:
if len(line) > 12:
request = line[11].split(' ')
idx = request[1].find('/', 9)
url = request[1][idx:]
parsed = {
'ip': line[2].split(':')[0],
'method': request[0],
'url': url,
'user_agent': line[12]
}
msg = "\"{0}\" ip=\"{ip}\" request_time=\"{5}\" elb_status=\"{7}\" backend_status=\"{8}\"" \
" bytes_received=\"{9}\" bytes_sent=\"{10}\" method=\"{method}\" url=\"{url}\"" \
" user_agent=\"{user_agent}\"\n"\
.format(*line, **parsed)
else:
s.send(token + "ERROR line too short: " + ' '.join(line) + "\n")
s.close()
class Dummy:
def connect(self, dummy):
pass
def send(self, data):
print(data),
def close(self):
pass
# test execution
if __name__ == "__main__":
import argparse
parser = argparse.ArgumentParser(description='Import s3 logs to logentries')
parser.add_argument('-t', dest='token', required=False, help='logentries token')
parser.add_argument('bucket', help='s3 bucket name')
parser.add_argument('key', help='s3 key')
args = parser.parse_args()
s3_out(args.bucket, args.key, args.token, dummy=True)