1
0
mirror of https://github.com/kepler155c/opus synced 2025-01-22 05:06:53 +00:00
opus/sys/apps/network/snmp.lua

224 lines
5.0 KiB
Lua
Raw Normal View History

local Event = require('opus.event')
local GPS = require('opus.gps')
local Socket = require('opus.socket')
local Util = require('opus.util')
2016-12-11 19:24:52 +00:00
2018-01-15 21:28:10 +00:00
local device = _G.device
local kernel = _G.kernel
local network = _G.network
local os = _G.os
local turtle = _G.turtle
2017-10-14 07:41:54 +00:00
2016-12-11 19:24:52 +00:00
-- move this into gps api
local gpsRequested
local gpsLastPoint
local gpsLastRequestTime
local function snmpConnection(socket)
2018-01-24 22:39:38 +00:00
while true do
local msg = socket:read()
if not msg then
break
end
if msg.type == 'reboot' then
os.reboot()
elseif msg.type == 'shutdown' then
os.shutdown()
elseif msg.type == 'ping' then
socket:write('pong')
elseif msg.type == 'script' then
2018-04-01 22:40:08 +00:00
local env = setmetatable(Util.shallowCopy(_ENV), { __index = _G })
local fn, err = load(msg.args, 'script', nil, env)
2018-01-24 22:39:38 +00:00
if fn then
kernel.run({
fn = fn,
2018-04-01 22:40:08 +00:00
env = env,
2018-01-24 22:39:38 +00:00
title = 'script',
})
else
_G.printError(err)
end
elseif msg.type == 'scriptEx' then
local s, m = pcall(function()
local env = setmetatable(Util.shallowCopy(_ENV), { __index = _G })
local fn, m = load(msg.args, 'script', nil, env)
if not fn then
error(m)
end
return { fn() }
end)
if s then
socket:write(m)
else
socket:write({ s, m })
end
elseif msg.type == 'gps' then
if gpsRequested then
repeat
os.sleep(0)
until not gpsRequested
end
if gpsLastPoint and os.clock() - gpsLastRequestTime < .5 then
socket:write(gpsLastPoint)
else
gpsRequested = true
local pt = GPS.getPoint(2)
if pt then
socket:write(pt)
else
print('snmp: Unable to get GPS point')
end
gpsRequested = false
gpsLastPoint = pt
if pt then
gpsLastRequestTime = os.clock()
end
end
elseif msg.type == 'info' then
local info = {
id = os.getComputerID(),
label = os.getComputerLabel(),
uptime = math.floor(os.clock()),
}
if turtle then
info.fuel = turtle.getFuelLevel()
info.status = turtle.getStatus()
end
socket:write(info)
end
end
2016-12-11 19:24:52 +00:00
end
2017-08-03 05:46:39 +00:00
Event.addRoutine(function()
2018-01-24 22:39:38 +00:00
print('snmp: listening on port 161')
2016-12-11 19:24:52 +00:00
2018-01-24 22:39:38 +00:00
while true do
local socket = Socket.server(161)
2016-12-11 19:24:52 +00:00
2018-01-24 22:39:38 +00:00
Event.addRoutine(function()
print('snmp: connection from ' .. socket.dhost)
2019-04-20 17:48:13 +00:00
local s, m = pcall(snmpConnection, socket)
2018-01-24 22:39:38 +00:00
print('snmp: closing connection to ' .. socket.dhost)
2019-04-20 17:48:13 +00:00
if not s and m then
print('snmp error')
_G.printError(m)
end
2018-01-24 22:39:38 +00:00
end)
end
2016-12-11 19:24:52 +00:00
end)
2017-08-03 05:46:39 +00:00
device.wireless_modem.open(999)
print('discovery: listening on port 999')
2016-12-11 19:24:52 +00:00
2017-10-14 07:41:54 +00:00
Event.on('modem_message', function(_, _, sport, id, info, distance)
2018-01-24 22:39:38 +00:00
if sport == 999 and tonumber(id) and type(info) == 'table' then
2019-07-08 17:52:53 +00:00
if type(info.label) == 'string' and type(info.id) == 'number' then
2018-01-24 22:39:38 +00:00
2019-05-20 19:19:39 +00:00
if not network[id] then
network[id] = { }
end
Util.merge(network[id], info)
network[id].distance = type(distance) == 'number' and distance
network[id].timestamp = os.clock()
2019-04-20 16:34:45 +00:00
2019-05-20 19:19:39 +00:00
if not network[id].label then
network[id].label = 'unknown'
end
if not network[id].active then
network[id].active = true
os.queueEvent('network_attach', network[id])
end
else
print('discovery: Invalid alive message ' .. id)
2018-01-24 22:39:38 +00:00
end
end
2016-12-11 19:24:52 +00:00
end)
2016-12-23 04:22:04 +00:00
local info = {
2018-01-24 22:39:38 +00:00
id = os.getComputerID()
2016-12-23 04:22:04 +00:00
}
2017-09-21 05:44:33 +00:00
local infoTimer = os.clock()
2016-12-23 04:22:04 +00:00
2019-07-08 17:52:53 +00:00
local function getSlots()
return Util.reduce(turtle.getInventory(), function(acc, v)
if v.count > 0 then
acc[v.index .. ',' .. v.count] = v.key
end
return acc
end, { })
end
2016-12-11 19:24:52 +00:00
local function sendInfo()
2018-01-24 22:39:38 +00:00
if os.clock() - infoTimer >= 1 then -- don't flood
infoTimer = os.clock()
info.label = os.getComputerLabel()
info.uptime = math.floor(os.clock())
info.group = network.getGroup()
2019-06-18 19:19:24 +00:00
if turtle and turtle.getStatus then
2018-01-24 22:39:38 +00:00
info.fuel = turtle.getFuelLevel()
info.status = turtle.getStatus()
info.point = turtle.point
2019-07-08 17:52:53 +00:00
info.inv = getSlots()
2018-01-24 22:39:38 +00:00
info.slotIndex = turtle.getSelectedSlot()
end
2018-02-09 13:01:02 +00:00
if device.neuralInterface then
info.status = device.neuralInterface.status
2019-03-06 16:48:09 +00:00
if not info.status and device.neuralInterface.getMetaOwner then
pcall(function()
2019-03-01 19:16:55 +00:00
local meta = device.neuralInterface.getMetaOwner()
2019-07-08 17:52:53 +00:00
local states = {
isWet = 'Swimming',
isElytraFlying = 'Flying',
isBurning = 'Burning',
isDead = 'Deceased',
isOnLadder = 'Climbing',
isRiding = 'Riding',
isSneaking = 'Sneaking',
isSprinting = 'Running',
}
for k,v in pairs(states) do
if meta[k] then
info.status = v
break
end
2019-03-01 19:16:55 +00:00
end
2019-07-08 17:52:53 +00:00
info.status = info.status or 'health: ' ..
math.floor(meta.health / meta.maxHealth * 100)
2019-03-06 16:48:09 +00:00
end)
end
2018-02-09 13:01:02 +00:00
end
2018-01-24 22:39:38 +00:00
device.wireless_modem.transmit(999, os.getComputerID(), info)
end
2016-12-11 19:24:52 +00:00
end
-- every 10 seconds, send out this computer's info
2017-08-03 05:46:39 +00:00
Event.onInterval(10, function()
2018-01-24 22:39:38 +00:00
sendInfo()
for _,c in pairs(_G.network) do
local elapsed = os.clock()-c.timestamp
if c.active and elapsed > 15 then
c.active = false
os.queueEvent('network_detach', c)
end
end
2016-12-11 19:24:52 +00:00
end)
2017-08-03 05:46:39 +00:00
Event.on('turtle_response', function()
2018-01-24 22:39:38 +00:00
if turtle.getStatus() ~= info.status or
turtle.fuel ~= info.fuel then
sendInfo()
end
2017-08-03 05:46:39 +00:00
end)
2019-07-08 04:16:28 +00:00
2019-07-08 17:52:53 +00:00
Event.onTimeout(1, sendInfo)