mirror of
https://github.com/kepler155c/opus
synced 2025-01-12 08:40:26 +00:00
socket improvement
This commit is contained in:
parent
ab73e3f2f9
commit
f866d2bd58
@ -16,7 +16,7 @@ mon.setBackgroundColor(colors.black)
|
|||||||
mon.clear()
|
mon.clear()
|
||||||
|
|
||||||
while true do
|
while true do
|
||||||
local socket = Socket.server(5901, true)
|
local socket = Socket.server(5901)
|
||||||
|
|
||||||
print('mirror: connection from ' .. socket.dhost)
|
print('mirror: connection from ' .. socket.dhost)
|
||||||
|
|
||||||
|
@ -46,13 +46,14 @@ process:newThread('telnet_read', function()
|
|||||||
ct[v.f](unpack(v.args))
|
ct[v.f](unpack(v.args))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
print('telnet_read exiting')
|
||||||
end)
|
end)
|
||||||
|
|
||||||
ct.clear()
|
ct.clear()
|
||||||
ct.setCursorPos(1, 1)
|
ct.setCursorPos(1, 1)
|
||||||
|
|
||||||
while true do
|
while true do
|
||||||
local e = { process:pullEvent() }
|
local e = { process:pullEvent(nil, true) }
|
||||||
local event = e[1]
|
local event = e[1]
|
||||||
|
|
||||||
if not socket.connected then
|
if not socket.connected then
|
||||||
@ -71,10 +72,11 @@ while true do
|
|||||||
event == 'mouse_click' or
|
event == 'mouse_click' or
|
||||||
event == 'mouse_drag' then
|
event == 'mouse_drag' then
|
||||||
|
|
||||||
socket:write({
|
if not socket:write({ type = 'shellRemote', event = e }) then
|
||||||
type = 'shellRemote',
|
socket:close()
|
||||||
event = e,
|
break
|
||||||
})
|
end
|
||||||
|
|
||||||
elseif event == 'terminate' then
|
elseif event == 'terminate' then
|
||||||
socket:close()
|
socket:close()
|
||||||
break
|
break
|
||||||
|
@ -53,7 +53,7 @@ local secretKey = os.getSecretKey()
|
|||||||
local publicKey = modexp(exchange.base, secretKey, exchange.primeMod)
|
local publicKey = modexp(exchange.base, secretKey, exchange.primeMod)
|
||||||
local password = SHA1.sha1(password)
|
local password = SHA1.sha1(password)
|
||||||
|
|
||||||
socket:write(Crypto.encrypt({ pk = publicKey }, password))
|
socket:write(Crypto.encrypt({ pk = publicKey, dh = os.getComputerID() }, password))
|
||||||
|
|
||||||
print(socket:read(2) or 'No response')
|
print(socket:read(2) or 'No response')
|
||||||
|
|
||||||
|
@ -298,6 +298,7 @@ function standardBlockDB:seedDB()
|
|||||||
[ '86:0' ] = 'pumpkin',
|
[ '86:0' ] = 'pumpkin',
|
||||||
[ '90:0' ] = 'air',
|
[ '90:0' ] = 'air',
|
||||||
[ '91:0' ] = 'pumpkin',
|
[ '91:0' ] = 'pumpkin',
|
||||||
|
[ '92:0' ] = 'flatten', -- cake
|
||||||
[ '93:0' ] = 'repeater',
|
[ '93:0' ] = 'repeater',
|
||||||
[ '94:0' ] = 'repeater',
|
[ '94:0' ] = 'repeater',
|
||||||
[ '96:0' ] = 'trapdoor',
|
[ '96:0' ] = 'trapdoor',
|
||||||
|
@ -31,41 +31,29 @@ function socketClass:read(timeout)
|
|||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
local timerId
|
local data, distance = transport.read(self)
|
||||||
local filter
|
if data then
|
||||||
|
return data, distance
|
||||||
if timeout then
|
|
||||||
timerId = os.startTimer(timeout)
|
|
||||||
elseif self.keepAlive then
|
|
||||||
timerId = os.startTimer(3)
|
|
||||||
else
|
|
||||||
filter = 'modem_message'
|
|
||||||
end
|
end
|
||||||
|
|
||||||
while true do
|
local timerId = os.startTimer(timeout or 5)
|
||||||
local e, s, dport, dhost, msg, distance = os.pullEvent(filter)
|
|
||||||
if e == 'modem_message' and
|
|
||||||
dport == self.sport and dhost == self.shost and
|
|
||||||
msg then
|
|
||||||
|
|
||||||
if msg.type == 'DISC' then
|
while true do
|
||||||
-- received disconnect from other end
|
local e, id = os.pullEvent()
|
||||||
self.connected = false
|
|
||||||
self:close()
|
if e == 'transport_' .. self.dport then
|
||||||
return
|
|
||||||
elseif msg.type == 'DATA' then
|
data, distance = transport.read(self)
|
||||||
if msg.data then
|
if data then
|
||||||
if timerId then
|
os.cancelTimer(timerId)
|
||||||
os.cancelTimer(timerId)
|
return data, distance
|
||||||
end
|
|
||||||
return msg.data, distance
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
elseif e == 'timer' and s == timerId then
|
|
||||||
|
elseif e == 'timer' and id == timerId then
|
||||||
if timeout or not self.connected then
|
if timeout or not self.connected then
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
timerId = os.startTimer(3)
|
timerId = os.startTimer(5)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -75,8 +63,9 @@ function socketClass:write(data)
|
|||||||
Logger.log('socket', 'write: No connection')
|
Logger.log('socket', 'write: No connection')
|
||||||
return false
|
return false
|
||||||
end
|
end
|
||||||
self.transmit(self.dport, self.dhost, {
|
transport.write(self, {
|
||||||
type = 'DATA',
|
type = 'DATA',
|
||||||
|
seq = self.wseq,
|
||||||
data = data,
|
data = data,
|
||||||
})
|
})
|
||||||
return true
|
return true
|
||||||
@ -91,45 +80,7 @@ function socketClass:close()
|
|||||||
self.connected = false
|
self.connected = false
|
||||||
end
|
end
|
||||||
device.wireless_modem.close(self.sport)
|
device.wireless_modem.close(self.sport)
|
||||||
end
|
transport.close(self)
|
||||||
|
|
||||||
-- write a ping every second (too much traffic!)
|
|
||||||
local function pinger(socket)
|
|
||||||
|
|
||||||
local process = require('process')
|
|
||||||
|
|
||||||
socket.keepAlive = true
|
|
||||||
|
|
||||||
Logger.log('socket', 'keepAlive enabled')
|
|
||||||
|
|
||||||
process:newThread('socket_ping', function()
|
|
||||||
local timerId = os.startTimer(1)
|
|
||||||
local timeStamp = os.clock()
|
|
||||||
|
|
||||||
while true do
|
|
||||||
local e, id, dport, dhost, msg = os.pullEvent()
|
|
||||||
|
|
||||||
if e == 'modem_message' then
|
|
||||||
if dport == socket.sport and
|
|
||||||
dhost == socket.shost and
|
|
||||||
msg and
|
|
||||||
msg.type == 'PING' then
|
|
||||||
|
|
||||||
timeStamp = os.clock()
|
|
||||||
end
|
|
||||||
elseif e == 'timer' and id == timerId then
|
|
||||||
if os.clock() - timeStamp > 3 then
|
|
||||||
Logger.log('socket', 'Connection timed out')
|
|
||||||
socket:close()
|
|
||||||
break
|
|
||||||
end
|
|
||||||
timerId = os.startTimer(1)
|
|
||||||
socket.transmit(socket.dport, socket.dhost, {
|
|
||||||
type = 'PING',
|
|
||||||
})
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
local Socket = { }
|
local Socket = { }
|
||||||
@ -139,12 +90,16 @@ local function loopback(port, sport, msg)
|
|||||||
end
|
end
|
||||||
|
|
||||||
local function newSocket(isLoopback)
|
local function newSocket(isLoopback)
|
||||||
for i = 16384, 32768 do
|
for i = 16384, 32767 do
|
||||||
if not device.wireless_modem.isOpen(i) then
|
if not device.wireless_modem.isOpen(i) then
|
||||||
local socket = {
|
local socket = {
|
||||||
shost = os.getComputerID(),
|
shost = os.getComputerID(),
|
||||||
sport = i,
|
sport = i,
|
||||||
transmit = device.wireless_modem.transmit,
|
transmit = device.wireless_modem.transmit,
|
||||||
|
wseq = math.random(100, 100000),
|
||||||
|
rseq = math.random(100, 100000),
|
||||||
|
timers = { },
|
||||||
|
messages = { },
|
||||||
}
|
}
|
||||||
setmetatable(socket, { __index = socketClass })
|
setmetatable(socket, { __index = socketClass })
|
||||||
|
|
||||||
@ -169,7 +124,9 @@ function Socket.connect(host, port)
|
|||||||
type = 'OPEN',
|
type = 'OPEN',
|
||||||
shost = socket.shost,
|
shost = socket.shost,
|
||||||
dhost = socket.dhost,
|
dhost = socket.dhost,
|
||||||
t = Crypto.encrypt({ ts = os.time() }, exchange.publicKey),
|
t = Crypto.encrypt({ ts = os.time(), seq = socket.seq }, exchange.publicKey),
|
||||||
|
rseq = socket.wseq,
|
||||||
|
wseq = socket.rseq,
|
||||||
})
|
})
|
||||||
|
|
||||||
local timerId = os.startTimer(3)
|
local timerId = os.startTimer(3)
|
||||||
@ -185,11 +142,10 @@ function Socket.connect(host, port)
|
|||||||
Logger.log('socket', 'connection established to %d %d->%d',
|
Logger.log('socket', 'connection established to %d %d->%d',
|
||||||
host, socket.sport, socket.dport)
|
host, socket.sport, socket.dport)
|
||||||
|
|
||||||
if msg.keepAlive then
|
|
||||||
pinger(socket)
|
|
||||||
end
|
|
||||||
os.cancelTimer(timerId)
|
os.cancelTimer(timerId)
|
||||||
|
|
||||||
|
transport.open(socket)
|
||||||
|
|
||||||
return socket
|
return socket
|
||||||
end
|
end
|
||||||
until e == 'timer' and id == timerId
|
until e == 'timer' and id == timerId
|
||||||
@ -199,7 +155,8 @@ end
|
|||||||
|
|
||||||
function trusted(msg, port)
|
function trusted(msg, port)
|
||||||
|
|
||||||
if port == 19 then -- no auth for trust server
|
if port == 19 or msg.shost == os.getComputerID() then
|
||||||
|
-- no auth for trust server or loopback
|
||||||
return true
|
return true
|
||||||
end
|
end
|
||||||
|
|
||||||
@ -214,7 +171,7 @@ function trusted(msg, port)
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
function Socket.server(port, keepAlive)
|
function Socket.server(port)
|
||||||
|
|
||||||
device.wireless_modem.open(port)
|
device.wireless_modem.open(port)
|
||||||
Logger.log('socket', 'Waiting for connections on port ' .. port)
|
Logger.log('socket', 'Waiting for connections on port ' .. port)
|
||||||
@ -232,18 +189,16 @@ function Socket.server(port, keepAlive)
|
|||||||
socket.dport = dport
|
socket.dport = dport
|
||||||
socket.dhost = msg.shost
|
socket.dhost = msg.shost
|
||||||
socket.connected = true
|
socket.connected = true
|
||||||
|
socket.wseq = msg.wseq
|
||||||
|
socket.rseq = msg.rseq
|
||||||
socket.transmit(socket.dport, socket.sport, {
|
socket.transmit(socket.dport, socket.sport, {
|
||||||
type = 'CONN',
|
type = 'CONN',
|
||||||
dhost = socket.dhost,
|
dhost = socket.dhost,
|
||||||
shost = socket.shost,
|
shost = socket.shost,
|
||||||
keepAlive = keepAlive,
|
|
||||||
})
|
})
|
||||||
Logger.log('socket', 'Connection established %d->%d', socket.sport, socket.dport)
|
Logger.log('socket', 'Connection established %d->%d', socket.sport, socket.dport)
|
||||||
|
|
||||||
if keepAlive then
|
transport.open(socket)
|
||||||
pinger(socket)
|
|
||||||
end
|
|
||||||
return socket
|
return socket
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -21,7 +21,7 @@ function Util.tryTimes(attempts, f, ...)
|
|||||||
return unpack(result)
|
return unpack(result)
|
||||||
end
|
end
|
||||||
|
|
||||||
function Util.print(pattern, ...)
|
function Util.tostring(pattern, ...)
|
||||||
|
|
||||||
local function serialize(tbl, width)
|
local function serialize(tbl, width)
|
||||||
local str = '{\n'
|
local str = '{\n'
|
||||||
@ -43,12 +43,15 @@ function Util.print(pattern, ...)
|
|||||||
end
|
end
|
||||||
|
|
||||||
if type(pattern) == 'string' then
|
if type(pattern) == 'string' then
|
||||||
print(string.format(pattern, ...))
|
return string.format(pattern, ...)
|
||||||
elseif type(pattern) == 'table' then
|
elseif type(pattern) == 'table' then
|
||||||
print(serialize(pattern, term.current().getSize()))
|
return serialize(pattern, term.current().getSize())
|
||||||
else
|
|
||||||
print(tostring(pattern))
|
|
||||||
end
|
end
|
||||||
|
return tostring(pattern)
|
||||||
|
end
|
||||||
|
|
||||||
|
function Util.print(pattern, ...)
|
||||||
|
print(Util.tostring(pattern, ...))
|
||||||
end
|
end
|
||||||
|
|
||||||
function Util.runFunction(env, fn, ...)
|
function Util.runFunction(env, fn, ...)
|
||||||
|
@ -101,7 +101,7 @@ process:newThread('discovery_server', function()
|
|||||||
while true do
|
while true do
|
||||||
local e, s, sport, id, info, distance = os.pullEvent('modem_message')
|
local e, s, sport, id, info, distance = os.pullEvent('modem_message')
|
||||||
|
|
||||||
if sport == 999 then
|
if sport == 999 and tonumber(id) and type(info) == 'table' then
|
||||||
if not network[id] then
|
if not network[id] then
|
||||||
network[id] = { }
|
network[id] = { }
|
||||||
end
|
end
|
||||||
|
@ -40,7 +40,7 @@ local function telnetHost(socket, termInfo)
|
|||||||
socket:close()
|
socket:close()
|
||||||
end)
|
end)
|
||||||
|
|
||||||
local queueThread = process:newThread('telnet_read', function()
|
process:newThread('telnet_read', function()
|
||||||
while true do
|
while true do
|
||||||
local data = socket:read()
|
local data = socket:read()
|
||||||
if not data then
|
if not data then
|
||||||
@ -49,7 +49,6 @@ local function telnetHost(socket, termInfo)
|
|||||||
|
|
||||||
if data.type == 'shellRemote' then
|
if data.type == 'shellRemote' then
|
||||||
local event = table.remove(data.event, 1)
|
local event = table.remove(data.event, 1)
|
||||||
|
|
||||||
shellThread:resume(event, unpack(data.event))
|
shellThread:resume(event, unpack(data.event))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -65,7 +64,10 @@ local function telnetHost(socket, termInfo)
|
|||||||
break
|
break
|
||||||
end
|
end
|
||||||
if socket.queue then
|
if socket.queue then
|
||||||
socket:write(socket.queue)
|
if not socket:write(socket.queue) then
|
||||||
|
print('telnet: connection lost to ' .. socket.dhost)
|
||||||
|
break
|
||||||
|
end
|
||||||
socket.queue = nil
|
socket.queue = nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -17,9 +17,9 @@ process:newThread('trust_server', function()
|
|||||||
socket:write('No password has been set')
|
socket:write('No password has been set')
|
||||||
else
|
else
|
||||||
data = Crypto.decrypt(data, password)
|
data = Crypto.decrypt(data, password)
|
||||||
if data and data.pk then
|
if data and data.pk and data.dh then
|
||||||
local trustList = Util.readTable('.known_hosts') or { }
|
local trustList = Util.readTable('.known_hosts') or { }
|
||||||
trustList[socket.dhost] = data.pk
|
trustList[data.dh] = data.pk
|
||||||
Util.writeTable('.known_hosts', trustList)
|
Util.writeTable('.known_hosts', trustList)
|
||||||
|
|
||||||
socket:write('Trust accepted')
|
socket:write('Trust accepted')
|
||||||
|
97
sys/services/transort.lua
Normal file
97
sys/services/transort.lua
Normal file
@ -0,0 +1,97 @@
|
|||||||
|
--[[
|
||||||
|
Low level socket protocol implementation.
|
||||||
|
|
||||||
|
* sequencing
|
||||||
|
* write acknowledgements
|
||||||
|
* background read buffering
|
||||||
|
]]--
|
||||||
|
|
||||||
|
multishell.setTitle(multishell.getCurrent(), 'Net transport')
|
||||||
|
|
||||||
|
local computerId = os.getComputerID()
|
||||||
|
|
||||||
|
_G.transport = {
|
||||||
|
timers = { },
|
||||||
|
sockets = { },
|
||||||
|
}
|
||||||
|
|
||||||
|
function transport.open(socket)
|
||||||
|
transport.sockets[socket.sport] = socket
|
||||||
|
end
|
||||||
|
|
||||||
|
function transport.read(socket)
|
||||||
|
local data = table.remove(socket.messages, 1)
|
||||||
|
if data then
|
||||||
|
return unpack(data)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function transport.write(socket, data)
|
||||||
|
|
||||||
|
--debug('>> ' .. Util.tostring({ type = 'DATA', seq = socket.wseq }))
|
||||||
|
socket.transmit(socket.dport, socket.dhost, data)
|
||||||
|
|
||||||
|
local timerId = os.startTimer(2)
|
||||||
|
|
||||||
|
transport.timers[timerId] = socket
|
||||||
|
socket.timers[socket.wseq] = timerId
|
||||||
|
|
||||||
|
socket.wseq = socket.wseq + 1
|
||||||
|
end
|
||||||
|
|
||||||
|
function transport.close(socket)
|
||||||
|
transport.sockets[socket.sport] = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
print('Net transport started')
|
||||||
|
|
||||||
|
while true do
|
||||||
|
local e, timerId, dport, dhost, msg, distance = os.pullEvent()
|
||||||
|
|
||||||
|
if e == 'timer' then
|
||||||
|
local socket = transport.timers[timerId]
|
||||||
|
if socket and socket.connected then
|
||||||
|
socket:close()
|
||||||
|
transport.timers[timerId] = nil
|
||||||
|
end
|
||||||
|
|
||||||
|
elseif e == 'modem_message' and dhost == computerId and msg then
|
||||||
|
local socket = transport.sockets[dport]
|
||||||
|
if socket and socket.connected then
|
||||||
|
|
||||||
|
--if msg.type then debug('<< ' .. Util.tostring(msg)) end
|
||||||
|
|
||||||
|
if msg.type == 'DISC' then
|
||||||
|
-- received disconnect from other end
|
||||||
|
socket.connected = false
|
||||||
|
socket:close()
|
||||||
|
|
||||||
|
elseif msg.type == 'ACK' then
|
||||||
|
local timerId = socket.timers[msg.seq]
|
||||||
|
|
||||||
|
os.cancelTimer(timerId)
|
||||||
|
socket.timers[msg.seq] = nil
|
||||||
|
transport.timers[timerId] = nil
|
||||||
|
|
||||||
|
elseif msg.type == 'DATA' and msg.data then
|
||||||
|
if msg.seq ~= socket.rseq then
|
||||||
|
socket:close()
|
||||||
|
else
|
||||||
|
socket.rseq = socket.rseq + 1
|
||||||
|
table.insert(socket.messages, { msg.data, distance })
|
||||||
|
|
||||||
|
-- use resume instead ??
|
||||||
|
if not socket.messages[2] then -- table size is 1
|
||||||
|
os.queueEvent('transport_' .. dport)
|
||||||
|
end
|
||||||
|
|
||||||
|
--debug('>> ' .. Util.tostring({ type = 'ACK', seq = msg.seq }))
|
||||||
|
socket.transmit(socket.dport, socket.dhost, {
|
||||||
|
type = 'ACK',
|
||||||
|
seq = msg.seq,
|
||||||
|
})
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in New Issue
Block a user