#!/usr/bin/env lua

--[[
    
    fsync.lua - synchronize /etc/config/flukso settings with the sensor board
                via the spid ctrl fifos

    Copyright (C) 2011 Bart Van Der Meerssche <bart.vandermeerssche@flukso.net>

    This program is free software: you can redistribute it and/or modify
    it under the terms of the GNU General Public License as published by
    the Free Software Foundation, either version 3 of the License, or
    (at your option) any later version.

    This program is distributed in the hope that it will be useful,
    but WITHOUT ANY WARRANTY; without even the implied warranty of
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    GNU General Public License for more details.

    You should have received a copy of the GNU General Public License
    along with this program.  If not, see <http://www.gnu.org/licenses/>.

]]--


local dbg        = require 'dbg'
local nixio      = require 'nixio'
nixio.fs         = require 'nixio.fs'
local uci        = require 'luci.model.uci'.cursor()
local luci       = require 'luci'
luci.json        = require 'luci.json'
local httpclient = require 'luci.httpclient'


local HW_CHECK_OVERRIDE	 = (arg[1] == '-f')

local CTRL_PATH		 = '/var/run/spid/ctrl'
local CTRL_PATH_IN	 = CTRL_PATH .. '/in'
local CTRL_PATH_OUT	 = CTRL_PATH .. '/out'

local O_RDWR_NONBLOCK	 = nixio.open_flags('rdwr', 'nonblock')
local O_RDWR_CREAT	 = nixio.open_flags('rdwr', 'creat')
local POLLIN		 = nixio.poll_flags('in')
local POLL_TIMEOUT_MS	 = 1000
local MAX_TRIES		 = 5

-- parse and load /etc/config/flukso
local flukso = uci:get_all('flukso')

local MAX_SENSORS	 = tonumber(flukso.main.max_sensors)
local MAX_ANALOG_SENSORS = tonumber(flukso.main.max_analog_sensors)
local RESET_COUNTERS	 = (flukso.main.reset_counters == '1')
local SYNC_TO_SERVER	 = (flukso.daemon.enable_wan_branch == '1')

local METERCONST_FACTOR	 = 0.449

-- sensor board commands
local GET_HW_VERSION	 = 'gh'
local GET_HW_VERSION_R	 = '^gh%s+(%d+)%s+(%d+)$'
local SET_ENABLE	 = 'se %d %d'
local SET_PHY_TO_LOG	 = 'sp' -- with [1..MAX_SENSORS] arguments
local SET_METERCONST	 = 'sm %d %d'
local SET_COUNTER	 = 'sc %d %d'
local COMMIT		 = 'ct'

-- LAN settings
local API_PATH		 = '/www/sensor/'
local CGI_SCRIPT	 = '/usr/bin/restful'
local AVAHI_PATH	 = '/etc/avahi/services/flukso.service'

-- WAN settings
local WAN_BASE_URL	 = flukso.daemon.wan_base_url .. 'sensor/'
local WAN_KEY		 = '0123456789abcdef0123456789abcdef'
uci:foreach('system', 'system', function(x) WAN_KEY = x.key end) -- quirky but it works

-- https header helpers
local FLUKSO_VERSION	 = '000'
uci:foreach('system', 'system', function(x) FLUKSO_VERSION = x.version end)

local USER_AGENT	 = 'Fluksometer v' .. FLUKSO_VERSION
local CACERT		 = flukso.daemon.cacert


--- Convert from Lua-style to c-style index.
-- @param index		Lua-style index startng at 1
-- @return 		C-style index starting at 0 
local function toc(index)
	return index - 1
end

--- Log exit status to syslog first, then exit.
-- @param code  	exit status code
-- @return		none 
local function exit(code)
	local level

	if code == 0 then
		level = 'info'
	else
		level = 'err'
	end

	nixio.syslog(level, string.format('fsync exit status: %d', code))
	os.exit(code)
end

