Files
Techmino/Zframework/websocket.lua
2021-03-30 01:36:25 +08:00

321 lines
6.6 KiB
Lua

-- local host="192.168.114.102"
-- local host="127.0.0.1"
local host="hdustea.3322.org"
local port="10026"
local path="/tech/socket/v1"
local wsThread=[[
-- lua + love2d threading websocket client
-- Original pure lua ver. by flaribbit and Particle_G and MrZ
-- Threading version by MrZ
local triggerCHN,sendCHN,readCHN=...
local byte,char=string.byte,string.char
local band,bor,bxor=bit.band,bit.bor,bit.bxor
local shl,shr=bit.lshift,bit.rshift
local SOCK=require"socket".tcp()
local JSON=require"Zframework.json"
local mask_key={1,14,5,14}
local function _send(opcode,message)
--Message type
SOCK:send(char(bor(0x80,opcode)))
if not message then
SOCK:send(char(0x80,unpack(mask_key)))
return 0
end
--Length
local length=#message
if length>65535 then
SOCK:send(char(bor(127,0x80),0,0,0,0,band(shr(length,24),0xff),band(shr(length,16),0xff),band(shr(length,8),0xff),band(length,0xff)))
elseif length>125 then
SOCK:send(char(bor(126,0x80),band(shr(length,8),0xff),band(length,0xff)))
else
SOCK:send(char(bor(length,0x80)))
end
SOCK:send(char(unpack(mask_key)))
local msgbyte={byte(message,1,length)}
for i=1,length do
msgbyte[i]=bxor(msgbyte[i],mask_key[(i-1)%4+1])
end
return SOCK:send(char(unpack(msgbyte)))
end
do--Connect
local host=sendCHN:demand()
local port=sendCHN:demand()
local path=sendCHN:demand()
local body=sendCHN:demand()
SOCK:settimeout(2.6)
local res,err=SOCK:connect(host,port)
if res then
--WebSocket handshake
if not body then body=""end
SOCK:send(
"GET "..path.." HTTP/1.1\r\n"..
"Host: "..host..":"..port.."\r\n"..
"Connection: Upgrade\r\n"..
"Upgrade: websocket\r\n"..
"Content-Type: application/json\r\n"..
"Content-Length: "..#body.."\r\n"..
"Sec-WebSocket-Version: 13\r\n"..
"Sec-WebSocket-Key: osT3F7mvlojIvf3/8uIsJQ==\r\n\r\n"..--secKey
body
)
--First line of HTTP
local l=SOCK:receive("*l")
local code,ctLen
if l then
code=l:find(" "); code=l:sub(code+1,code+3)
repeat
l=SOCK:receive("*l")
if not ctLen and l:find"length"then
ctLen=tonumber(l:match"%d+")
end
until l==""
end
if code=="101"then
readCHN:push("success")
else
local reason=JSON.decode(SOCK:receive(ctLen))
readCHN:push(code..":"..(reason and reason.message or"Server Error"))
end
else
readCHN:push(err)
end
SOCK:settimeout(6.26)
end
local buffer
while true do--Running
triggerCHN:demand()
while sendCHN:getCount()>=2 do
local op=sendCHN:pop()
local message=sendCHN:pop()
_send(op,message)
end
while true do--Read
--Byte 0-1
local res,err=SOCK:receive(2)
if not res then break end
local op=band(byte(res,1),0x0f)
local fin=band(byte(res,1),0x80)==0x80
--Calculating data length
local length=band(byte(res,2),0x7f)
if length==126 then
res=SOCK:receive(2)
length=shl(byte(res,1),8)+byte(res,2)
elseif length==127 then
local b={byte(SOCK:receive(8),1,8)}
length=shl(b[5],24)+shl(b[6],16)+shl(b[7],8)+b[8]
end
--Receive data
res=""
while length>0 do
local t=SOCK:receive(length)
if t then
res=res..t
length=length-#t
else--Time out!
res=false
break
end
end
--React
if res then
if op==8 then--8=close
readCHN:push(op)
SOCK:close()
if type(res)=="string"then
local reason=JSON.decode(res)
readCHN:push(reason and reason.message or"Server Error")
else
readCHN:push("Server Error")
end
elseif op==0 then--0=continue
buffer=buffer..res
if fin then
-- print("FIN=1 (c")
readCHN:push(buffer)
buffer=""
else
-- print("FIN=0 (c")
end
else
readCHN:push(op)
if fin then
-- print("OP: "..op.."\tFIN=1")
readCHN:push(res)
else
-- print("OP: "..op.."\tFIN=0")
buffer=res
-- print("START pack: "..res)
end
end
else
--TIMEOUT
SOCK:close()
readCHN:push(8)
readCHN:push("WS time out")
end
end
end
]]
local timer=love.timer.getTime
local WS={}
local wsList=setmetatable({},{
__index=function(l,k)
local ws={
real=false,
status="dead",
lastPongTime=timer(),
sendTimer=0,
alertTimer=0,
pongTimer=0,
}
l[k]=ws
return ws
end
})
function WS.connect(name,subPath,body)
local ws={
real=true,
thread=love.thread.newThread(wsThread),
triggerCHN=love.thread.newChannel(),
sendCHN=love.thread.newChannel(),
readCHN=love.thread.newChannel(),
lastPingTime=0,
lastPongTime=timer(),
pingInterval=26,
status="connecting",--connecting, running, dead
sendTimer=0,
alertTimer=0,
pongTimer=0,
}
wsList[name]=ws
ws.thread:start(ws.triggerCHN,ws.sendCHN,ws.readCHN)
ws.sendCHN:push(host)
ws.sendCHN:push(port)
ws.sendCHN:push(path..subPath)
ws.sendCHN:push(body or"")
end
function WS.status(name)
local ws=wsList[name]
return ws.status or"dead"
end
function WS.getTimers(name)
local ws=wsList[name]
return ws.pongTimer,ws.sendTimer,ws.alertTimer
end
function WS.setPingInterval(name,time)
local ws=wsList[name]
ws.pingInterval=math.max(time or 1,2.6)
end
function WS.alert(name)
local ws=wsList[name]
ws.alertTimer=1
end
local OPcode={
continue=0,
text=1,
binary=2,
close=8,
ping=9,
pong=10,
}
local OPname={
[0]="continue",
[1]="text",
[2]="binary",
[8]="close",
[9]="ping",
[10]="pong",
}
function WS.send(name,message,op)
local ws=wsList[name]
if ws.real then
ws.sendCHN:push(op and OPcode[op]or 2)--2=binary
ws.sendCHN:push(message)
ws.lastPingTime=timer()
ws.sendTimer=1
end
end
function WS.read(name)
local ws=wsList[name]
if ws.real and ws.readCHN:getCount()>=2 then
local op=ws.readCHN:pop()
local message=ws.readCHN:pop()
if op==8 then ws.status="dead"end--8=close
ws.lastPongTime=timer()
ws.pongTimer=1
return message,OPname[op]or op
end
end
function WS.close(name)
local ws=wsList[name]
if ws.real then
ws.sendCHN:push(8)--close
ws.sendCHN:push("")
ws.status="dead"
end
end
function WS.update(dt)
local time=timer()
for name,ws in next,wsList do
if ws.real then
ws.triggerCHN:push(0)
if ws.status=="connecting"then
local mes=ws.readCHN:pop()
if mes then
if mes=="success"then
ws.status="running"
ws.lastPingTime=time
ws.lastPongTime=time
ws.pongTimer=1
else
ws.status="dead"
LOG.print(text.wsFailed.." "..mes,"warn")
end
end
elseif time-ws.lastPingTime>ws.pingInterval then
ws.sendCHN:push(9)
ws.sendCHN:push("")--ping
ws.lastPingTime=time
end
if time-ws.lastPongTime>10+3*ws.pingInterval then
WS.close(name)
end
if ws.sendTimer>0 then ws.sendTimer=ws.sendTimer-dt end
if ws.pongTimer>0 then ws.pongTimer=ws.pongTimer-dt end
if ws.alertTimer>0 then ws.alertTimer=ws.alertTimer-dt end
end
end
end
return WS