From 6957aaf27b02be78adaead106695e087a9c9e5fb Mon Sep 17 00:00:00 2001 From: Alexis von Glasow Date: Fri, 17 Mar 2017 16:21:01 +0100 Subject: [PATCH] Add task to deploy formatters Add templates to generate file to format lines based on regex --- README.md | 110 +++++++++++++++++++++-- defaults/main.yml | 4 + tasks/formatting.yml | 34 +++++++ tasks/main.yml | 7 ++ templates/le_formatters/formatters.py.j2 | 29 ++++++ tests/test.yml | 2 +- 6 files changed, 179 insertions(+), 7 deletions(-) create mode 100644 tasks/formatting.yml create mode 100644 templates/le_formatters/formatters.py.j2 diff --git a/README.md b/README.md index 8699f88..df3c2c9 100644 --- a/README.md +++ b/README.md @@ -1,12 +1,16 @@ Logentries ========= -A short role to add configuration of logentries +[![Build Status](https://travis-ci.org/inovia-team/ansible-logentries.svg?branch=master)](https://travis-ci.org/inovia-team/ansible-logentries) + +--- + +A role to add configuration of logentries and allow to format log Requirements ------------ -An account logentries +An account [logentries](https://logentries.com/) Role Variables -------------- @@ -31,6 +35,38 @@ Role Variables logentries_files_on_server: "logfile2", logentries_destination_on_logentries: "log_name_on_logentries" } + logentries_formatters: + format: + - { + method_name: "format_syslog_log", + regexp: "regex", + output_line: "'%s %s %s %s %s'%(self.token, self.identity, self.hostname, self.log_name, line)" + } + formatters: + - { + logentries_project: "syslog", + method_name: "format_syslog_log", + } + +`method_name` in section `logentries_formatters.formatters` *MUST* be the same as +`method_name` in `logentries_formatters.format` it's the link between the logs and how to +format it. Same for `logentries_project` which *MUST* be the same as +`logentries_project` in section `logentries_logs` + +The format of output line should be formatted with mandatory parts, +`output_line` must have at least + + "%s %s %s %s"%(self.token, self.identity, self.hostname, self.log_name) + +The result of parsing line with regexp is stored in variable `pl` + + (?P^[^{]*)(?P{.*})(?P.*) + +All parts of this regexp will be stored as an array and will be accessible: + + pl['firstpart'] + pl['secondpart'] + pl['lastpart'] Example Playbook ---------------- @@ -57,16 +93,37 @@ Add variables in inventory logentries_files_on_server: "/var/log/webserver/error.log", logentries_destination_on_logentries: "myhostname/webserver/error.log" } + logentries_formatters: + format: + - { + method_name: "format_webserver_log", + regexp: "regexp", + output_line: "'%s %s %s %s %s'%(self.identity, self.hostname, self.log_name, self.token, line)" + } + - { + method_name: "format_syslog_log", + regexp: "(?P^[^{]*)(?P{.*})(?P.*)", + output_line: "'%s{\"identity\":\"%s\", \"hostname\":\"%s\", \"appname\":\"%s\", \"data\": %s, \"metadata\":\"%s\", \"lastpart\":\"%s\"}'%(self.token, self.identity, self.hostname, self.log_name, pl['secondpart'], pl['firstpart'], pl['lastpart'])" + } + formatters: + - { + logentries_project: "syslog", + method_name: "format_syslog_log", + } + - { + logentries_project: "project-webserver-error.log", + method_name: "format_webserver_log", + } -This configuration will generate 2 files in /etc/le/conf.d: +This configuration will generate 2 files in `/etc/le/conf.d`: -syslog.conf file will contains: +`syslog.conf` file will contains: [stage-syslog] path = /var/log/syslog destination = project-dev/monolith-syslog -project.conf file will contains: +`project.conf` file will contains: [stage-project-default-log] path = /var/www/project/stage/log/filename.log @@ -76,12 +133,53 @@ project.conf file will contains: path = /var/log/webserver/error.log destination = myhostname/webserver/error.log +If you add formatting part this will add 2 others files `__init__.py` and +`formatters.py` in folder `/etc/le/le_formatters` + +In `formatters.py` you will have a list of methods: + + import re + + class Form(object): + """Formats lines based of pattern given.""" + + def __init__(self, identity, hostname, log_name, token): + self.identity = identity + self.hostname = hostname + self.log_name = log_name + self.token = token + + def format_syslog_log(self, line): + line = line.rstrip() + m = re.match(r"(?P^[^{]*)(?P{.*})(?P.*)", line) + if m: + pl = m.groupdict() + return '%s{"identity":"%s", "hostname":"%s", "appname":"%s", "data": %s, "metadata":"%s", "lastpart":"%s"}'%(self.token, self.identity, self.hostname, self.log_name, pl['secondpart'], pl['firstpart'], pl['lastpart']) + else: + return line + + def format_webserver_log(self, line): + line = line.rstrip() + m = re.match(r"regexp", line) + if m: + pl = m.groupdict() + return '%s %s %s %s %s'%(self.identity, self.hostname, self.log_name, self.token, line) + else: + return line + + formatters = { + 'development-syslog': lambda hostname, log_name, token: Form('development-syslog', hostname, log_name, token).format_syslog_log, + 'development-mobilities-default': lambda hostname, log_name, token: Form('development-mobilities-default', hostname, log_name, token).format_webserver_log, + } + + Use role in playbook +-------------------- - name: logentries hosts: all roles: - - logentries + - ansible-logentries tags: - logentries diff --git a/defaults/main.yml b/defaults/main.yml index 7c1e09e..7da64cf 100644 --- a/defaults/main.yml +++ b/defaults/main.yml @@ -3,3 +3,7 @@ logentries_logs: [] logentries_account_key: "" logentries_stage: "" + +# formatters +logentries_formatters: "" + diff --git a/tasks/formatting.yml b/tasks/formatting.yml new file mode 100644 index 0000000..55a03f6 --- /dev/null +++ b/tasks/formatting.yml @@ -0,0 +1,34 @@ +--- +- name: create folder containing formatters + file: + path: /etc/le/le_formatters/ + state: directory + mode: "u=rwx,g=rx,o=rx" + owner: root + become: yes + tags: + - logentries + - formatters + +- name: initiate module + file: + path: /etc/le/le_formatters/__init__.py + state: touch + mode: "u=rwx,g=rx,o=rx" + owner: root + become: yes + tags: + - logentries + - formatters + +- name: deploy formatting class + template: + src: le_formatters/formatters.py.j2 + dest: /etc/le/le_formatters/formatters.py + validate: python %s + notify: + - Restart logentries + tags: + - logentries + - formatters + diff --git a/tasks/main.yml b/tasks/main.yml index e5c693f..edcda09 100644 --- a/tasks/main.yml +++ b/tasks/main.yml @@ -54,3 +54,10 @@ tags: - logentries +- name: deploy formatting + include: formatting.yml + when: logentries_formatters + tags: + - logentries + - formatters + diff --git a/templates/le_formatters/formatters.py.j2 b/templates/le_formatters/formatters.py.j2 new file mode 100644 index 0000000..70f8da2 --- /dev/null +++ b/templates/le_formatters/formatters.py.j2 @@ -0,0 +1,29 @@ +import re + +class Form(object): + """Formats lines based of pattern given.""" + + def __init__(self, identity, hostname, log_name, token): + self.identity = identity + self.hostname = hostname + self.log_name = log_name + self.token = token + +{% for format in logentries_formatters.format %} + def {{ format.method_name }}(self, line): + line = line.rstrip() + m = re.match(r"{{ format.regexp }}", line) + if m: + pl = m.groupdict() + return {{ format.output_line }} + else: + return line + +{% endfor %} + +formatters = { +{% for format in logentries_formatters.formatters %} + '{{ logentries_stage }}-{{ format.logentries_project }}': lambda hostname, log_name, token: Form('{{ logentries_stage }}-{{ format.logentries_project }}', hostname, log_name, token).{{ format.method_name }}, +{% endfor %} +} + diff --git a/tests/test.yml b/tests/test.yml index f54b43f..fb2f0fb 100644 --- a/tests/test.yml +++ b/tests/test.yml @@ -2,4 +2,4 @@ - hosts: localhost remote_user: root roles: - - logentries \ No newline at end of file + - ansible-logentries