--- Create a pair of file descriptors [fd] to the spid control fifo's.
-- @return		ctrl object containing the fd's, a line-based iterator and poll flags 
local function ctrl_init()
	local ctrl = { fdin    = nixio.open(CTRL_PATH_IN, O_RDWR_NONBLOCK),
                       fdout   = nixio.open(CTRL_PATH_OUT, O_RDWR_NONBLOCK),
                       events  = POLLIN,
                       revents = 0 }

	if ctrl.fdin == nil or ctrl.fdout == nil then
		print('Error. Unable to open the ctrl fifos.')
		print('Exiting...')
		exit(1)
	end

	-- acquire an exclusive lock on the ctrl fifos or exit
	if not (ctrl.fdin:lock('tlock') and ctrl.fdout:lock('tlock')) then
		print('Error. Detected a lock on one of the ctrl fifos.')
		print('Exiting...')
		exit(1)
	end

	ctrl.fd = ctrl.fdout -- need this entry for nixio.poll
	ctrl.line = ctrl.fdout:linesource()

	return ctrl
end

--- Close the spid control fifo's.
-- @param code  	ctrl object
-- @return		none 
local function ctrl_close(ctrl)
	ctrl.fdin:close()
	ctrl.fdout:close()
end

--- Send a command to the control fifo.
-- @param ctrl  	ctrl object
-- @param cmd		command to send
-- @return		none 
local function send(ctrl, cmd)
	while ctrl.line() do end -- flush the out fifo

	for i = 1, MAX_TRIES do
		ctrl.fdin:write(cmd .. '\n')

		local poll, errno, errmsg = nixio.poll({ ctrl }, POLL_TIMEOUT_MS)

		if poll < 0 then
			print('Error. Poll failed with error message: ' .. errmsg)

		elseif poll == 0 then
			print('Error. Poll timed out after ' .. POLL_TIMEOUT_MS .. 'ms')

		elseif poll > 0 then
			reply = ctrl.line()

			if cmd:sub(1, 1) == 's' then
				if reply == cmd then
					print(reply .. ' .. ok')
					return reply
				else
					print(reply .. ' .. nok .. should be ' .. cmd .. ' instead')
				end
			elseif cmd:sub(1, 2) == reply:sub(1, 2) then
				print(reply .. ' .. ok')
				return reply
			else
				print(reply .. ' .. nok')
			end	
		end
	end

	print(MAX_TRIES .. ' write attempts failed. Exiting ...')
	exit(2) 
end

--- Check the sensor board hardware version.
-- @param ctrl  	ctrl object
-- @return		none 
local function check_hw_version(ctrl) 
	local hw_major, hw_minor = send(ctrl, GET_HW_VERSION):match(GET_HW_VERSION_R)

	if hw_major ~= flukso.main.hw_major or hw_minor > flukso.main.hw_minor then
		print(string.format('Hardware check (major: %s, minor: %s) .. nok', hw_major, hw_minor))
		if hw_major ~= flukso.main.hw_major then
			print('Error. Major version does not match.')
		end

		if hw_minor > flukso.main.hw_minor then
			print('Error. Sensor board minor version is not supported by this package.')
		end

		if HW_CHECK_OVERRIDE then
			print('Overridden. Good luck!')
		else
			print('Use -f to override this check at your own peril.')
			exit(3)
		end
	else
		print(string.format('Hardware check (major: %s, minor: %s) .. ok', hw_major, hw_minor))
	end
end

--- Disable all sensors in the sensor board.
-- @param ctrl  	ctrl object
-- @return		none 
local function disable_all_sensors(ctrl)
	for i = 1, MAX_SENSORS do
		local cmd = string.format(SET_ENABLE, toc(i), 0)
		send(ctrl, cmd)
	end
end

--- Populate the physical (port) to logical (sensor) map on the sensor board.
-- @param ctrl  	ctrl object
-- @return		none 
local function set_phy_to_log(ctrl)
	local phy_to_log = {}

	for i = 1, MAX_SENSORS do
		if flukso[tostring(i)] ~= nil then
			if flukso[tostring(i)]['class'] == 'analog' and i > MAX_ANALOG_SENSORS then
				print(string.format('Error. Analog sensor %s should be less than or equal to max_analog_sensors (%s)', i, MAX_ANALOG_SENSORS))
				exit(4)
			end

			local ports = flukso[tostring(i)].port or {}

			for j = 1, #ports do
				if tonumber(ports[j]) > MAX_SENSORS then
					print(string.format('Error. Port numbering in sensor %s should be less than or equal to max_sensors (%s)', i, MAX_SENSORS))
					exit(5)

				else
					phy_to_log[toc(tonumber(ports[j]))] = toc(i)
				end
			end
		end
	end

	-- ports that are not in use are mapped to sensor id 0xff
	for i = 0, MAX_SENSORS - 1 do
		if not phy_to_log[i] then
			phy_to_log[i] = 0xff
		end
	end

	local cmd = SET_PHY_TO_LOG .. ' ' .. table.concat(phy_to_log, ' ', 0)
	send(ctrl, cmd)
