rest
This commit is contained in:
parent
dbf6214dd0
commit
087a6afacf
@ -23,10 +23,14 @@
|
||||
-- Prefixes used within the mesh.
|
||||
-- prefix6 is required, prefix4 can be omitted if next_node.ip4
|
||||
-- is not set.
|
||||
prefix4 = '10.0.0.0/20',
|
||||
prefix6 = 'fdff:cafe:cafe:cafe::/64',
|
||||
|
||||
-- [olsr] prefix configuration
|
||||
node_prefix6 = 'fdff:cafe:cafe:cafe::/64',
|
||||
node_prefix4 = '10.12.0.0/16',
|
||||
node_prefix4_range = 24,
|
||||
node_prefix4_temporary = true,
|
||||
|
||||
-- Timezone of your community.
|
||||
-- See https://openwrt.org/docs/guide-user/base-system/system_configuration#time_zones
|
||||
@ -90,8 +94,12 @@
|
||||
vxlan = true,
|
||||
-- [olsr] OLSR configuration with v1/v2 parallel mesh
|
||||
olsrd = {
|
||||
v1 = {
|
||||
enable = true,
|
||||
},
|
||||
v2 = {
|
||||
enable = true,
|
||||
ip6_exclusive_mode = true,
|
||||
}
|
||||
},
|
||||
},
|
||||
|
76
docs/features/olsrd.rst
Normal file
76
docs/features/olsrd.rst
Normal file
@ -0,0 +1,76 @@
|
||||
OLSRD
|
||||
===========
|
||||
|
||||
Gluon supports OLSRD, both version 1 and 2 in the following modes:
|
||||
|
||||
- olsrd
|
||||
- v4 only
|
||||
- olsrd2
|
||||
- v4 only
|
||||
- v6 only
|
||||
- dual-stack
|
||||
|
||||
olsrdv1 support is intended mostly for migration purposes
|
||||
and as such v1 IPv6 support is not going to be added
|
||||
|
||||
Configuration
|
||||
-------------
|
||||
|
||||
The LAN will automatically be determined by the specified prefix and prefix6
|
||||
|
||||
The following options exist
|
||||
|
||||
.. code-block:: lua
|
||||
{
|
||||
mesh {
|
||||
olsrd = {
|
||||
v1 = {
|
||||
-- Enable v1
|
||||
-- enable = true,
|
||||
|
||||
-- Set additional olsrd configuration
|
||||
-- config = {
|
||||
-- DebugLevel = 0,
|
||||
-- IpVersion = 4,
|
||||
-- AllowNoInt = yes,
|
||||
-- },
|
||||
},
|
||||
v2 = {
|
||||
-- Enable v2
|
||||
enable = true,
|
||||
|
||||
-- Make v2 IPv6 exclusive
|
||||
-- ip6_exclusive_mode = true,
|
||||
|
||||
-- Make v2 IPv4 exclusive (useful for v1 co-existence)
|
||||
-- ip4_exclusive_mode = true,
|
||||
|
||||
-- Set additional olsrd2 configuration
|
||||
-- config = {
|
||||
--
|
||||
-- }
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Static IP managment
|
||||
-------------------
|
||||
This feature is enabled as part of olsrd support
|
||||
|
||||
Static IP managment has the following options
|
||||
|
||||
.. code-block:: lua
|
||||
{
|
||||
-- Auto-assign addresses from an IPv4 range
|
||||
node_prefix4 = '10.12.23.0/16',
|
||||
node_prefix4_range = 24, -- range of node_prefix4 that should be randomized with mac
|
||||
node_prefix4_temporary = true, -- (def: true) flag to indicate whether or not this is a temporary range that will need manual change for permanent assignments or not
|
||||
|
||||
-- Auto-assign addresses from an IPv6 range
|
||||
node_prefix6 = 'fdff:cafe:cafe:cafe:23::/64',
|
||||
node_prefix6_range = 64, -- (def: 64) range of node_prefix6 that should be randomized with mac
|
||||
node_prefix6_temporary = true, -- (def: false) flag to indicate whether or not this is a temporary range that will need manual change for permanent assignments or not
|
||||
}
|
||||
|
||||
Note that these addresses are intended to be temporary (TODO: should they or would dynamic4 and dynamic4IsTmp be better?)
|
@ -28,6 +28,7 @@ Several Freifunk communities in Germany use Gluon as the foundation of their Fre
|
||||
features/dns-forwarder
|
||||
features/monitoring
|
||||
features/multidomain
|
||||
features/olsrd
|
||||
features/authorized-keys
|
||||
features/roles
|
||||
features/vpn
|
||||
|
@ -46,6 +46,9 @@ when(_'status-page' and _'mesh-batman-adv-15', {
|
||||
'gluon-status-page-mesh-batman-adv',
|
||||
})
|
||||
|
||||
when(_'status-page' and _'mesh-olsrd', {
|
||||
'gluon-status-page-mesh-olsrd'
|
||||
})
|
||||
|
||||
when(_'mesh-babel', {
|
||||
'gluon-radvd',
|
||||
|
@ -3,6 +3,8 @@ include $(TOPDIR)/rules.mk
|
||||
PKG_NAME:=gluon-mesh-olsrd
|
||||
PKG_VERSION=1
|
||||
|
||||
PKG_BUILD_DEPENDS += libjson-c
|
||||
|
||||
include ../gluon.mk
|
||||
|
||||
define Package/gluon-mesh-olsrd
|
||||
@ -11,8 +13,16 @@ define Package/gluon-mesh-olsrd
|
||||
+gluon-core \
|
||||
+kmod-macvlan \
|
||||
@IPV6 \
|
||||
+olsrd \
|
||||
+oonf-olsrd2 \
|
||||
+firewall \
|
||||
+libgluonutil \
|
||||
+libjson-c \
|
||||
+libubox +libuclient \
|
||||
+olsrd-mod-jsoninfo \
|
||||
+olsrd-mod-httpinfo \
|
||||
+olsrd-mod-txtinfo \
|
||||
+liblua \
|
||||
+ip-full \
|
||||
+gluon-layer3-common \
|
||||
+gluon-l3roamd \
|
||||
@ -21,5 +31,26 @@ define Package/gluon-mesh-olsrd
|
||||
PROVIDES:=gluon-mesh-provider
|
||||
endef
|
||||
|
||||
define Package/gluon-mesh-olsrd/install
|
||||
$(Gluon/Build/Install)
|
||||
|
||||
$(INSTALL_DIR) $(1)/usr/lib/lua/gluon
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/olsrd.so $(1)/usr/lib/lua/gluon/
|
||||
$(INSTALL_DIR) $(1)/usr/lib/
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/libolsrdhelper.so $(1)/usr/lib/
|
||||
endef
|
||||
|
||||
define Build/InstallDev
|
||||
$(INSTALL_DIR) $(1)/usr/lib/lua/gluon
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/olsrd.so $(1)/usr/lib/lua/gluon/
|
||||
$(INSTALL_DIR) $(1)/usr/lib/
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/libolsrdhelper.so $(1)/usr/lib/
|
||||
|
||||
$(INSTALL_DIR) $(1)/usr/include/gluon-mesh-olsrd
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/libolsrdhelper.h $(1)/usr/include/gluon-mesh-olsrd/
|
||||
$(INSTALL_DIR) $(1)/usr/lib/pkgconfig/
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/libolsrdhelper.pc $(1)/usr/lib/pkgconfig/
|
||||
endef
|
||||
|
||||
|
||||
$(eval $(call BuildPackageGluon,gluon-mesh-olsrd))
|
||||
|
@ -1,5 +1,20 @@
|
||||
need_string_match(in_domain({'next_node', 'ip6'}), '^[%x:]+$', false)
|
||||
need_string_match(in_domain({'next_node', 'ip4'}), '^%d+.%d+.%d+.%d+$', false)
|
||||
|
||||
need_boolean({'mesh', 'olsrd', 'v2', 'enable'}, false)
|
||||
need_table({'mesh', 'olsrd', 'v2', 'config'}, nil, false)
|
||||
if need_boolean({'mesh', 'olsrd', 'v1_6', 'enable'}, false) then
|
||||
need_table({'mesh', 'olsrd', 'v1_6', 'config'}, nil, false)
|
||||
end
|
||||
|
||||
if need_boolean({'mesh', 'olsrd', 'v1_4', 'enable'}, false) then
|
||||
need_table({'mesh', 'olsrd', 'v1_4', 'config'}, nil, false)
|
||||
end
|
||||
|
||||
if need_boolean({'mesh', 'olsrd', 'v2', 'enable'}, false) then
|
||||
need_table({'mesh', 'olsrd', 'v2', 'config'}, nil, false)
|
||||
need_boolean({'mesh', 'olsrd', 'v2', 'ip6_exclusive_mode'}, false)
|
||||
need_boolean({'mesh', 'olsrd', 'v2', 'ip4_exclusive_mode'}, false)
|
||||
if need_boolean({'mesh', 'olsrd', 'v2', 'ip4_exclusive_mode'}, false) and need_boolean({'mesh', 'olsrd', 'v2', 'ip6_exclusive_mode'}, false) then
|
||||
-- FIXME: we could check the value but idk how to do that. basically both options are xor.
|
||||
error('you cant enable both olsrv2 ip4 and ip6 exclusive mode')
|
||||
end
|
||||
end
|
||||
|
@ -7,4 +7,6 @@ reload_running() {
|
||||
fi
|
||||
}
|
||||
|
||||
reload_running olsrd
|
||||
reload_running olsrd2
|
||||
reload_running olsrd6
|
||||
|
@ -7,4 +7,6 @@ reload_running() {
|
||||
fi
|
||||
}
|
||||
|
||||
reload_running olsrd
|
||||
reload_running olsrd2
|
||||
reload_running olsrd6
|
||||
|
@ -3,5 +3,7 @@
|
||||
. /lib/gluon/autoupdater/lib.sh
|
||||
|
||||
|
||||
start_enabled olsrd
|
||||
start_enabled olsrd2
|
||||
start_enabled olsrd6
|
||||
wifi up
|
||||
|
@ -3,5 +3,7 @@
|
||||
. /lib/gluon/autoupdater/lib.sh
|
||||
|
||||
|
||||
stop olsrd
|
||||
stop olsrd2
|
||||
stop olsrd6
|
||||
wifi down
|
||||
|
@ -4,6 +4,8 @@ local uci = require('simple-uci').cursor()
|
||||
local site = require 'gluon.site'
|
||||
local util = require 'gluon.util'
|
||||
local wireless = require 'gluon.wireless'
|
||||
local sysconfig = require 'gluon.sysconfig'
|
||||
local olsrd = require 'gluon.olsrd'
|
||||
|
||||
local mesh_interfaces = util.get_role_interfaces(uci, 'mesh')
|
||||
local uplink_interfaces = util.get_role_interfaces(uci, 'uplink')
|
||||
@ -22,6 +24,10 @@ for _, iface in ipairs(mesh_interfaces) do
|
||||
end
|
||||
end
|
||||
|
||||
-- NOTE: this is a gigantic hack that overrides the gluon rules because they rely on gluon_proto
|
||||
-- this will need a rework once olsr and gluon_proto go together
|
||||
-- (i suspect it will pretty much just take an "ip addr add <v4> <intf>" and the same for v6 for the statics to work)
|
||||
-- (we could prob also do multiple v6 then on loopback?)
|
||||
|
||||
local intf = {
|
||||
wired_mesh = {},
|
||||
@ -41,8 +47,13 @@ end
|
||||
wireless.foreach_radio(uci, function(radio, _, _)
|
||||
local radio_name = radio['.name']
|
||||
table.insert(intf.radio_mesh, 'mesh_' .. radio_name)
|
||||
if uci:get('network', 'ibss_' .. radio_name, 'proto') then
|
||||
table.insert(intf.radio_mesh, 'ibss_' .. radio_name)
|
||||
end
|
||||
end)
|
||||
|
||||
local vpn_mesh = false
|
||||
|
||||
if pcall(function() require 'gluon.mesh-vpn' end) then
|
||||
local vpn_core = require 'gluon.mesh-vpn'
|
||||
|
||||
@ -50,9 +61,26 @@ if pcall(function() require 'gluon.mesh-vpn' end) then
|
||||
-- mesh_vpn is a interface that has the right ifname
|
||||
-- we can't use mesh-vpn (dash instead of underscore) since it's not a uci interface
|
||||
table.insert(intf.vpn_mesh, 'mesh_vpn')
|
||||
vpn_mesh = true
|
||||
end
|
||||
end
|
||||
|
||||
local function has_role_mesh(ifname)
|
||||
local roles = uci:get('gluon', 'iface_' .. ifname, 'role')
|
||||
|
||||
local has_role = false
|
||||
|
||||
if roles then
|
||||
for _, r in ipairs(roles) do
|
||||
if r == 'mesh' then
|
||||
has_role = true
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
return has_role
|
||||
end
|
||||
|
||||
table.insert(intf.wired_mesh, 'loopback')
|
||||
|
||||
local has_uplink_mesh = false
|
||||
@ -85,6 +113,130 @@ if has_other_mesh then
|
||||
table.insert(intf.wired_mesh, 'mesh_other')
|
||||
end
|
||||
|
||||
-- for _,i in pairs(mesh_interfaces) do
|
||||
-- local name = i
|
||||
--
|
||||
-- if util.contains(uplink_interfaces, i) then
|
||||
-- name = 'uplink'
|
||||
-- i = 'br-wan'
|
||||
-- elseif util.contains(client_interfaces, i) then
|
||||
-- name = 'client'
|
||||
-- i = 'br-client'
|
||||
-- else
|
||||
-- name = 'other'
|
||||
-- i = 'br-mesh_other'
|
||||
-- end
|
||||
--
|
||||
-- local iname = 'olsr_mesh_' .. name
|
||||
--
|
||||
-- if not util.contains(intf.wired_mesh, iname) then
|
||||
-- uci:section('network', 'device', 'm_' .. name, {
|
||||
-- name = 'm_' .. name,
|
||||
-- type = 'macvlan',
|
||||
-- ifname = i,
|
||||
-- })
|
||||
--
|
||||
-- uci:section('network', 'interface', iname, {
|
||||
-- ifname = 'm_' .. name,
|
||||
-- })
|
||||
--
|
||||
-- table.insert(intf.wired_mesh, iname)
|
||||
-- end
|
||||
-- end
|
||||
|
||||
for _, _olsr in ipairs({
|
||||
{ 'v1_4', 4, 698, 'olsrd' },
|
||||
{ 'v1_6', 6, 698, 'olsrd6' },
|
||||
}) do
|
||||
local _olsr_key, _olsr_ip, _olsr_port, _olsr_name = unpack(_olsr)
|
||||
local cfg = site.mesh.olsrd[_olsr_key]
|
||||
local uci_name = _olsr_name
|
||||
|
||||
uci:delete_all(uci_name, 'Interface')
|
||||
uci:delete_all(uci_name, 'LoadPlugin')
|
||||
|
||||
if cfg ~= nil and cfg.enable(false) then
|
||||
os.execute('/etc/init.d/' .. _olsr_name .. ' enable')
|
||||
|
||||
-- set config
|
||||
local olsrConfig = {
|
||||
IpVersion = _olsr_ip,
|
||||
FIBMetric = 'flat',
|
||||
LinkQualityLevel = '2',
|
||||
OlsrPort = _olsr_port,
|
||||
Willingness = '3'
|
||||
}
|
||||
|
||||
local extraConf = cfg.config()
|
||||
if extraConf then
|
||||
for k, v in pairs(extraConf) do
|
||||
olsrConfig[k] = extraConf[k]
|
||||
end
|
||||
end
|
||||
|
||||
uci:delete_all(uci_name, 'olsrd')
|
||||
uci:section(uci_name, 'olsrd', nil, olsrConfig)
|
||||
|
||||
-- add jsoninfo
|
||||
uci:section(uci_name, 'LoadPlugin', 'jsoninfo', {
|
||||
library = olsrd.find_module_version('olsrd_jsoninfo'),
|
||||
accept = '127.0.0.1',
|
||||
})
|
||||
-- add txtinfo
|
||||
uci:section(uci_name, 'LoadPlugin', 'txtinfo', {
|
||||
library = olsrd.find_module_version('olsrd_txtinfo'),
|
||||
accept = '127.0.0.1',
|
||||
})
|
||||
-- add httpinfo
|
||||
uci:section(uci_name, 'LoadPlugin', 'httpinfo', {
|
||||
library = olsrd.find_module_version('olsrd_httpinfo'),
|
||||
Net = {'127.0.0.1'},
|
||||
})
|
||||
|
||||
if #intf.wired_mesh then
|
||||
uci:section(uci_name, 'Interface', 'wired_mesh', {
|
||||
interface = intf.wired_mesh,
|
||||
Mode = 'mesh', -- should be mode 'ether' but that appears broken for some reason
|
||||
})
|
||||
end
|
||||
|
||||
if #intf.vpn_mesh then
|
||||
uci:section(uci_name, 'Interface', 'vpn_mesh', {
|
||||
interface = intf.vpn_mesh,
|
||||
Mode = 'mesh',
|
||||
})
|
||||
end
|
||||
|
||||
if #intf.radio_mesh then
|
||||
uci:section(uci_name, 'Interface', 'radio_mesh', {
|
||||
interface = intf.radio_mesh,
|
||||
Mode = 'mesh',
|
||||
})
|
||||
end
|
||||
|
||||
uci:section('firewall', 'rule', 'allow_olsr' .. _olsr_ip .. '_mesh', {
|
||||
-- src = 'mesh',
|
||||
-- HACK: this is a temporary hack because firewall doesn't want to work
|
||||
src = '*',
|
||||
dest_port = _olsr_port,
|
||||
proto = 'udp',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:section('firewall', 'rule', 'allow_olsr' .. _olsr_ip .. '_wired_mesh', {
|
||||
src = 'wired_mesh',
|
||||
dest_port = _olsr_port,
|
||||
proto = 'udp',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
else
|
||||
-- site.mesh.olsrd.v1/v1_4.enable false
|
||||
os.execute('/etc/init.d/' .. _olsr_name .. ' disable')
|
||||
uci:delete('firewall', 'allow_' .. _olsr_name .. '_mesh')
|
||||
uci:delete('firewall', 'allow_' .. _olsr_name .. '_wired_mesh')
|
||||
end
|
||||
end
|
||||
|
||||
uci:delete_all('olsrd2', 'interface')
|
||||
|
||||
if site.mesh.olsrd.v2() ~= nil and site.mesh.olsrd.v2.enable(false) then
|
||||
@ -126,6 +278,20 @@ if site.mesh.olsrd.v2() ~= nil and site.mesh.olsrd.v2.enable(false) then
|
||||
lan = cfg.lan()
|
||||
end
|
||||
|
||||
if cfg.ip6_exclusive_mode(false) then
|
||||
table.insert(addrs, '-0.0.0.0/0')
|
||||
else
|
||||
table.insert(addrs, '-127.0.0.1/8')
|
||||
end
|
||||
if cfg.ip4_exclusive_mode(false) then
|
||||
table.insert(addrs, '-::/0')
|
||||
else
|
||||
table.insert(addrs, '-::1/128')
|
||||
end
|
||||
|
||||
if site.prefix4() and not cfg.ip6_exclusive_mode(false) then
|
||||
table.insert(lan, site.prefix4())
|
||||
end
|
||||
if site.prefix6() and not cfg.ip4_exclusive_mode(false) then
|
||||
table.insert(lan, site.prefix6())
|
||||
end
|
||||
@ -182,6 +348,22 @@ else
|
||||
uci:delete('firewall', 'allow_olsr2_mesh')
|
||||
uci:delete('firewall', 'allow_olsr2_wired_mesh')
|
||||
end
|
||||
|
||||
uci:section('firewall', 'rule', 'hack_ssh', {
|
||||
src = '*',
|
||||
dest_port = '22',
|
||||
proto = 'tcp',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:section('firewall', 'rule', 'hack_http', {
|
||||
src = '*',
|
||||
dest_port = '80',
|
||||
proto = 'tcp',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:save('olsrd')
|
||||
uci:save('olsrd2')
|
||||
uci:save('firewall')
|
||||
uci:save('network')
|
||||
|
@ -0,0 +1,79 @@
|
||||
#!/usr/bin/lua
|
||||
|
||||
local uci = require('simple-uci').cursor()
|
||||
local site = require 'gluon.site'
|
||||
local util = require 'gluon.util'
|
||||
local wireless = require 'gluon.wireless'
|
||||
|
||||
networks = uci:get_list('firewall', 'drop', 'network')
|
||||
util.remove_from_set(networks, 'client')
|
||||
uci:set_list('firewall', 'drop', 'network', networks)
|
||||
|
||||
uci:section('firewall', 'rule', 'allow_mesh_ping', {
|
||||
name = 'Allow mesh ping',
|
||||
src = 'mesh',
|
||||
proto = 'icmp',
|
||||
icmp_type = {
|
||||
'echo-request'
|
||||
},
|
||||
family = 'ipv4',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:section('firewall', 'rule', 'allow_mesh_mld', {
|
||||
name = 'Allow mesh MLD',
|
||||
src = 'mesh',
|
||||
proto = 'icmp',
|
||||
src_ip = 'fe80::/10',
|
||||
icmp_type = {
|
||||
'130/0',
|
||||
'131/0',
|
||||
'132/0',
|
||||
'143/0',
|
||||
},
|
||||
family = 'ipv6',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:section('firewall', 'rule', 'allow_mesh_icmpv6_input', {
|
||||
name = 'Allow mesh icmpv6 input',
|
||||
src = 'mesh',
|
||||
proto = 'icmp',
|
||||
icmp_type = {
|
||||
'echo-request',
|
||||
'echo-reply',
|
||||
'destination-unreachable',
|
||||
'packet-too-big',
|
||||
'time-exceeded',
|
||||
'bad-header',
|
||||
'unknown-header-type',
|
||||
'router-solicitation',
|
||||
'neighbour-solicitation',
|
||||
'router-advertisement',
|
||||
'neighbour-advertisement'
|
||||
},
|
||||
limit = '1000/sec',
|
||||
family = 'ipv6',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:section('firewall', 'rule', 'allow_mesh_icmpv6_forward', {
|
||||
name = 'Allow mesh icmpv6 forward',
|
||||
src = 'mesh',
|
||||
dest = '*',
|
||||
proto = 'icmp',
|
||||
icmp_type = {
|
||||
'echo-request',
|
||||
'echo-reply',
|
||||
'destination-unreachable',
|
||||
'packet-too-big',
|
||||
'time-exceeded',
|
||||
'bad-header',
|
||||
'unknown-header-type'
|
||||
},
|
||||
limit = '1000/sec',
|
||||
family = 'ipv6',
|
||||
target = 'ACCEPT',
|
||||
})
|
||||
|
||||
uci:save('firewall')
|
139
package/gluon-mesh-olsrd/luasrc/usr/bin/olsrd-cli
Executable file
139
package/gluon-mesh-olsrd/luasrc/usr/bin/olsrd-cli
Executable file
@ -0,0 +1,139 @@
|
||||
#!/usr/bin/lua
|
||||
|
||||
local uci = require('simple-uci').cursor()
|
||||
local site = require 'gluon.site'
|
||||
local util = require 'gluon.util'
|
||||
local wireless = require 'gluon.wireless'
|
||||
local sysconfig = require 'gluon.sysconfig'
|
||||
local util = require 'gluon.util'
|
||||
local olsrd = require 'gluon.olsrd'
|
||||
|
||||
-- Utils
|
||||
|
||||
function printf(...)
|
||||
print(string.format(...))
|
||||
end
|
||||
|
||||
-- Print contents of `tbl`, with indentation.
|
||||
-- `indent` sets the initial level of indentation.
|
||||
-- src https://gist.github.com/xytis/5361405
|
||||
function tprint (tbl, indent)
|
||||
if not indent then indent = 0 end
|
||||
for k, v in pairs(tbl) do
|
||||
formatting = string.rep(' ', indent) .. k .. ': '
|
||||
if type(v) == 'table' then
|
||||
print(formatting)
|
||||
tprint(v, indent + 1)
|
||||
else
|
||||
print(formatting .. tostring(v))
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
-- src https://stackoverflow.com/a/24823383/3990041
|
||||
function table.slice(tbl, first, last, step)
|
||||
local sliced = {}
|
||||
|
||||
for i = first or 1, last or #tbl, step or 1 do
|
||||
sliced[#sliced+1] = tbl[i]
|
||||
end
|
||||
|
||||
return sliced
|
||||
end
|
||||
|
||||
-- CLI lib
|
||||
|
||||
function exec_cmd(args, sub)
|
||||
if sub[args[1]] == nil then
|
||||
return cmd_err('does not exist')
|
||||
else
|
||||
local cmd = sub[args[1]]
|
||||
if cmd[3] ~= nil and #args > 1 then
|
||||
exec_cmd(table.slice(args, 2), cmd[3])
|
||||
else
|
||||
cmd[1](unpack(table.slice(args, 2)))
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
function list_cmd(level, sub)
|
||||
for key, cmd in pairs(sub) do
|
||||
printf('%s%s: %s', string.rep(' ', level), key, cmd[2])
|
||||
if cmd[3] ~= nil then
|
||||
list_cmd(level + 1, cmd[3])
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
function show_help()
|
||||
printf('Usage: %s <command>', arg[0])
|
||||
list_cmd(1, sub)
|
||||
end
|
||||
|
||||
function cmd_err(msg, no_show_help)
|
||||
-- since argv0 is at... well... 0... even though this is lua...
|
||||
--- ...slice just returns arg without argv0 as the for starts at 1
|
||||
printf('Error: Command "%s" %s', table.concat(table.slice(arg), ' '), msg)
|
||||
if not no_show_help then
|
||||
printf('')
|
||||
show_help()
|
||||
end
|
||||
os.exit(2)
|
||||
end
|
||||
|
||||
function dummy()
|
||||
cmd_err('requires a subcommand')
|
||||
end
|
||||
|
||||
-- our stuff
|
||||
|
||||
function show_info()
|
||||
local info = olsrd.oi()
|
||||
tprint(info)
|
||||
end
|
||||
|
||||
function olsr1_nodeinfo(...)
|
||||
if #{ ... } == 0 then
|
||||
return cmd_err('requires at least one argument (example: "all")', true)
|
||||
end
|
||||
|
||||
local query = table.concat({ ... }, '/')
|
||||
local res = olsrd.olsr1_get_nodeinfo(query)
|
||||
tprint(res)
|
||||
end
|
||||
|
||||
function olsr2_nodeinfo_raw(...)
|
||||
if #{ ... } == 0 then
|
||||
return cmd_err('requires at least one argument (example: "nhdpinfo link")', true)
|
||||
end
|
||||
|
||||
local query = table.concat({ ... }, ' ')
|
||||
local res = olsrd.olsr2_get_nodeinfo_raw(query)
|
||||
print(res)
|
||||
end
|
||||
|
||||
function olsr2_nodeinfo_json(...)
|
||||
if #{ ... } == 0 then
|
||||
return cmd_err('requires at least one argument (example: "nhdpinfo jsonraw link")', true)
|
||||
end
|
||||
|
||||
local query = table.concat({ ... }, ' ')
|
||||
local res = olsrd.olsr2_get_nodeinfo(query)
|
||||
tprint(res)
|
||||
end
|
||||
|
||||
sub = {
|
||||
info = { show_info, 'Show information about status of olsr1 and olsr2' },
|
||||
help = { show_help, 'Show help' },
|
||||
olsr1 = { dummy, 'OLSRv1 Control Commands', {
|
||||
nodeinfo = { olsr1_nodeinfo, 'OLSRv1 Nodeinfo' }
|
||||
} },
|
||||
olsr2 = { dummy, 'OLSRv2 Control Commands', {
|
||||
nodeinfo = { dummy, 'OLSRv2 Nodeinfo', {
|
||||
raw = { olsr2_nodeinfo_raw, 'OLSRv2 Nodeinfo Raw' },
|
||||
json = { olsr2_nodeinfo_json, 'OLSRv2 Nodeinfo JSON' }
|
||||
} }
|
||||
} }
|
||||
}
|
||||
|
||||
exec_cmd(table.slice(arg), sub)
|
34
package/gluon-mesh-olsrd/src/Makefile
Normal file
34
package/gluon-mesh-olsrd/src/Makefile
Normal file
@ -0,0 +1,34 @@
|
||||
all: respondd.so libolsrdhelper.so olsrd.so
|
||||
|
||||
CFLAGS += -Wall -D_GNU_SOURCE
|
||||
|
||||
ifeq ($(origin PKG_CONFIG), undefined)
|
||||
PKG_CONFIG = pkg-config
|
||||
ifeq ($(shell which $(PKG_CONFIG) 2>/dev/null),)
|
||||
$(error $(PKG_CONFIG) not found)
|
||||
endif
|
||||
endif
|
||||
|
||||
CFLAGS += $(shell pkg-config --cflags json-c)
|
||||
LDFLAGS += $(shell pkg-config --libs json-c)
|
||||
|
||||
SOURCES_HELPER = libolsrdhelper.c libolsrdhelper-neigh.c uclient.c
|
||||
FILES_HELPER = $(SOURCES_HELPER) libolsrdhelper.h uclient.h
|
||||
|
||||
SOURCES_RESPONDD = respondd.c respondd-nodeinfo.c respondd-neighbours.c
|
||||
FILES_RESPONDD = $(SOURCES_RESPONDD) respondd-common.h
|
||||
|
||||
SOURCES_CLI = olsrd-cli.c
|
||||
FILES_CLI = $(SOURCES_CLI)
|
||||
|
||||
SOURCES_LUA = olsrd.c
|
||||
FILES_LUA = $(SOURCES_LUA)
|
||||
|
||||
respondd.so: libolsrdhelper.so $(FILES_RESPONDD)
|
||||
$(CC) $(CFLAGS) $(LDFLAGS) -I. -L. -shared -fPIC -fvisibility=hidden -o $@ $(SOURCES_RESPONDD) $(LDLIBS) -lgluonutil -lolsrdhelper -luci
|
||||
|
||||
libolsrdhelper.so: libolsrdhelper.h libolsrdhelper.c
|
||||
$(CC) $(CFLAGS) $(LDFLAGS) -shared -fPIC -o $@ $(SOURCES_HELPER) $(LDLIBS) -lgluonutil -luclient
|
||||
|
||||
olsrd.so: libolsrdhelper.so $(FILES_LUA)
|
||||
$(CC) $(LUA_CFLAGS) $(CFLAGS) $(LDFLAGS) -I. -L. -shared -fPIC -o $@ $(SOURCES_LUA) $(LDLIBS) -lgluonutil -lolsrdhelper -luci -llua-jsonc
|
164
package/gluon-mesh-olsrd/src/libolsrdhelper-neigh.c
Normal file
164
package/gluon-mesh-olsrd/src/libolsrdhelper-neigh.c
Normal file
@ -0,0 +1,164 @@
|
||||
/*
|
||||
Copyright (c) 2021, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "libolsrdhelper.h"
|
||||
|
||||
struct json_object * olsr1_get_neigh(void) {
|
||||
json_object *resp;
|
||||
int error = olsr1_get_nodeinfo("links", &resp);
|
||||
|
||||
if (error) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
json_object *out = json_object_new_object();
|
||||
|
||||
if (!out)
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
|
||||
links
|
||||
|
||||
localIP "10.12.11.43"
|
||||
remoteIP "10.12.11.1"
|
||||
olsrInterface "mesh-vpn"
|
||||
ifName "mesh-vpn"
|
||||
validityTime 141239
|
||||
symmetryTime 123095
|
||||
asymmetryTime 25552910
|
||||
vtime 124000
|
||||
currentLinkStatus "SYMMETRIC"
|
||||
previousLinkStatus "SYMMETRIC"
|
||||
hysteresis 0
|
||||
pending false
|
||||
lostLinkTime 0
|
||||
helloTime 0
|
||||
lastHelloTime 0
|
||||
seqnoValid false
|
||||
seqno 0
|
||||
lossHelloInterval 3000
|
||||
lossTime 3595
|
||||
lossMultiplier 65536
|
||||
linkCost 1.084961
|
||||
linkQuality 1
|
||||
neighborLinkQuality 0.921
|
||||
|
||||
*/
|
||||
|
||||
json_object *links = json_object_object_get(resp, "links");
|
||||
if (!links)
|
||||
return NULL;
|
||||
|
||||
int linkcount = json_object_array_length(links);
|
||||
|
||||
for (int i = 0; i < linkcount; i++) {
|
||||
struct json_object *link = json_object_array_get_idx(links, i);
|
||||
if (!link)
|
||||
return NULL;
|
||||
|
||||
struct json_object *neigh = json_object_new_object();
|
||||
if (!neigh)
|
||||
return NULL;
|
||||
|
||||
json_object_object_add(neigh, "ifname", json_object_object_get(link, "ifName"));
|
||||
// TODO: do we need this? should we set this? (we could pick the one peer that we currently route 0.0.0.0 over...)
|
||||
json_object_object_add(neigh, "best", json_object_new_boolean(0));
|
||||
json_object_object_add(neigh, "etx", json_object_object_get(link, "etx"));
|
||||
json_object_object_add(neigh, "ip", json_object_object_get(link, "remoteIP"));
|
||||
|
||||
json_object_object_add(out, json_object_get_string(json_object_object_get(link, "remoteIP")), neigh);
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
struct json_object * olsr2_get_neigh(void) {
|
||||
json_object *resp;
|
||||
int error = olsr2_get_nodeinfo("nhdpinfo jsonraw link", &resp);
|
||||
|
||||
if (error) {
|
||||
return NULL;
|
||||
}
|
||||
|
||||
json_object *out = json_object_new_object();
|
||||
|
||||
if (!out)
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
|
||||
links
|
||||
|
||||
"if":"olsr12",
|
||||
"link_bindto":"fe80::ec67:efff:fed3:d856",
|
||||
"link_vtime_value":20,
|
||||
"link_itime_value":2,
|
||||
"link_symtime":19.786,
|
||||
"link_heardtime":19.886,
|
||||
"link_vtime":39.886,
|
||||
"link_status":"symmetric",
|
||||
"link_dualstack":"-",
|
||||
"link_mac":"b2:d6:9d:88:c1:58",
|
||||
"link_flood_local":"true",
|
||||
"link_flood_remote":"true",
|
||||
"link_flood_willingness":7,
|
||||
"neighbor_originator":"fdff:182f:da60:abc::66",
|
||||
"neighbor_dualstack":"-",
|
||||
"domain":0,
|
||||
"domain_metric":"ff_dat_metric",
|
||||
"domain_metric_in":"1.02kbit/s",
|
||||
"domain_metric_in_raw":2105088,
|
||||
"domain_metric_out":"1.02kbit/s",
|
||||
"domain_metric_out_raw":2105088,
|
||||
|
||||
*/
|
||||
|
||||
json_object *links = json_object_object_get(resp, "link");
|
||||
if (!links)
|
||||
return NULL;
|
||||
|
||||
int linkcount = json_object_array_length(links);
|
||||
|
||||
for (int i = 0; i < linkcount; i++) {
|
||||
struct json_object *link = json_object_array_get_idx(links, i);
|
||||
if (!link)
|
||||
return NULL;
|
||||
|
||||
struct json_object *neigh = json_object_new_object();
|
||||
if (!neigh)
|
||||
return NULL;
|
||||
|
||||
json_object_object_add(neigh, "ifname", json_object_object_get(link, "if"));
|
||||
// TODO: do we need this? should we set this? (we could pick the one peer that we currently route 0.0.0.0 over...)
|
||||
json_object_object_add(neigh, "best", json_object_new_boolean(0));
|
||||
json_object_object_add(neigh, "etx", json_object_object_get(link, "link_vtime"));
|
||||
json_object_object_add(neigh, "ip", json_object_object_get(link, "neighbor_originator"));
|
||||
|
||||
json_object_object_add(out, json_object_get_string(json_object_object_get(link, "link_mac")), neigh);
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
415
package/gluon-mesh-olsrd/src/libolsrdhelper.c
Normal file
415
package/gluon-mesh-olsrd/src/libolsrdhelper.c
Normal file
@ -0,0 +1,415 @@
|
||||
/*
|
||||
Copyright (c) 2021, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "libolsrdhelper.h"
|
||||
#include "uclient.h"
|
||||
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
#include <net/if.h>
|
||||
#include <stdbool.h>
|
||||
#include <sys/wait.h>
|
||||
|
||||
#include <libubox/blobmsg.h>
|
||||
#include <libubox/uloop.h>
|
||||
|
||||
#define USER_AGENT "olsrdhelper (libuclient)"
|
||||
|
||||
#define BASE_URL_1 "http://127.0.0.1:9090"
|
||||
#define BASE_URL_1_LEN sizeof(BASE_URL_1)
|
||||
#define BASE_URL_2 "http://127.0.0.1:1980/telnet"
|
||||
#define BASE_URL_2_LEN sizeof(BASE_URL_2)
|
||||
|
||||
// 10 mb
|
||||
#define MAX_RESPONSE (1024 * 1024 * 1024 * 10)
|
||||
#define PACKET (1024 * 1024)
|
||||
|
||||
struct list_obj {
|
||||
char* data;
|
||||
size_t len;
|
||||
struct list_obj *next;
|
||||
};
|
||||
|
||||
struct get_data_ctx {
|
||||
struct list_obj *current;
|
||||
struct list_obj *start;
|
||||
size_t total_length;
|
||||
int error;
|
||||
};
|
||||
|
||||
struct recv_json_ctx {
|
||||
size_t size;
|
||||
int error;
|
||||
json_object *parsed;
|
||||
|
||||
struct get_data_ctx *get_data;
|
||||
};
|
||||
|
||||
struct recv_txt_ctx {
|
||||
size_t size;
|
||||
int error;
|
||||
char *data;
|
||||
|
||||
struct get_data_ctx *get_data;
|
||||
};
|
||||
|
||||
/** Recieves all the data */
|
||||
struct get_data_ctx * get_all_data_init() {
|
||||
struct get_data_ctx *ctx = malloc(sizeof(struct get_data_ctx));
|
||||
|
||||
if (!ctx)
|
||||
goto fail;
|
||||
|
||||
*ctx = (struct get_data_ctx){
|
||||
.total_length = 0,
|
||||
.error = 0
|
||||
};
|
||||
|
||||
ctx->start = malloc(sizeof(struct list_obj));
|
||||
if (!ctx->start)
|
||||
goto fail;
|
||||
|
||||
*ctx->start = (struct list_obj){
|
||||
.len = 0,
|
||||
.next = NULL
|
||||
};
|
||||
|
||||
ctx->current = ctx->start;
|
||||
|
||||
return ctx;
|
||||
|
||||
fail:
|
||||
if (ctx) {
|
||||
if (ctx->start)
|
||||
free(ctx->start);
|
||||
|
||||
free(ctx);
|
||||
}
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
// returns true when we are not done (if (process) return)
|
||||
bool get_all_data_process(struct uclient *cl, struct get_data_ctx *ctx) {
|
||||
char buf[PACKET];
|
||||
size_t len;
|
||||
|
||||
while (true) {
|
||||
len = uclient_read_account(cl, buf, sizeof(buf));
|
||||
|
||||
if (len == -1) {
|
||||
ctx->error = 1;
|
||||
return false;
|
||||
}
|
||||
|
||||
if (!len) {
|
||||
return false;
|
||||
}
|
||||
|
||||
ctx->current->data = malloc(len);
|
||||
if (!ctx->current->data) {
|
||||
ctx->error = 1;
|
||||
return false;
|
||||
}
|
||||
ctx->current->len = len;
|
||||
|
||||
memcpy(ctx->current->data, buf, len);
|
||||
ctx->total_length += len;
|
||||
|
||||
ctx->current->next = malloc(sizeof(struct list_obj));
|
||||
*ctx->current->next = (struct list_obj){
|
||||
.len = 0,
|
||||
.next = NULL
|
||||
};
|
||||
ctx->current = ctx->current->next;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
int get_all_data_finalize(struct get_data_ctx *ctx, char ** data, size_t * size) {
|
||||
if (ctx->error) {
|
||||
int error = ctx->error;
|
||||
|
||||
// TODO: free
|
||||
|
||||
return error;
|
||||
};
|
||||
|
||||
ctx->total_length++; // trailing null
|
||||
|
||||
*data = malloc(ctx->total_length);
|
||||
if (!data)
|
||||
return 1;
|
||||
|
||||
size_t offset = 0;
|
||||
|
||||
struct list_obj *prev;
|
||||
|
||||
ctx->current = ctx->start;
|
||||
while(ctx->current) {
|
||||
if (ctx->current->len) {
|
||||
memcpy(*data + offset, ctx->current->data, ctx->current->len);
|
||||
offset += ctx->current->len;
|
||||
free(ctx->current->data);
|
||||
}
|
||||
|
||||
prev = ctx->current;
|
||||
ctx->current = ctx->current->next;
|
||||
free(prev);
|
||||
}
|
||||
|
||||
(*data)[ctx->total_length - 1] = '\0';
|
||||
*size = ctx->total_length;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/** Receives data from uclient and writes it to memory */
|
||||
static void recv_json_cb(struct uclient *cl) {
|
||||
struct recv_json_ctx *ctx = uclient_get_custom(cl);
|
||||
|
||||
if (get_all_data_process(cl, ctx->get_data)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
static void recv_json_eof_cb(struct uclient *cl) {
|
||||
struct recv_json_ctx *ctx = uclient_get_custom(cl);
|
||||
|
||||
char * data;
|
||||
size_t size;
|
||||
|
||||
if (get_all_data_finalize(ctx->get_data, &data, &size)) {
|
||||
ctx->error = UCLIENT_ERROR_SIZE_MISMATCH;
|
||||
return;
|
||||
}
|
||||
|
||||
if (data[0] != "{"[0]) {
|
||||
ctx->error = UCLIENT_ERROR_NOT_JSON;
|
||||
return;
|
||||
}
|
||||
|
||||
// TODO: handle parser error, add error code for malformed json
|
||||
ctx->parsed = json_tokener_parse(data);
|
||||
}
|
||||
|
||||
/** Receives data from uclient and writes it to memory */
|
||||
static void recv_txt_cb(struct uclient *cl) {
|
||||
struct recv_txt_ctx *ctx = uclient_get_custom(cl);
|
||||
|
||||
if (get_all_data_process(cl, ctx->get_data)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
static void recv_txt_eof_cb(struct uclient *cl) {
|
||||
struct recv_txt_ctx *ctx = uclient_get_custom(cl);
|
||||
|
||||
if (get_all_data_finalize(ctx->get_data, &ctx->data, &ctx->size)) {
|
||||
ctx->error = UCLIENT_ERROR_SIZE_MISMATCH;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
bool success_exit(char *cmd, ...) {
|
||||
pid_t pid = fork();
|
||||
|
||||
if (pid == 0) {
|
||||
va_list val;
|
||||
char **args = NULL;
|
||||
int argc;
|
||||
|
||||
// Determine number of variadic arguments
|
||||
va_start(val, cmd);
|
||||
argc = 2; // leading command + trailing NULL
|
||||
while (va_arg(val, char *) != NULL)
|
||||
argc++;
|
||||
va_end(val);
|
||||
|
||||
// Allocate args, put references to command / variadic arguments + NULL in args
|
||||
args = (char **) malloc(argc * sizeof(char*));
|
||||
args[0] = cmd;
|
||||
va_start(val, cmd);
|
||||
int i = 0;
|
||||
do {
|
||||
i++;
|
||||
args[i] = va_arg(val, char *);
|
||||
// since this triggers AFTERWARDS, the trailing null still gets pushed
|
||||
} while (args[i] != NULL);
|
||||
va_end(val);
|
||||
|
||||
execv(cmd, args);
|
||||
exit(1);
|
||||
return false;
|
||||
}
|
||||
|
||||
int status;
|
||||
|
||||
if (waitpid(pid, &status, 0) == -1) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (WIFEXITED(status)) {
|
||||
return WEXITSTATUS(status) == 0;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
// get enabled from site.conf mesh.olsrX.enabled, get running from service X status
|
||||
int oi(struct olsr_info **out) {
|
||||
int ex = 0;
|
||||
|
||||
json_object *site = gluonutil_load_site_config();
|
||||
if (!site)
|
||||
goto fail;
|
||||
|
||||
json_object *mesh = json_object_object_get(site, "mesh");
|
||||
if (!mesh)
|
||||
goto fail;
|
||||
|
||||
json_object *olsrd = json_object_object_get(mesh, "olsrd");
|
||||
if (!olsrd)
|
||||
goto fail;
|
||||
|
||||
json_object *v1 = json_object_object_get(olsrd, "v1_4");
|
||||
json_object *v2 = json_object_object_get(olsrd, "v2");
|
||||
|
||||
struct olsr_info *info = (struct olsr_info *) malloc(sizeof(struct olsr_info));
|
||||
|
||||
*info = (struct olsr_info){
|
||||
.olsr1 = {
|
||||
.enabled = false,
|
||||
.running = false,
|
||||
},
|
||||
.olsr2 = {
|
||||
.enabled = false,
|
||||
.running = false,
|
||||
}
|
||||
};
|
||||
|
||||
if (v1 && json_object_get_boolean(json_object_object_get(v1, "enable"))) {
|
||||
info->olsr1.enabled = true;
|
||||
|
||||
if (success_exit("/etc/init.d/olsrd", "running", NULL)) {
|
||||
info->olsr1.running = true;
|
||||
}
|
||||
}
|
||||
|
||||
if (v2 && json_object_get_boolean(json_object_object_get(v2, "enable"))) {
|
||||
info->olsr2.enabled = true;
|
||||
|
||||
if (success_exit("/etc/init.d/olsrd2", "running", NULL)) {
|
||||
info->olsr2.running = true;
|
||||
}
|
||||
}
|
||||
|
||||
*out = info;
|
||||
|
||||
end:
|
||||
return ex;
|
||||
|
||||
fail:
|
||||
ex = 1;
|
||||
goto end;
|
||||
}
|
||||
|
||||
int olsr1_get_nodeinfo(const char *path, json_object **out) {
|
||||
char url[BASE_URL_1_LEN + strlen(path) + 2];
|
||||
sprintf(url, "%s/%s", BASE_URL_1, path);
|
||||
|
||||
uloop_init();
|
||||
struct recv_json_ctx json_ctx = { };
|
||||
json_ctx.get_data = get_all_data_init();
|
||||
if (!json_ctx.get_data)
|
||||
return 1;
|
||||
|
||||
int err_code = get_url(url, &recv_json_cb, &recv_json_eof_cb, &json_ctx, -1);
|
||||
uloop_done();
|
||||
|
||||
if (err_code) {
|
||||
return err_code;
|
||||
}
|
||||
|
||||
if (json_ctx.error) {
|
||||
return json_ctx.error;
|
||||
}
|
||||
|
||||
*out = json_ctx.parsed;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int olsr2_get_nodeinfo(const char *cmd, json_object **out) {
|
||||
char url[BASE_URL_2_LEN + strlen(cmd) + 2];
|
||||
sprintf(url, "%s/%s", BASE_URL_2, cmd);
|
||||
|
||||
uloop_init();
|
||||
struct recv_json_ctx json_ctx = { };
|
||||
json_ctx.get_data = get_all_data_init();
|
||||
if (!json_ctx.get_data)
|
||||
return 1;
|
||||
int err_code = get_url(url, &recv_json_cb, &recv_json_eof_cb, &json_ctx, -1);
|
||||
uloop_done();
|
||||
|
||||
if (err_code) {
|
||||
return err_code;
|
||||
}
|
||||
|
||||
if (json_ctx.error) {
|
||||
return json_ctx.error;
|
||||
}
|
||||
|
||||
*out = json_ctx.parsed;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int olsr2_get_nodeinfo_raw(const char *cmd, char **out) {
|
||||
char url[BASE_URL_2_LEN + strlen(cmd) + 2];
|
||||
sprintf(url, "%s/%s", BASE_URL_2, cmd);
|
||||
|
||||
uloop_init();
|
||||
struct recv_txt_ctx txt_ctx = { };
|
||||
txt_ctx.get_data = get_all_data_init();
|
||||
if (!txt_ctx.get_data)
|
||||
return 1;
|
||||
int err_code = get_url(url, &recv_txt_cb, &recv_txt_eof_cb, &txt_ctx, -1);
|
||||
uloop_done();
|
||||
|
||||
if (err_code) {
|
||||
return err_code;
|
||||
}
|
||||
|
||||
if (txt_ctx.error) {
|
||||
return txt_ctx.error;
|
||||
}
|
||||
|
||||
*out = txt_ctx.data;
|
||||
|
||||
return 0;
|
||||
}
|
56
package/gluon-mesh-olsrd/src/libolsrdhelper.h
Executable file
56
package/gluon-mesh-olsrd/src/libolsrdhelper.h
Executable file
@ -0,0 +1,56 @@
|
||||
/*
|
||||
Copyright (c) 2021, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <json-c/json.h>
|
||||
#include <libubox/uclient.h>
|
||||
#include <libgluonutil.h>
|
||||
#include <uci.h>
|
||||
|
||||
struct olsr1_info {
|
||||
bool enabled;
|
||||
bool running;
|
||||
};
|
||||
|
||||
struct olsr2_info {
|
||||
bool enabled;
|
||||
bool running;
|
||||
};
|
||||
|
||||
struct olsr_info {
|
||||
struct olsr1_info olsr1;
|
||||
struct olsr2_info olsr2;
|
||||
};
|
||||
|
||||
int oi(struct olsr_info **out);
|
||||
|
||||
int olsr1_get_nodeinfo(const char *path, json_object **out);
|
||||
|
||||
int olsr2_get_nodeinfo(const char *cmd, json_object **out);
|
||||
int olsr2_get_nodeinfo_raw(const char *cmd, char **out);
|
||||
|
||||
struct json_object * olsr1_get_neigh(void);
|
||||
struct json_object * olsr2_get_neigh(void);
|
36
package/gluon-mesh-olsrd/src/libolsrdhelper.pc
Normal file
36
package/gluon-mesh-olsrd/src/libolsrdhelper.pc
Normal file
@ -0,0 +1,36 @@
|
||||
# SPDX-License-Identifier: MIT
|
||||
#
|
||||
# olsr helper functions library
|
||||
#
|
||||
# Copyright (c) 2022, Maciej Krüger <maciej@xeredo.it>
|
||||
#
|
||||
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
# of this software and associated documentation files (the "Software"), to deal
|
||||
# in the Software without restriction, including without limitation the rights
|
||||
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
# copies of the Software, and to permit persons to whom the Software is
|
||||
# furnished to do so, subject to the following conditions:
|
||||
#
|
||||
# The above copyright notice and this permission notice shall be included in
|
||||
# all copies or substantial portions of the Software.
|
||||
#
|
||||
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
# THE SOFTWARE.
|
||||
|
||||
prefix=/usr
|
||||
exec_prefix=/usr
|
||||
libdir=${exec_prefix}/lib
|
||||
includedir=${prefix}/include/gluon-mesh-olsrd
|
||||
|
||||
Name: libolsrdhelper
|
||||
Version: 1
|
||||
Description: olsr helper functions library
|
||||
Requires.private: json-c
|
||||
Libs: -luclient -lolsrdhelper -L${libdir} -lgluonutil -luci
|
||||
Libs.private:
|
||||
Cflags: -I${includedir}
|
168
package/gluon-mesh-olsrd/src/olsrd.c
Normal file
168
package/gluon-mesh-olsrd/src/olsrd.c
Normal file
@ -0,0 +1,168 @@
|
||||
/*
|
||||
|
||||
Copyright 2022 Maciej Krüger <maciej@xeredo.it>
|
||||
|
||||
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
|
||||
|
||||
Unless required by applicable law or agreed to in writing, software
|
||||
distributed under the License is distributed on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
See the License for the specific language governing permissions and
|
||||
limitations under the License.
|
||||
|
||||
*/
|
||||
|
||||
#include <dirent.h>
|
||||
#include <errno.h>
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <lua.h>
|
||||
#include <lualib.h>
|
||||
#include <lauxlib.h>
|
||||
#include <lua-jsonc.h>
|
||||
|
||||
#include <libolsrdhelper.h>
|
||||
|
||||
#define OLSRD "gluon.olsrd"
|
||||
|
||||
static int find_module_version (lua_State *L) {
|
||||
const char *mod = luaL_checkstring(L, 1);
|
||||
|
||||
DIR *d = opendir("/usr/lib");
|
||||
|
||||
if (d == NULL)
|
||||
return luaL_error(L, "cannot open /usr/lib: %s", strerror(errno));
|
||||
|
||||
struct dirent *entry;
|
||||
while ((entry = readdir(d)) != NULL) {
|
||||
if (entry->d_type == DT_REG && !strncmp(mod, entry->d_name, strlen(mod))) {
|
||||
lua_pushstring(L, entry->d_name);
|
||||
closedir(d);
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
closedir(d);
|
||||
return luaL_error(L, "mod %s not found", mod);
|
||||
}
|
||||
|
||||
static int lua_olsr1_get_nodeinfo (lua_State *L) {
|
||||
const char *query = luaL_checkstring(L, 1);
|
||||
|
||||
json_object *resp;
|
||||
|
||||
if (olsr1_get_nodeinfo(query, &resp))
|
||||
return luaL_error(L, "olsr1_get_nodeinfo(%s) failed", query);
|
||||
|
||||
lua_jsonc_push_json(L, resp);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int lua_olsr2_get_nodeinfo (lua_State *L) {
|
||||
const char *query = luaL_checkstring(L, 1);
|
||||
|
||||
json_object *resp;
|
||||
|
||||
if (olsr2_get_nodeinfo(query, &resp))
|
||||
return luaL_error(L, "olsr2_get_nodeinfo(%s) failed", query);
|
||||
|
||||
lua_jsonc_push_json(L, resp);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int lua_olsr2_get_nodeinfo_raw (lua_State *L) {
|
||||
const char *query = luaL_checkstring(L, 1);
|
||||
|
||||
char *resp;
|
||||
|
||||
if (olsr2_get_nodeinfo_raw(query, &resp))
|
||||
return luaL_error(L, "olsr2_get_nodeinfo_raw(%s) failed", query);
|
||||
|
||||
lua_pushstring(L, resp);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int lua_olsr1_get_neigh (lua_State *L) {
|
||||
json_object *resp = olsr1_get_neigh();
|
||||
|
||||
if (!resp)
|
||||
return luaL_error(L, "olsr2_get_neigh() failed");
|
||||
|
||||
lua_jsonc_push_json(L, resp);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int lua_olsr2_get_neigh (lua_State *L) {
|
||||
json_object *resp = olsr2_get_neigh();
|
||||
|
||||
if (!resp)
|
||||
return luaL_error(L, "olsr2_get_neigh() failed");
|
||||
|
||||
lua_jsonc_push_json(L, resp);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int lua_oi (lua_State *L) {
|
||||
struct olsr_info *info;
|
||||
|
||||
if (oi(&info))
|
||||
return luaL_error(L, "olsr_info() call failed");
|
||||
|
||||
lua_newtable(L);
|
||||
|
||||
|
||||
lua_newtable(L); // olsr1
|
||||
|
||||
lua_pushboolean(L, info->olsr1.enabled);
|
||||
lua_setfield(L, -2, "enabled");
|
||||
|
||||
lua_pushboolean(L, info->olsr1.running);
|
||||
lua_setfield(L, -2, "running");
|
||||
|
||||
lua_setfield(L, -2, "olsr1");
|
||||
|
||||
|
||||
lua_newtable(L); // olsr2
|
||||
|
||||
lua_pushboolean(L, info->olsr2.enabled);
|
||||
lua_setfield(L, -2, "enabled");
|
||||
|
||||
lua_pushboolean(L, info->olsr2.running);
|
||||
lua_setfield(L, -2, "running");
|
||||
|
||||
lua_setfield(L, -2, "olsr2");
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static const luaL_reg olsrd_methods[] = {
|
||||
{ "find_module_version", find_module_version },
|
||||
|
||||
{ "oi", lua_oi },
|
||||
|
||||
{ "olsr1_get_nodeinfo", lua_olsr1_get_nodeinfo },
|
||||
|
||||
{ "olsr2_get_nodeinfo", lua_olsr2_get_nodeinfo },
|
||||
{ "olsr2_get_nodeinfo_raw", lua_olsr2_get_nodeinfo_raw },
|
||||
|
||||
{ "olsr1_get_neigh", lua_olsr1_get_neigh },
|
||||
{ "olsr2_get_neigh", lua_olsr2_get_neigh },
|
||||
{ }
|
||||
};
|
||||
|
||||
int luaopen_gluon_olsrd(lua_State *L)
|
||||
{
|
||||
luaL_register(L, OLSRD, olsrd_methods);
|
||||
|
||||
return 1;
|
||||
}
|
36
package/gluon-mesh-olsrd/src/respondd-common.h
Normal file
36
package/gluon-mesh-olsrd/src/respondd-common.h
Normal file
@ -0,0 +1,36 @@
|
||||
/*
|
||||
Copyright (c) 2021, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#define make_safe_fnc(name, real_name) \
|
||||
struct json_object * name (void) { \
|
||||
return make_safe(&(real_name)); \
|
||||
}
|
||||
|
||||
struct json_object * make_safe(struct json_object * (*fnc)(void));
|
||||
|
||||
struct json_object * respondd_provider_nodeinfo(void);
|
||||
struct json_object * respondd_provider_neighbours(void);
|
67
package/gluon-mesh-olsrd/src/respondd-neighbours.c
Normal file
67
package/gluon-mesh-olsrd/src/respondd-neighbours.c
Normal file
@ -0,0 +1,67 @@
|
||||
/*
|
||||
Copyright (c) 2022, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "respondd-common.h"
|
||||
|
||||
#include <libgluonutil.h>
|
||||
|
||||
#include <json-c/json.h>
|
||||
|
||||
#include <libolsrdhelper.h>
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
struct json_object * real_respondd_provider_neighbours(void) {
|
||||
struct olsr_info *info;
|
||||
|
||||
struct json_object *ret = json_object_new_object();
|
||||
|
||||
if (oi(&info))
|
||||
return ret;
|
||||
|
||||
struct json_object *neigh = json_object_new_array();
|
||||
|
||||
if (info->olsr1.enabled && info->olsr1.running) {
|
||||
struct json_object *neigh1 = olsr1_get_neigh();
|
||||
if (neigh1)
|
||||
for (int i = 0; i < json_object_array_length(neigh1); i++)
|
||||
json_object_array_add(neigh, json_object_array_get_idx(neigh1, i));
|
||||
}
|
||||
|
||||
if (info->olsr2.enabled && info->olsr2.running) {
|
||||
struct json_object *neigh2 = olsr2_get_neigh();
|
||||
if (neigh2)
|
||||
for (int i = 0; i < json_object_array_length(neigh2); i++)
|
||||
json_object_array_add(neigh, json_object_array_get_idx(neigh2, i));
|
||||
}
|
||||
|
||||
json_object_object_add(ret, "neigh", neigh);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
make_safe_fnc(respondd_provider_neighbours, real_respondd_provider_neighbours)
|
466
package/gluon-mesh-olsrd/src/respondd-nodeinfo.c
Normal file
466
package/gluon-mesh-olsrd/src/respondd-nodeinfo.c
Normal file
@ -0,0 +1,466 @@
|
||||
/*
|
||||
Copyright (c) 2022, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "respondd-common.h"
|
||||
|
||||
#include <libgluonutil.h>
|
||||
|
||||
#include <json-c/json.h>
|
||||
|
||||
#include <libolsrdhelper.h>
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
static json_object * olsr1_get_plugins(void) {
|
||||
json_object *resp;
|
||||
|
||||
if (olsr1_get_nodeinfo("plugins", &resp))
|
||||
return NULL;
|
||||
|
||||
return json_object_object_get(resp, "plugins");
|
||||
}
|
||||
|
||||
static json_object * olsr1_get_version (void) {
|
||||
json_object *resp;
|
||||
|
||||
if (olsr1_get_nodeinfo("version", &resp))
|
||||
return NULL;
|
||||
|
||||
return json_object_object_get(json_object_object_get(resp, "version"), "version");
|
||||
}
|
||||
|
||||
static json_object * olsr2_get_version (void) {
|
||||
json_object *resp;
|
||||
|
||||
if (olsr2_get_nodeinfo("systeminfo jsonraw version", &resp))
|
||||
return NULL;
|
||||
|
||||
return json_object_object_get(json_object_object_get(resp, "version"), "version_text");
|
||||
}
|
||||
|
||||
static json_object * olsr1_get_addresses (void) {
|
||||
json_object *resp;
|
||||
|
||||
if (olsr1_get_nodeinfo("interfaces", &resp))
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
|
||||
interfaces []
|
||||
name "m_uplink"
|
||||
configured true
|
||||
hostEmulation false
|
||||
hostEmulationAddress "0.0.0.0"
|
||||
olsrInterface
|
||||
-- might be false (and then ipAddress key is missing)
|
||||
up true
|
||||
ipv4Address "10.12.23.234"
|
||||
ipv4Netmask "255.255.0.0"
|
||||
ipv4Broadcast "10.12.255.255"
|
||||
mode "mesh"
|
||||
ipv6Address "::"
|
||||
ipv6Multicast "::"
|
||||
-- we need this
|
||||
ipAddress "10.12.23.234"
|
||||
..
|
||||
InterfaceConfiguration {…}
|
||||
InterfaceConfigurationDefaults {…}
|
||||
*/
|
||||
|
||||
json_object *out = json_object_new_array();
|
||||
|
||||
json_object *intfs = json_object_object_get(resp, "interfaces");
|
||||
|
||||
for (int i = 0; i < json_object_array_length(intfs); i++) {
|
||||
struct json_object *el = json_object_array_get_idx(intfs, i);
|
||||
struct json_object *olsr = json_object_object_get(el, "olsrInterface");
|
||||
struct json_object *ip = json_object_object_get(olsr, "ipAddress"); // might be null (up=false)
|
||||
if (ip) {
|
||||
json_object_array_add(out, ip);
|
||||
}
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
static json_object * olsr2_get_addresses (void) {
|
||||
/*
|
||||
|
||||
> olsrv2info jsonraw originator
|
||||
{"originator": [{
|
||||
"originator":"-"},{
|
||||
"originator":"fdff:182f:da60:abc:23:ee1a:dec6:d17c"}]}
|
||||
|
||||
if you're wondering "what the fuck": me too, me too
|
||||
|
||||
*/
|
||||
|
||||
json_object *resp;
|
||||
|
||||
if (olsr2_get_nodeinfo("olsrv2info jsonraw originator", &resp))
|
||||
return NULL;
|
||||
|
||||
json_object *out = json_object_new_array();
|
||||
|
||||
json_object *origs = json_object_object_get(resp, "originator");
|
||||
|
||||
for (int i = 0; i < json_object_array_length(origs); i++) {
|
||||
struct json_object *el = json_object_array_get_idx(origs, i);
|
||||
if (json_object_get_string(el)[0] != "-"[0]) {
|
||||
json_object_array_add(out, el);
|
||||
}
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
struct json_object * olsr1_get_interfaces (void) {
|
||||
json_object *resp;
|
||||
|
||||
if (olsr1_get_nodeinfo("interfaces", &resp))
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
|
||||
interfaces []
|
||||
name "m_uplink"
|
||||
configured true
|
||||
hostEmulation false
|
||||
hostEmulationAddress "0.0.0.0"
|
||||
olsrInterface
|
||||
-- might be false (and then ipAddress key is missing)
|
||||
up true
|
||||
ipv4Address "10.12.23.234"
|
||||
ipv4Netmask "255.255.0.0"
|
||||
ipv4Broadcast "10.12.255.255"
|
||||
mode "mesh"
|
||||
ipv6Address "::"
|
||||
ipv6Multicast "::"
|
||||
-- we need this
|
||||
ipAddress "10.12.23.234"
|
||||
..
|
||||
InterfaceConfiguration {…}
|
||||
InterfaceConfigurationDefaults {…}
|
||||
*/
|
||||
|
||||
json_object *out = json_object_new_object();
|
||||
|
||||
json_object *intfs = json_object_object_get(resp, "interfaces");
|
||||
|
||||
for (int i = 0; i < json_object_array_length(intfs); i++) {
|
||||
json_object *el = json_object_array_get_idx(intfs, i);
|
||||
json_object *olsr = json_object_object_get(el, "olsrInterface");
|
||||
|
||||
json_object *intf = json_object_new_object();
|
||||
json_object_object_add(out,
|
||||
json_object_get_string(json_object_object_get(el, "name")),
|
||||
intf
|
||||
);
|
||||
|
||||
json_object_object_add(intf, "configured", json_object_object_get(el, "configured"));
|
||||
json_object_object_add(intf, "up", json_object_object_get(olsr, "up"));
|
||||
json_object_object_add(intf, "ipAddress", json_object_object_get(olsr, "ipAddress"));
|
||||
json_object_object_add(intf, "mode", json_object_object_get(olsr, "mode"));
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
struct json_object * olsr2_get_interfaces (void) {
|
||||
json_object *resp;
|
||||
|
||||
if (olsr2_get_nodeinfo("nhdpinfo jsonraw interface", &resp))
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
|
||||
we're currently just using nhdpinfo, but layer2info might be interesting at some point
|
||||
|
||||
> nhdpinfo jsonraw interface
|
||||
{"interface": [{
|
||||
"if":"ibss0",
|
||||
"if_bindto_v4":"-",
|
||||
"if_bindto_v6":"-",
|
||||
"if_mac":"b8:69:f4:0d:1a:3c",
|
||||
"if_flooding_v4":"false",
|
||||
"if_flooding_v6":"false",
|
||||
"if_dualstack_mode":"-"},{
|
||||
"if":"lo",
|
||||
"if_bindto_v4":"-",
|
||||
"if_bindto_v6":"fdff:182f:da60:abc:23:ee1a:dec6:d17c",
|
||||
"if_mac":"00:00:00:00:00:00",
|
||||
"if_flooding_v4":"false",
|
||||
"if_flooding_v6":"false",
|
||||
"if_dualstack_mode":"-"},{
|
||||
|
||||
> layer2info jsonraw interface
|
||||
{"interface": [{
|
||||
"if":"ibss0",
|
||||
"if_index":14,
|
||||
"if_local_addr":"b8:69:f4:0d:1a:3c",
|
||||
"if_type":"wireless",
|
||||
"if_dlep":"false",
|
||||
"if_ident":"",
|
||||
"if_ident_addr":"",
|
||||
"if_lastseen":0,
|
||||
"if_frequency1":"0",
|
||||
"if_frequency2":"0",
|
||||
"if_bandwidth1":"0",
|
||||
"if_bandwidth2":"0",
|
||||
"if_noise":"-92",
|
||||
"if_ch_active":"40448.827",
|
||||
"if_ch_busy":"1015.889",
|
||||
"if_ch_rx":"263.867",
|
||||
"if_ch_tx":"127.433",
|
||||
"if_mtu":"0",
|
||||
"if_mcs_by_probing":"true",
|
||||
"if_rx_only_unicast":"false",
|
||||
"if_tx_only_unicast":"false",
|
||||
"if_frequency1_origin":"",
|
||||
"if_frequency2_origin":"",
|
||||
"if_bandwidth1_origin":"",
|
||||
"if_bandwidth2_origin":"",
|
||||
"if_noise_origin":"nl80211",
|
||||
"if_ch_active_origin":"nl80211",
|
||||
"if_ch_busy_origin":"nl80211",
|
||||
"if_ch_rx_origin":"nl80211",
|
||||
"if_ch_tx_origin":"nl80211",
|
||||
"if_mtu_origin":"",
|
||||
"if_mcs_by_probing_origin":"nl80211",
|
||||
"if_rx_only_unicast_origin":"",
|
||||
"if_tx_only_unicast_origin":""},{
|
||||
"if":"lo",
|
||||
"if_index":1,
|
||||
"if_local_addr":"00:00:00:00:00:00",
|
||||
"if_type":"wireless",
|
||||
"if_dlep":"false",
|
||||
"if_ident":"",
|
||||
"if_ident_addr":"",
|
||||
"if_lastseen":0,
|
||||
"if_frequency1":"0",
|
||||
"if_frequency2":"0",
|
||||
"if_bandwidth1":"0",
|
||||
"if_bandwidth2":"0",
|
||||
"if_noise":"0",
|
||||
"if_ch_active":"0",
|
||||
"if_ch_busy":"0",
|
||||
"if_ch_rx":"0",
|
||||
"if_ch_tx":"0",
|
||||
"if_mtu":"0",
|
||||
"if_mcs_by_probing":"false",
|
||||
"if_rx_only_unicast":"false",
|
||||
"if_tx_only_unicast":"false",
|
||||
"if_frequency1_origin":"",
|
||||
"if_frequency2_origin":"",
|
||||
"if_bandwidth1_origin":"",
|
||||
"if_bandwidth2_origin":"",
|
||||
"if_noise_origin":"",
|
||||
"if_ch_active_origin":"",
|
||||
"if_ch_busy_origin":"",
|
||||
"if_ch_rx_origin":"",
|
||||
"if_ch_tx_origin":"",
|
||||
"if_mtu_origin":"",
|
||||
"if_mcs_by_probing_origin":"",
|
||||
"if_rx_only_unicast_origin":"",
|
||||
"if_tx_only_unicast_origin":""},{
|
||||
|
||||
*/
|
||||
|
||||
json_object *out = json_object_new_object();
|
||||
|
||||
json_object *intfs = json_object_object_get(resp, "interface");
|
||||
|
||||
for (int i = 0; i < json_object_array_length(intfs); i++) {
|
||||
json_object *el = json_object_array_get_idx(intfs, i);
|
||||
|
||||
json_object *intf = json_object_new_object();
|
||||
json_object_object_add(out,
|
||||
json_object_get_string(json_object_object_get(el, "if")),
|
||||
intf
|
||||
);
|
||||
|
||||
json_object_object_add(intf, "mac", json_object_object_get(el, "if_mac"));
|
||||
json_object_object_add(intf, "v4", json_object_object_get(el, "if_bindto_v4"));
|
||||
json_object_object_add(intf, "v6", json_object_object_get(el, "if_bindto_v6"));
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
/* static struct json_object * get_mesh(void) {
|
||||
struct json_object *ret = json_object_new_object();
|
||||
struct json_object *bat0_interfaces = json_object_new_object();
|
||||
json_object_object_add(bat0_interfaces, "interfaces", get_mesh_subifs("bat0"));
|
||||
json_object_object_add(ret, "bat0", bat0_interfaces);
|
||||
return ret;
|
||||
} */
|
||||
|
||||
struct json_object * real_respondd_provider_nodeinfo(void) {
|
||||
struct olsr_info *info;
|
||||
|
||||
struct json_object *ret = json_object_new_object();
|
||||
|
||||
if (oi(&info))
|
||||
return ret;
|
||||
|
||||
/* struct json_object *network = json_object_new_object();
|
||||
json_object_object_add(network, "addresses", get_addresses());
|
||||
json_object_object_add(network, "mesh", get_mesh());
|
||||
json_object_object_add(ret, "network", network); */
|
||||
|
||||
/*
|
||||
|
||||
TODO: get interfaces and return in following schema
|
||||
|
||||
{
|
||||
interfaces: {
|
||||
$intf_name: {
|
||||
olsr1: {
|
||||
configured,
|
||||
up: intf.olsrInterface.up,
|
||||
}
|
||||
olsr2: {
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
*/
|
||||
|
||||
struct json_object *network = json_object_new_object();
|
||||
|
||||
struct json_object *n_addresses = json_object_new_array();
|
||||
|
||||
json_object_object_add(network, "addresses", n_addresses);
|
||||
|
||||
struct json_object *n_interfaces = json_object_new_object();
|
||||
|
||||
json_object_object_add(network, "interfaces", n_interfaces);
|
||||
|
||||
json_object_object_add(ret, "network", network);
|
||||
|
||||
struct json_object *software = json_object_new_object();
|
||||
|
||||
if (info->olsr1.enabled) {
|
||||
struct json_object *software_olsr1 = json_object_new_object();
|
||||
|
||||
json_object_object_add(software_olsr1, "running", json_object_new_boolean(info->olsr1.running));
|
||||
|
||||
if (info->olsr1.running) {
|
||||
struct json_object *version = olsr1_get_version();
|
||||
if (version)
|
||||
json_object_object_add(software_olsr1, "version", version);
|
||||
|
||||
struct json_object *plugins = olsr1_get_plugins();
|
||||
if (plugins)
|
||||
json_object_object_add(software_olsr1, "plugins", plugins);
|
||||
|
||||
struct json_object *addresses = olsr1_get_addresses();
|
||||
if (addresses) {
|
||||
json_object_object_add(software_olsr1, "addresses", addresses);
|
||||
|
||||
for (int i = 0; i < json_object_array_length(addresses); i++)
|
||||
json_object_array_add(n_addresses, json_object_array_get_idx(addresses, i));
|
||||
}
|
||||
|
||||
struct json_object *interfaces = olsr1_get_interfaces();
|
||||
if (interfaces) {
|
||||
json_object_object_add(software_olsr1, "interfaces", interfaces);
|
||||
|
||||
struct json_object_iterator it = json_object_iter_begin(interfaces);
|
||||
struct json_object_iterator itEnd = json_object_iter_end(interfaces);
|
||||
|
||||
while (!json_object_iter_equal(&it, &itEnd)) {
|
||||
const char * name = json_object_iter_peek_name(&it);
|
||||
json_object *append_key = json_object_object_get(n_interfaces, name);
|
||||
|
||||
if (!append_key) {
|
||||
append_key = json_object_new_object();
|
||||
json_object_object_add(n_interfaces, name, append_key);
|
||||
}
|
||||
|
||||
json_object_object_add(append_key, "olsr1",
|
||||
json_object_object_get(interfaces, name));
|
||||
json_object_iter_next(&it);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
json_object_object_add(software, "olsr1", software_olsr1);
|
||||
}
|
||||
|
||||
if (info->olsr2.enabled) {
|
||||
struct json_object *software_olsr2 = json_object_new_object();
|
||||
|
||||
json_object_object_add(software_olsr2, "running", json_object_new_boolean(info->olsr2.running));
|
||||
|
||||
if (info->olsr2.running) {
|
||||
struct json_object *version = olsr2_get_version();
|
||||
if (version)
|
||||
json_object_object_add(software_olsr2, "version", version);
|
||||
|
||||
struct json_object *addresses = olsr2_get_addresses();
|
||||
if (addresses) {
|
||||
json_object_object_add(software_olsr2, "addresses", addresses);
|
||||
|
||||
for (int i = 0; i < json_object_array_length(addresses); i++)
|
||||
json_object_array_add(n_addresses, json_object_array_get_idx(addresses, i));
|
||||
}
|
||||
|
||||
struct json_object *interfaces = olsr2_get_interfaces();
|
||||
if (interfaces) {
|
||||
json_object_object_add(software_olsr2, "interfaces", interfaces);
|
||||
|
||||
struct json_object_iterator it = json_object_iter_begin(interfaces);
|
||||
struct json_object_iterator itEnd = json_object_iter_end(interfaces);
|
||||
|
||||
while (!json_object_iter_equal(&it, &itEnd)) {
|
||||
const char * name = json_object_iter_peek_name(&it);
|
||||
json_object *append_key = json_object_object_get(n_interfaces, name);
|
||||
|
||||
if (!append_key) {
|
||||
append_key = json_object_new_object();
|
||||
json_object_object_add(n_interfaces, name, append_key);
|
||||
}
|
||||
|
||||
json_object_object_add(append_key, "olsr2",
|
||||
json_object_object_get(interfaces, name));
|
||||
json_object_iter_next(&it);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
json_object_object_add(software, "olsr2", software_olsr2);
|
||||
}
|
||||
|
||||
json_object_object_add(ret, "software", software);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
make_safe_fnc(respondd_provider_nodeinfo, real_respondd_provider_nodeinfo)
|
89
package/gluon-mesh-olsrd/src/respondd.c
Normal file
89
package/gluon-mesh-olsrd/src/respondd.c
Normal file
@ -0,0 +1,89 @@
|
||||
/*
|
||||
Copyright (c) 2021, Maciej Krüger <maciej@xeredo.it>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
#include "respondd-common.h"
|
||||
|
||||
#include <respondd.h>
|
||||
|
||||
#include <json-c/json.h>
|
||||
|
||||
#define RESP_SIZE 1024 * 1024 * 1024
|
||||
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
#include <sys/mman.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/wait.h>
|
||||
#include <unistd.h>
|
||||
|
||||
struct json_object * make_safe(struct json_object * (*fnc)(void)) {
|
||||
char * shared = mmap(NULL, RESP_SIZE, PROT_READ | PROT_WRITE,
|
||||
MAP_SHARED | MAP_ANONYMOUS, -1, 0);
|
||||
|
||||
pid_t pid = fork();
|
||||
|
||||
struct json_object *r = NULL;
|
||||
|
||||
if (pid == 0) {
|
||||
struct json_object *resp = fnc();
|
||||
if (!resp) {
|
||||
_exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
const char *resp_str = json_object_to_json_string(resp);
|
||||
size_t len = strlen(resp_str);
|
||||
if (len > RESP_SIZE) {
|
||||
_exit(EXIT_FAILURE);
|
||||
}
|
||||
|
||||
memcpy(shared, resp_str, len);
|
||||
_exit(EXIT_SUCCESS);
|
||||
} else {
|
||||
int status;
|
||||
|
||||
if (waitpid(pid, &status, 0) == -1) {
|
||||
goto ret;
|
||||
}
|
||||
|
||||
if (WIFEXITED(status)) {
|
||||
if (WEXITSTATUS(status) == 0) {
|
||||
r = json_tokener_parse(shared);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ret:
|
||||
munmap(shared, RESP_SIZE);
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
__attribute__ ((visibility ("default")))
|
||||
const struct respondd_provider_info respondd_providers[] = {
|
||||
{"nodeinfo", respondd_provider_nodeinfo},
|
||||
{"neighbours", respondd_provider_neighbours},
|
||||
{}
|
||||
};
|
281
package/gluon-mesh-olsrd/src/uclient.c
Normal file
281
package/gluon-mesh-olsrd/src/uclient.c
Normal file
@ -0,0 +1,281 @@
|
||||
/*
|
||||
Copyright (c) 2017, Jan-Philipp Litza <janphilipp@litza.de>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
|
||||
|
||||
#include "uclient.h"
|
||||
|
||||
#include <libubox/blobmsg.h>
|
||||
#include <libubox/uloop.h>
|
||||
|
||||
#include <limits.h>
|
||||
#include <stdio.h>
|
||||
|
||||
|
||||
#define TIMEOUT_MSEC 300000
|
||||
|
||||
static const char *const user_agent = "OLSRDHelper.so (using libuclient)";
|
||||
|
||||
|
||||
const char *uclient_get_errmsg(int code) {
|
||||
static char http_code_errmsg[16];
|
||||
if (code & UCLIENT_ERROR_STATUS_CODE) {
|
||||
snprintf(http_code_errmsg, 16, "HTTP error %d",
|
||||
code & (~UCLIENT_ERROR_STATUS_CODE));
|
||||
return http_code_errmsg;
|
||||
}
|
||||
switch(code) {
|
||||
case UCLIENT_ERROR_CONNECT:
|
||||
return "Connection failed";
|
||||
case UCLIENT_ERROR_TIMEDOUT:
|
||||
return "Connection timed out";
|
||||
case UCLIENT_ERROR_REDIRECT_FAILED:
|
||||
return "Failed to redirect";
|
||||
case UCLIENT_ERROR_TOO_MANY_REDIRECTS:
|
||||
return "Too many redirects";
|
||||
case UCLIENT_ERROR_CONNECTION_RESET_PREMATURELY:
|
||||
return "Connection reset prematurely";
|
||||
case UCLIENT_ERROR_SIZE_MISMATCH:
|
||||
return "Incorrect file size";
|
||||
case UCLIENT_ERROR_NOT_JSON:
|
||||
return "Response not json";
|
||||
default:
|
||||
return "Unknown error";
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void request_done(struct uclient *cl, int err_code) {
|
||||
uclient_data(cl)->err_code = err_code;
|
||||
uclient_disconnect(cl);
|
||||
uloop_end();
|
||||
}
|
||||
|
||||
|
||||
static void header_done_cb(struct uclient *cl) {
|
||||
const struct blobmsg_policy policy = {
|
||||
.name = "content-length",
|
||||
.type = BLOBMSG_TYPE_STRING,
|
||||
};
|
||||
struct blob_attr *tb_len;
|
||||
|
||||
if (uclient_data(cl)->retries < 10) {
|
||||
int ret = uclient_http_redirect(cl);
|
||||
if (ret < 0) {
|
||||
request_done(cl, UCLIENT_ERROR_REDIRECT_FAILED);
|
||||
return;
|
||||
}
|
||||
if (ret > 0) {
|
||||
uclient_data(cl)->retries++;
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
switch (cl->status_code) {
|
||||
case 200:
|
||||
break;
|
||||
case 301:
|
||||
case 302:
|
||||
case 307:
|
||||
request_done(cl, UCLIENT_ERROR_TOO_MANY_REDIRECTS);
|
||||
return;
|
||||
default:
|
||||
request_done(cl, UCLIENT_ERROR_STATUS_CODE | cl->status_code);
|
||||
return;
|
||||
}
|
||||
|
||||
blobmsg_parse(&policy, 1, &tb_len, blob_data(cl->meta), blob_len(cl->meta));
|
||||
if (tb_len) {
|
||||
char *endptr;
|
||||
|
||||
errno = 0;
|
||||
unsigned long long val = strtoull(blobmsg_get_string(tb_len), &endptr, 10);
|
||||
if (!errno && !*endptr && val <= SSIZE_MAX) {
|
||||
if (uclient_data(cl)->length >= 0 && uclient_data(cl)->length != (ssize_t)val) {
|
||||
request_done(cl, UCLIENT_ERROR_SIZE_MISMATCH);
|
||||
return;
|
||||
}
|
||||
|
||||
uclient_data(cl)->length = val;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static void eof_cb(struct uclient *cl) {
|
||||
request_done(cl, cl->data_eof ? 0 : UCLIENT_ERROR_CONNECTION_RESET_PREMATURELY);
|
||||
if (cl->data_eof) {
|
||||
printf("calling eof normally\n");
|
||||
uclient_data(cl)->eof(cl);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
ssize_t uclient_read_account(struct uclient *cl, char *buf, int len) {
|
||||
struct uclient_data *d = uclient_data(cl);
|
||||
int r = uclient_read(cl, buf, len);
|
||||
|
||||
if (r >= 0) {
|
||||
d->downloaded += r;
|
||||
|
||||
if (d->length >= 0 && d->downloaded > d->length) {
|
||||
request_done(cl, UCLIENT_ERROR_SIZE_MISMATCH);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
// src https://github.com/curl/curl/blob/2610142139d14265ed9acf9ed83cdf73d6bb4d05/lib/escape.c
|
||||
|
||||
/* Portable character check (remember EBCDIC). Do not use isalnum() because
|
||||
its behavior is altered by the current locale.
|
||||
See https://datatracker.ietf.org/doc/html/rfc3986#section-2.3
|
||||
*/
|
||||
bool Curl_isunreserved(unsigned char in)
|
||||
{
|
||||
switch(in) {
|
||||
case '0': case '1': case '2': case '3': case '4':
|
||||
case '5': case '6': case '7': case '8': case '9':
|
||||
case 'a': case 'b': case 'c': case 'd': case 'e':
|
||||
case 'f': case 'g': case 'h': case 'i': case 'j':
|
||||
case 'k': case 'l': case 'm': case 'n': case 'o':
|
||||
case 'p': case 'q': case 'r': case 's': case 't':
|
||||
case 'u': case 'v': case 'w': case 'x': case 'y': case 'z':
|
||||
case 'A': case 'B': case 'C': case 'D': case 'E':
|
||||
case 'F': case 'G': case 'H': case 'I': case 'J':
|
||||
case 'K': case 'L': case 'M': case 'N': case 'O':
|
||||
case 'P': case 'Q': case 'R': case 'S': case 'T':
|
||||
case 'U': case 'V': case 'W': case 'X': case 'Y': case 'Z':
|
||||
case '-': case '.': case '_': case '~':
|
||||
return true;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
char *curl_easy_escape(const char *string, int inlength)
|
||||
{
|
||||
size_t length;
|
||||
|
||||
if (inlength < 0)
|
||||
return NULL;
|
||||
|
||||
length = (inlength ? (size_t)inlength : strlen(string));
|
||||
if (!length)
|
||||
return strdup("");
|
||||
|
||||
char * out = malloc((length * 3) + 1);
|
||||
|
||||
if (!out)
|
||||
return NULL;
|
||||
|
||||
size_t offset = 0;
|
||||
|
||||
// this isn't pretending like we're complying to any spec other than urlencode, thx
|
||||
int slashes = 0;
|
||||
|
||||
while (length--) {
|
||||
unsigned char in = *string; /* we need to treat the characters unsigned */
|
||||
|
||||
if (slashes == 3) {
|
||||
if (Curl_isunreserved(in)) {
|
||||
/* append this */
|
||||
out[offset] = in;
|
||||
offset++;
|
||||
} else {
|
||||
/* encode it */
|
||||
if (snprintf(out + offset, 4, "%%%02X", in) != 3) {
|
||||
free(out);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
offset += 3;
|
||||
}
|
||||
} else {
|
||||
out[offset] = in;
|
||||
offset++;
|
||||
|
||||
if (in == '/') {
|
||||
slashes++;
|
||||
}
|
||||
}
|
||||
|
||||
string++;
|
||||
}
|
||||
|
||||
out[offset] = '\0';
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
int get_url(const char *user_url, void (*read_cb)(struct uclient *cl), void (*eof2_cb)(struct uclient *cl), void *cb_data, ssize_t len) {
|
||||
char *url = curl_easy_escape(user_url, 0);
|
||||
if (!url)
|
||||
return UCLIENT_ERROR_CONNECT;
|
||||
|
||||
struct uclient_data d = { .custom = cb_data, .length = len, .eof = eof2_cb };
|
||||
struct uclient_cb cb = {
|
||||
.header_done = header_done_cb,
|
||||
.data_read = read_cb,
|
||||
.data_eof = eof_cb,
|
||||
.error = request_done,
|
||||
};
|
||||
|
||||
struct uclient *cl = uclient_new(url, NULL, &cb);
|
||||
if (!cl)
|
||||
goto err;
|
||||
|
||||
cl->priv = &d;
|
||||
if (uclient_set_timeout(cl, TIMEOUT_MSEC))
|
||||
goto err;
|
||||
if (uclient_connect(cl))
|
||||
goto err;
|
||||
if (uclient_http_set_request_type(cl, "GET"))
|
||||
goto err;
|
||||
if (uclient_http_reset_headers(cl))
|
||||
goto err;
|
||||
if (uclient_http_set_header(cl, "User-Agent", user_agent))
|
||||
goto err;
|
||||
if (uclient_request(cl))
|
||||
goto err;
|
||||
uloop_run();
|
||||
uclient_free(cl);
|
||||
free(url);
|
||||
|
||||
if (!d.err_code && d.length >= 0 && d.downloaded != d.length)
|
||||
return UCLIENT_ERROR_SIZE_MISMATCH;
|
||||
|
||||
return d.err_code;
|
||||
|
||||
err:
|
||||
if (cl)
|
||||
uclient_free(cl);
|
||||
|
||||
free(url);
|
||||
|
||||
return UCLIENT_ERROR_CONNECT;
|
||||
}
|
64
package/gluon-mesh-olsrd/src/uclient.h
Normal file
64
package/gluon-mesh-olsrd/src/uclient.h
Normal file
@ -0,0 +1,64 @@
|
||||
/*
|
||||
Copyright (c) 2017, Jan-Philipp Litza <janphilipp@litza.de>
|
||||
All rights reserved.
|
||||
|
||||
Redistribution and use in source and binary forms, with or without
|
||||
modification, are permitted provided that the following conditions are met:
|
||||
|
||||
1. Redistributions of source code must retain the above copyright notice,
|
||||
this list of conditions and the following disclaimer.
|
||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
||||
this list of conditions and the following disclaimer in the documentation
|
||||
and/or other materials provided with the distribution.
|
||||
|
||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||||
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
||||
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
|
||||
#include <libubox/uclient.h>
|
||||
#include <sys/types.h>
|
||||
|
||||
|
||||
struct uclient_data {
|
||||
/* data that can be passed in by caller and used in custom callbacks */
|
||||
void *custom;
|
||||
/* data used by uclient callbacks */
|
||||
int retries;
|
||||
int err_code;
|
||||
ssize_t downloaded;
|
||||
ssize_t length;
|
||||
void (*eof)(struct uclient *cl);
|
||||
};
|
||||
|
||||
enum uclient_own_error_code {
|
||||
UCLIENT_ERROR_REDIRECT_FAILED = 32,
|
||||
UCLIENT_ERROR_TOO_MANY_REDIRECTS,
|
||||
UCLIENT_ERROR_CONNECTION_RESET_PREMATURELY,
|
||||
UCLIENT_ERROR_SIZE_MISMATCH,
|
||||
UCLIENT_ERROR_STATUS_CODE = 1024,
|
||||
UCLIENT_ERROR_NOT_JSON = 2048
|
||||
};
|
||||
|
||||
inline struct uclient_data * uclient_data(struct uclient *cl) {
|
||||
return (struct uclient_data *)cl->priv;
|
||||
}
|
||||
|
||||
inline void * uclient_get_custom(struct uclient *cl) {
|
||||
return uclient_data(cl)->custom;
|
||||
}
|
||||
|
||||
|
||||
ssize_t uclient_read_account(struct uclient *cl, char *buf, int len);
|
||||
|
||||
int get_url(const char *url, void (*read_cb)(struct uclient *cl), void (*eof2_cb)(struct uclient *cl), void *cb_data, ssize_t len);
|
||||
const char *uclient_get_errmsg(int code);
|
20
package/gluon-status-page-mesh-olsrd/Makefile
Normal file
20
package/gluon-status-page-mesh-olsrd/Makefile
Normal file
@ -0,0 +1,20 @@
|
||||
include $(TOPDIR)/rules.mk
|
||||
|
||||
PKG_NAME:=gluon-status-page-mesh-olsrd
|
||||
PKG_VERSION:=1
|
||||
|
||||
include ../gluon.mk
|
||||
|
||||
define Package/gluon-status-page-mesh-olsrd
|
||||
TITLE:=OLSRD-data provider for gluon-status-page
|
||||
DEPENDS:=+gluon-status-page +gluon-mesh-olsrd +libjson-c +libubox +libuclient
|
||||
endef
|
||||
|
||||
define Package/gluon-status-page-mesh-olsrd/install
|
||||
$(Gluon/Build/Install)
|
||||
|
||||
$(INSTALL_DIR) $(1)/lib/gluon/status-page/providers
|
||||
$(INSTALL_BIN) $(PKG_BUILD_DIR)/neighbours-olsrd $(1)/lib/gluon/status-page/providers/
|
||||
endef
|
||||
|
||||
$(eval $(call BuildPackageGluon,gluon-status-page-mesh-olsrd))
|
@ -0,0 +1,5 @@
|
||||
#!/bin/sh
|
||||
|
||||
CMD='exec /lib/gluon/status-page/providers/neighbours-olsrd'
|
||||
|
||||
exec /usr/sbin/sse-multiplex "$CMD"
|
@ -0,0 +1,10 @@
|
||||
return {
|
||||
provider = '/cgi-bin/dyn/neighbours-olsrd',
|
||||
-- List of mesh-specific attributes, each a tuple of
|
||||
-- 1) the internal identifier (JSON key)
|
||||
-- 2) human-readable key (not translatable yet)
|
||||
-- 3) value suffix (optional)
|
||||
attrs = {
|
||||
{'etx', 'Quality', ' '},
|
||||
},
|
||||
}
|
19
package/gluon-status-page-mesh-olsrd/src/Makefile
Normal file
19
package/gluon-status-page-mesh-olsrd/src/Makefile
Normal file
@ -0,0 +1,19 @@
|
||||
all: neighbours-olsrd
|
||||
|
||||
CFLAGS += -Wall -D_GNU_SOURCE
|
||||
|
||||
ifeq ($(origin PKG_CONFIG), undefined)
|
||||
PKG_CONFIG = pkg-config
|
||||
ifeq ($(shell which $(PKG_CONFIG) 2>/dev/null),)
|
||||
$(error $(PKG_CONFIG) not found)
|
||||
endif
|
||||
endif
|
||||
|
||||
CFLAGS_JSONC = $(shell $(PKG_CONFIG) --cflags json-c)
|
||||
LDFLAGS_JSONC = $(shell $(PKG_CONFIG) --libs json-c)
|
||||
|
||||
CFLAGS_OLSR += $(shell $(PKG_CONFIG) --cflags libolsrdhelper)
|
||||
LDFLAGS_OLSR += $(shell $(PKG_CONFIG) --libs libolsrdhelper)
|
||||
|
||||
neighbours-olsrd: neighbours-olsrd.c
|
||||
$(CC) $(CFLAGS) $(CFLAGS_JSONC) $(CFLAGS_OLSR) $(LDFLAGS) $(LDFLAGS_JSONC) $(LDFLAGS_OLSR) -o $@ $^ $(LDLIBS)
|
95
package/gluon-status-page-mesh-olsrd/src/neighbours-olsrd.c
Normal file
95
package/gluon-status-page-mesh-olsrd/src/neighbours-olsrd.c
Normal file
@ -0,0 +1,95 @@
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
#include <json-c/json.h>
|
||||
|
||||
#include <libubox/uclient.h>
|
||||
#include <libubox/blobmsg.h>
|
||||
#include <libubox/uloop.h>
|
||||
#include <libolsrdhelper.h>
|
||||
|
||||
static json_object *neighbours(void) {
|
||||
json_object *resp;
|
||||
if (olsr1_get_nodeinfo("links", &resp))
|
||||
return NULL;
|
||||
|
||||
json_object *out = json_object_new_object();
|
||||
if (!out)
|
||||
return NULL;
|
||||
|
||||
/*
|
||||
|
||||
links
|
||||
|
||||
localIP "10.12.11.43"
|
||||
remoteIP "10.12.11.1"
|
||||
olsrInterface "mesh-vpn"
|
||||
ifName "mesh-vpn"
|
||||
validityTime 141239
|
||||
symmetryTime 123095
|
||||
asymmetryTime 25552910
|
||||
vtime 124000
|
||||
currentLinkStatus "SYMMETRIC"
|
||||
previousLinkStatus "SYMMETRIC"
|
||||
hysteresis 0
|
||||
pending false
|
||||
lostLinkTime 0
|
||||
helloTime 0
|
||||
lastHelloTime 0
|
||||
seqnoValid false
|
||||
seqno 0
|
||||
lossHelloInterval 3000
|
||||
lossTime 3595
|
||||
lossMultiplier 65536
|
||||
linkCost 1.084961
|
||||
linkQuality 1
|
||||
neighborLinkQuality 0.921
|
||||
|
||||
*/
|
||||
|
||||
// TODO: use olsr1_get_neigh and olsr2_get_neigh, iterate over both, then copy stuffs into the right format (and use mac as primary)
|
||||
|
||||
json_object *links = json_object_object_get(resp, "links");
|
||||
if (!links)
|
||||
return NULL;
|
||||
|
||||
int linkcount = json_object_array_length(links);
|
||||
|
||||
for (int i = 0; i < linkcount; i++) {
|
||||
struct json_object *link = json_object_array_get_idx(links, i);
|
||||
if (!link)
|
||||
return NULL;
|
||||
|
||||
struct json_object *neigh = json_object_new_object();
|
||||
if (!neigh)
|
||||
return NULL;
|
||||
|
||||
json_object_object_add(neigh, "ifname", json_object_object_get(link, "ifName"));
|
||||
// TODO: do we need this? should we set this? (we could pick the one peer that we currently route 0.0.0.0 over...)
|
||||
json_object_object_add(neigh, "best", json_object_new_boolean(0));
|
||||
json_object_object_add(neigh, "etx", json_object_object_get(link, "etx"));
|
||||
|
||||
json_object_object_add(out, json_object_get_string(json_object_object_get(link, "remoteIP")), neigh);
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
int main(void) {
|
||||
struct json_object *obj;
|
||||
|
||||
printf("Content-type: text/event-stream\n\n");
|
||||
fflush(stdout);
|
||||
|
||||
while (1) {
|
||||
obj = neighbours();
|
||||
if (obj) {
|
||||
printf("data: %s\n\n", json_object_to_json_string_ext(obj, JSON_C_TO_STRING_PLAIN));
|
||||
fflush(stdout);
|
||||
json_object_put(obj);
|
||||
}
|
||||
sleep(10);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
@ -0,0 +1,28 @@
|
||||
From e5c0192003d6a0afee97abb9a947bc0944883091 Mon Sep 17 00:00:00 2001
|
||||
From: =?UTF-8?q?Maciej=20Kr=C3=BCger?= <mkg20001@gmail.com>
|
||||
Date: Sun, 10 Apr 2022 01:58:41 +0200
|
||||
Subject: [PATCH] oonf-olsrd2: add support to check if service is running
|
||||
|
||||
---
|
||||
oonf-olsrd2/files/olsrd2.init | 7 +++++++
|
||||
1 file changed, 7 insertions(+)
|
||||
|
||||
diff --git a/oonf-olsrd2/files/olsrd2.init b/oonf-olsrd2/files/olsrd2.init
|
||||
index debae98..b6c1e9a 100755
|
||||
--- a/oonf-olsrd2/files/olsrd2.init
|
||||
+++ b/oonf-olsrd2/files/olsrd2.init
|
||||
@@ -3,4 +3,11 @@
|
||||
START=82
|
||||
DAEMON='olsrd2'
|
||||
|
||||
+running() {
|
||||
+ test -e "/tmp/run/olsrd2.pid" && test -e "/proc/$(cat "/tmp/run/olsrd2.pid")" && return 0
|
||||
+ return 1
|
||||
+}
|
||||
+
|
||||
+extra_command "running" "Check if service is running"
|
||||
+
|
||||
. /lib/functions/oonf_init.sh
|
||||
--
|
||||
2.35.1
|
||||
|
Loading…
Reference in New Issue
Block a user