From b42e5aa50f085e74dbe7f9b20fcb080ea28f289c Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Marek=20Vavru=C5=A1a?= <marek.vavrusa@nic.cz>
Date: Sun, 24 May 2015 23:05:27 +0200
Subject: [PATCH] modules/graphite: push metrics to graphite server(s)

---
 modules/graphite/README.rst   | 48 ++++++++++++++++++++++
 modules/graphite/graphite.lua | 75 +++++++++++++++++++++++++++++++++++
 modules/graphite/graphite.mk  |  2 +
 modules/modules.mk            |  3 +-
 4 files changed, 127 insertions(+), 1 deletion(-)
 create mode 100644 modules/graphite/README.rst
 create mode 100644 modules/graphite/graphite.lua
 create mode 100644 modules/graphite/graphite.mk

diff --git a/modules/graphite/README.rst b/modules/graphite/README.rst
new file mode 100644
index 000000000..da6e3b095
--- /dev/null
+++ b/modules/graphite/README.rst
@@ -0,0 +1,48 @@
+.. _mod-graphite:
+
+Graphite module
+---------------
+
+The module sends statistics over the Graphite_ protocol to either Graphite_, Metronome_, InfluxDB_ or any compatible storage. This allows powerful visualization over metrics collected by Knot DNS Resolver. 
+
+.. tip:: The Graphite server is challenging to get up and running, InfluxDB_ combined with Grafana_ are much easier, and provide richer set of options and available front-ends. Metronome_ by PowerDNS alternatively provides a mini-graphite server for much simpler setups.
+
+Example configuration
+^^^^^^^^^^^^^^^^^^^^^
+
+Only the ``host`` parameter is mandatory.
+
+.. warning:: It uses UDP so it doesn't guarantee the delivery, make sure the target server supports UDP.
+
+.. code-block:: lua
+
+	modules = {
+		graphite = {
+			prefix = hostname(), -- optional metric prefix
+			host = '127.0.0.1',  -- graphite server address
+			port = 2003,         -- graphite server port
+			interval = 5 * sec   -- publish interval
+		}
+	}
+
+The module support sending data to multiple servers at once.
+
+.. code-block:: lua
+
+	modules = {
+		graphite = {
+			host = { '127.0.0.1', '1.2.3.4', '::1' },
+		}
+	}
+
+Dependencies
+^^^^^^^^^^^^
+
+* `luasocket <http://w3.impa.br/~diego/software/luasocket/>`_ available in LuaRocks
+
+    ``$ luarocks install socket``
+
+.. _Graphite: http://graphite.readthedocs.org/en/latest/feeding-carbon.html
+.. _InfluxDB: http://influxdb.com/
+.. _Metronome: https://github.com/ahuPowerDNS/metronome
+.. _Grafana: http://grafana.org/
\ No newline at end of file
diff --git a/modules/graphite/graphite.lua b/modules/graphite/graphite.lua
new file mode 100644
index 000000000..db75317fb
--- /dev/null
+++ b/modules/graphite/graphite.lua
@@ -0,0 +1,75 @@
+--- @module graphite
+local graphite = {}
+
+function graphite.init(module)
+	graphite.socket = require('socket')
+	graphite.ev = nil
+	graphite.cli = {}
+	graphite.prefix = nil
+	return 0
+end
+
+function graphite.deinit(module)
+	if graphite.ev then event.cancel(graphite.ev) end
+	return 0
+end
+
+-- @function Publish results to the Graphite server(s)
+function graphite.publish()
+	local now = os.time()
+	if not graphite.cli then error("no graphite server configured") end
+	local now_metrics = stats.list()
+	if type(now_metrics) ~= 'table' then
+		return 0 -- No metrics to watch
+	end
+	for key,val in pairs(now_metrics) do
+		for i in pairs(graphite.cli) do
+			msg = key..' '..val..' '..now..'\n'
+			if graphite.prefix then
+				msg = prefix..'.'..msg
+			end
+			graphite.cli[i]:send(msg)
+		end
+	end
+	return 0
+end
+
+-- @function Make connection to Graphite server.
+function graphite.add_server(graphite, host, port)
+	local cli, err, status
+	if host:find(':') then
+		cli, err = graphite.socket.udp6()
+	else
+		cli, err = graphite.socket.udp()
+	end
+	if not cli then
+		error(err)
+	end
+	status, err = cli:setpeername(host, port)
+	if not status then
+		error(err)
+	end
+	table.insert(graphite.cli, cli)
+	return 0
+end
+
+function graphite.config(conf)
+	-- config defaults
+	if not conf.port then conf.port = 2003 end
+	if not conf.interval then conf.interval = 5 * sec end
+	if conf.prefix then graphite.prefix = conf.prefix end
+	-- connect to host(s)
+	if type(conf.host) == 'table' then
+		for key, val in pairs(conf.host) do
+			graphite:add_server(val, conf.port)
+		end
+	else
+		graphite:add_server(conf.host, conf.port)
+	end
+	-- start publishing stats
+	if graphite.ev then event.cancel(graphite.ev) end
+	graphite.ev = event.recurrent(conf.interval, graphite.publish)
+	return 0
+end
+
+return graphite
diff --git a/modules/graphite/graphite.mk b/modules/graphite/graphite.mk
new file mode 100644
index 000000000..615f8be50
--- /dev/null
+++ b/modules/graphite/graphite.mk
@@ -0,0 +1,2 @@
+graphite_SOURCES := graphite.lua
+$(call make_lua_module,graphite)
diff --git a/modules/modules.mk b/modules/modules.mk
index d0b08484a..dfbf1d66b 100644
--- a/modules/modules.mk
+++ b/modules/modules.mk
@@ -5,7 +5,8 @@ modules_TARGETS := hints \
 
 # List of Lua modules
 ifeq ($(HAS_lua),yes)
-modules_TARGETS += ketcd 
+modules_TARGETS += ketcd \
+                   graphite
 endif
 
 # List of Golang modules
-- 
GitLab