end

--- Populate each sensor's meterconstant on the sensor board.
-- @param ctrl  	ctrl object
-- @return		none 
local function set_meterconst(ctrl)
	for i = 1, MAX_SENSORS do
		local cmd

		if flukso[tostring(i)] == nil then
			cmd = string.format(SET_METERCONST, toc(i), 0)

		elseif flukso[tostring(i)]['class'] == 'analog' then
			local voltage = tonumber(flukso[tostring(i)].voltage or "0")
			local current = tonumber(flukso[tostring(i)].current or "0")
			cmd = string.format(SET_METERCONST, toc(i), math.floor(METERCONST_FACTOR * voltage * current))

		elseif flukso[tostring(i)]['class'] == 'pulse'then
			local meterconst = tonumber(flukso[tostring(i)].constant or "0")
			cmd = string.format(SET_METERCONST, toc(i), meterconst)

		else
			cmd = string.format(SET_METERCONST, toc(i), 0)
		end

		if cmd then send(ctrl,cmd) end
	end
end

--- Reset each sensor's counter on the sensor board.
-- @param ctrl  	ctrl object
-- @return		none 
local function reset_counters(ctrl)
	for i = 1, MAX_SENSORS do
		local cmd = string.format(SET_COUNTER, toc(i), 0)
		send(ctrl, cmd)
	end

	uci:set('flukso', 'main', 'reset_counters', 0)
	uci:commit('flukso')
end

--- Activate the enabled sensors on the sensor board.
-- @param ctrl  	ctrl object
-- @return		none 
local function enable_sensors(ctrl)
	for i = 1, MAX_SENSORS do
		if flukso[tostring(i)] ~= nil and flukso[tostring(i)].enable == '1' then
			cmd = string.format(SET_ENABLE, toc(i), 1)
			send(ctrl, cmd)
		end
	end
end

--- Commit all changes on the sensor board.
-- @param ctrl  	ctrl object
-- @return		none 
local function commit(ctrl)
	send(ctrl, COMMIT)
end


--- Map /sensor/xyz endpoints to the cgi script.
-- @return		none 
local function create_symlinks()
	-- make sure /www/sensor exists
	nixio.fs.mkdirr(API_PATH)

	-- clean up old symlinks
	for symlink in nixio.fs.dir(API_PATH) do
		nixio.fs.unlink(API_PATH .. symlink)
	end

	-- generate new symlinks
	for i = 1, MAX_SENSORS do
		if flukso[tostring(i)] ~= nil
			and flukso[tostring(i)].enable == '1'
			and flukso[tostring(i)].id
			and flukso[tostring(i)].class ~= 'uart'
			then

			local sensor_id = flukso[tostring(i)].id

			if sensor_id then
				nixio.fs.symlink(CGI_SCRIPT, API_PATH .. sensor_id)
				print(string.format('ln -s %s %s%s .. ok', CGI_SCRIPT, API_PATH, sensor_id))
			end
		end
	end
end

