mirror of
https://git.openwrt.org/project/luci.git
synced 2025-01-18 23:45:02 +00:00
0a6a2f1fc5
If UCI store values as a list instead of an option, conversation for the collectd configuration is not work correctly. This is due to the use of a DynamicList type element in the UI (for example, for the RRATimespans field). For this function argument val receives as array instead of a string, so no additional conversion is required. Signed-off-by: Aleksey Kolosov <softovick@gmail.com>
312 lines
6.6 KiB
Text
Executable file
312 lines
6.6 KiB
Text
Executable file
#!/usr/bin/env ucode
|
|
/*
|
|
Luci statistics - collectd configuration generator
|
|
(c) 2008-2022 Jo-Philipp Wich <jo@mein.io>
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
import { lsdir, open } from 'fs';
|
|
import { cursor } from 'uci';
|
|
|
|
const uci = cursor();
|
|
const sections = uci.get_all('luci_statistics');
|
|
|
|
const plugins = {
|
|
collectd: [
|
|
[ 'BaseDir', 'Include', 'PIDFile', 'PluginDir', 'TypesDB', 'Interval', 'ReadThreads', 'Hostname' ],
|
|
[],
|
|
[]
|
|
],
|
|
logfile: [
|
|
[ 'LogLevel', 'File' ],
|
|
[ 'Timestamp' ],
|
|
[]
|
|
],
|
|
};
|
|
|
|
function parse_units(ustr) {
|
|
let val = 0;
|
|
|
|
// unit map
|
|
const map = {
|
|
y : 60 * 60 * 24 * 366,
|
|
m : 60 * 60 * 24 * 31,
|
|
w : 60 * 60 * 24 * 7,
|
|
d : 60 * 60 * 24,
|
|
h : 60 * 60,
|
|
min: 60
|
|
};
|
|
|
|
// parse input string
|
|
for (let spec in match(lc(ustr), /([0-9.]+)([a-z]*)/g)) {
|
|
let num = +spec[1];
|
|
let mul = map[spec[2]] ?? map[substr(spec[2], 0, 1)] ?? 1;
|
|
|
|
val += num * mul;
|
|
}
|
|
|
|
return int(val);
|
|
}
|
|
|
|
const preprocess = {
|
|
RRATimespans: function(val) {
|
|
return join(' ', map(type(val) == 'array' ? val : split(val, /\s+/), parse_units));
|
|
}
|
|
};
|
|
|
|
|
|
function _bool(s, n, nopad) {
|
|
if (s == '1')
|
|
return `${nopad ? '' : '\t'}${n} true\n`;
|
|
|
|
if (s == '0')
|
|
return `${nopad ? '' : '\t'}${n} false\n`;
|
|
|
|
return '';
|
|
}
|
|
|
|
function _string(s, n, nopad) {
|
|
if (s) {
|
|
if (n == 'Port' || n == 'Irq' || match(s, /[^0-9]/)) {
|
|
if (!match(s, /[^\w]/) && n != 'Port' && n != 'Irq')
|
|
return `${nopad ? '' : '\t'}${n} ${trim(s)}\n`;
|
|
else
|
|
return `${nopad ? '' : '\t'}${n} "${trim(s)}"\n`;
|
|
}
|
|
else {
|
|
return `${nopad ? '' : '\t'}${n} ${trim(s)}\n`;
|
|
}
|
|
}
|
|
|
|
return '';
|
|
}
|
|
|
|
function _expand(s, n, nopad) {
|
|
let str = "";
|
|
|
|
if (type(s) == 'string') {
|
|
for (let v in split(s, /\s+/))
|
|
str += _string(v, n, nopad);
|
|
}
|
|
else if (type(s) == 'array') {
|
|
for (let v in s)
|
|
str += _string(v, n, nopad);
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function _list_expand(c, l, nopad) {
|
|
let str = '';
|
|
|
|
for (let n in l) {
|
|
if (c[n]) {
|
|
if (preprocess[n])
|
|
c[n] = preprocess[n](c[n]);
|
|
|
|
let m = match(n, /^(\w+)ses$/);
|
|
let k;
|
|
|
|
if (m)
|
|
k = `${m[1]}s`;
|
|
else
|
|
k = replace(n, /^(\w+)s$/, '$1');
|
|
|
|
str += _expand(c[n], k, nopad);
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
|
|
function config_generic(c, singles, bools, lists, nopad) {
|
|
let str = '';
|
|
|
|
if (c) {
|
|
for (let key in singles) {
|
|
if (preprocess[key])
|
|
c[key] = preprocess[key](c[key]);
|
|
|
|
str += _string(c[key], key, nopad);
|
|
}
|
|
|
|
for (let key in bools) {
|
|
if (preprocess[key])
|
|
c[key] = preprocess[key](c[key]);
|
|
|
|
str += _bool(c[key], key, nopad);
|
|
}
|
|
|
|
if (lists)
|
|
str += _list_expand(c, lists, nopad);
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function config_exec(c) {
|
|
let str = "";
|
|
|
|
for (let k, s in sections) {
|
|
for (let key, type in { Exec: 'collectd_exec_input', NotificationExec: 'collectd_exec_notify' }) {
|
|
if (s['.type'] == type) {
|
|
let cmd = replace(trim(s.cmdline), /\s+/g, '" "');
|
|
let user = s.cmduser ?? 'nobody';
|
|
let group = s.cmdgroup;
|
|
|
|
if (cmd)
|
|
str += `\t${key} "${user}${group ? `:${group}` : ''}" "${cmd}"\n`;
|
|
}
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function config_curl(c) {
|
|
let str = "";
|
|
|
|
for (let k, s in sections) {
|
|
if (s['.type'] == 'collectd_curl_page') {
|
|
str += `\t<Page "${s.name}">\n`
|
|
+ `\t\tURL "${s.url}"\n`
|
|
+ `\t\tMeasureResponseTime true\n`
|
|
+ `\t</Page>\n`;
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function config_iptables(c) {
|
|
let str = "";
|
|
|
|
for (let k, s in sections) {
|
|
for (let type, verb in { collectd_iptables_match: 'Chain', collectd_iptables_match6: 'Chain6' }) {
|
|
if (s['.type'] == type) {
|
|
let tname = `${s.table}`;
|
|
let chain = `${s.chain}`;
|
|
|
|
if (match(tname, /^\S+$/) && match(chain, /^\S+$/)) {
|
|
str += `\t${verb} "${tname}" "${chain}"`;
|
|
|
|
let rule = `${s.rule}`;
|
|
|
|
if (match(rule, /^\S+$/)) {
|
|
str += ` "${rule}"`;
|
|
|
|
let name = `${s.name}`;
|
|
|
|
if (match(name, /^\S+$/))
|
|
str += ` "${name}"`;
|
|
}
|
|
|
|
}
|
|
str += '\n';
|
|
}
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function config_network(c) {
|
|
let str = '';
|
|
|
|
for (let k, s in sections) {
|
|
for (let key, type in { Listen: 'collectd_network_listen', Server: 'collectd_network_server' }) {
|
|
if (s['.type'] == type && s.host) {
|
|
if (s.port)
|
|
str += `\t${key} "${s.host}" "${s.port}"\n`;
|
|
else
|
|
str += `\t${key} "${s.host}"\n`;
|
|
}
|
|
}
|
|
}
|
|
|
|
return str
|
|
+ _string(c.MaxPacketSize, 'MaxPacketSize')
|
|
+ _string(c.TimeToLive, 'TimeToLive')
|
|
+ _bool(c.Forward, 'Forward')
|
|
+ _bool(c.ReportStats, 'ReportStats')
|
|
;
|
|
}
|
|
|
|
function config_mqtt(c) {
|
|
let str = "";
|
|
|
|
for (let k, s in sections) {
|
|
if (s['.type'] === 'collectd_mqtt_block') {
|
|
const isPublish = s['blocktype'] === 'Publish';
|
|
|
|
str += isPublish ? `\t<Publish "${s.name}">\n` : `\t<Subscribe "${s.name}">\n`;
|
|
|
|
str += `\t\tHost "${s.Host}"\n`;
|
|
str += s['Port'] ? `\t\tPort ${s.Port}\n` : '';
|
|
str += s['User'] ? `\t\tUser "${s.User}"\n` : '';
|
|
str += s['Password'] ? `\t\tPassword "${s.Password}"\n` : '';
|
|
str += s['Qos'] ? `\t\tQos ${s.Qos}\n` : '';
|
|
str += s['Prefix'] ? `\t\tPrefix ${s.Prefix}\n` : '';
|
|
str += s['Retain'] ? `\t\tRetain ${s.Retain}\n` : '';
|
|
str += s['StoreRates'] ? `\t\tRetain ${s.StoreRates}\n` : '';
|
|
str += s['CleanSession'] ? `\t\tRetain ${s.CleanSession}\n` : '';
|
|
str += s['Topic'] ? `\t\tTopic "${s.Topic}"\n` : '';
|
|
|
|
str += isPublish ? `\t</Publish>\n` : `\t</Subscribe>\n`;
|
|
}
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
function section(plugin) {
|
|
let config = sections[`collectd_${plugin}`] ?? sections.collectd;
|
|
|
|
if (config && (plugin == 'collectd' || config.enable == '1')) {
|
|
let params;
|
|
|
|
if (type(plugins[plugin]) == 'function')
|
|
params = plugins[plugin](config);
|
|
else
|
|
params = config_generic(config, ...plugins[plugin], plugin == 'collectd');
|
|
|
|
if (plugin != 'collectd')
|
|
print(`LoadPlugin ${plugin}\n${length(params) ? `<Plugin ${plugin}>\n${params}</Plugin>\n` : ''}\n`);
|
|
else
|
|
print(`${params ?? ''}\n`);
|
|
}
|
|
}
|
|
|
|
|
|
let plugin_dir = '/usr/share/luci/statistics/plugins';
|
|
|
|
for (let filename in lsdir(plugin_dir)) {
|
|
let name = replace(filename, /\.json$/, '');
|
|
|
|
switch (name) {
|
|
case 'exec': plugins[name] = config_exec; break;
|
|
case 'iptables': plugins[name] = config_iptables; break;
|
|
case 'curl': plugins[name] = config_curl; break;
|
|
case 'network': plugins[name] = config_network; break;
|
|
case 'mqtt': plugins[name] = config_mqtt; break;
|
|
default:
|
|
plugins[name] = json(open(`${plugin_dir}/${filename}`))?.legend;
|
|
}
|
|
}
|
|
|
|
|
|
section('collectd');
|
|
section('logfile');
|
|
|
|
for (let plugin in plugins)
|
|
if (plugin != 'collectd' && plugin != 'logfile')
|
|
section(plugin);
|