--- Generate a new flukso.service xml file for the avahi-daemon.
-- @return		none 
local function create_avahi_config()
	avahi = { head = {}, body = {}, tail = {} }

	avahi.head[1] = '<?xml version="1.0" standalone="no"?><!--*-nxml-*-->'
	avahi.head[2] = '<!DOCTYPE service-group SYSTEM "avahi-service.dtd">'
	avahi.head[3] = '<service-group>'
	avahi.head[4] = ' <name replace-wildcards="yes">Flukso RESTful API on %h</name>'
	avahi.head[5] = '  <service>'
	avahi.head[6] = '    <type>_flukso._tcp</type>'
	avahi.head[7] = '    <port>8080</port>'

	for i = 1, MAX_SENSORS do
		if flukso[tostring(i)] ~= nil
			and flukso[tostring(i)].enable == '1'
			and flukso[tostring(i)].id
			and flukso[tostring(i)].class ~= 'uart'
			then

			avahi.body[#avahi.body + 1] = string.format('    <txt-record>id%d=%s</txt-record>' , i, flukso[tostring(i)].id)
		end
	end

	avahi.tail[1] = '    <txt-record>path=/sensor</txt-record>'
	avahi.tail[2] = '    <txt-record>version=1.0</txt-record>'
	avahi.tail[3] = '  </service>'
	avahi.tail[4] = '</service-group>'

	-- remove the old flukso.service
	nixio.fs.unlink(AVAHI_PATH)

	-- generate the new one
	fd = nixio.open(AVAHI_PATH, O_RDWR_CREAT)
	print(string.format('generating a new %s', AVAHI_PATH))

	for i = 1, #avahi.head do
		fd:write(avahi.head[i] .. '\n')
	end

	for i = 1, #avahi.body do
		fd:write(avahi.body[i] .. '\n')
	end

	for i = 1, #avahi.tail do
		fd:write(avahi.tail[i] .. '\n')
	end
end

--- POST each sensor's parameters to the /sensor/xyz endpoint
-- @return		none
local function phone_home()
	local function json_config(i) -- type(i) --> "string"
		local config = {}

		config["class"]    = flukso[i]["class"]
		config["type"]     = flukso[i]["type"]
		config["function"] = flukso[i]["function"]
		config["voltage"]  = tonumber(flukso[i]["voltage"])
		config["current"]  = tonumber(flukso[i]["current"])
		config["constant"] = tonumber(flukso[i]["constant"])
		config["enable"]   = tonumber(flukso[i]["enable"])

		if config["class"] == "analog" then
			local phase = tonumber(flukso.main.phase)

			if phase == 1 or 
			   phase == 3 and i == "1" then
				config["phase"] = phase
			end
		end

		return luci.json.encode{ config = config }
	end

	local headers = {}
	headers['Content-Type'] = 'application/json'
	headers['X-Version'] = '1.0'
	headers['User-Agent'] = USER_AGENT

	local options = {}
	options.sndtimeo = 5
	options.rcvtimeo = 5

	options.tls_context_set_verify = 'peer'
	options.cacert = CACERT
	options.method  = 'POST'
	options.headers = headers

	local http_persist = httpclient.create_persistent()

	for i = 1, MAX_SENSORS do
		if flukso[tostring(i)] ~= nil and flukso[tostring(i)].id then
			local sensor_id = flukso[tostring(i)].id

			if i ~= MAX_SENSORS then
				options.headers['Connection'] = 'keep-alive'
			else
				options.headers['Connection'] = 'close'
			end

			options.body = json_config(tostring(i))
			options.headers['Content-Length'] = tostring(#options.body)

			local hash = nixio.crypto.hmac('sha1', WAN_KEY)
			hash:update(options.body)
			options.headers['X-Digest'] = hash:final()

			local url = WAN_BASE_URL .. sensor_id
			local response, code, call_info = http_persist(url, options)

			local level

			if code == 200 or code == 204 then
				level = 'info'
			else
				level = 'err'
			end

			nixio.syslog(level, string.format('%s %s: %s', options.method, url, code))

			-- if available, send additional error info to the syslog
			if type(call_info) == 'string' then
				nixio.syslog('err', call_info)
			elseif type(call_info) == 'table'  then
				local auth_error = call_info.headers['WWW-Authenticate']

				if auth_error then
					nixio.syslog('err', string.format('WWW-Authenticate: %s', auth_error))
				end
			end
		end
	end
end

-- open the connection to the syslog deamon, specifying our identity
nixio.openlog('fsync', 'pid')

-- sync config to sensor board
local ctrl = ctrl_init()

check_hw_version(ctrl)
disable_all_sensors(ctrl)
set_phy_to_log(ctrl)
set_meterconst(ctrl)

if RESET_COUNTERS then
	reset_counters(ctrl)
end

enable_sensors(ctrl)
commit(ctrl)

ctrl_close(ctrl)


-- sync config locally
create_symlinks()
create_avahi_config()


-- sync config with the server
if SYNC_TO_SERVER then
	phone_home()
end


print(arg[0] .. ' completed successfully. Bye!')
exit(0)