-
-
Save creationix/079e11345249525c7e2e to your computer and use it in GitHub Desktop.
wifi.setmode(wifi.STATION) | |
wifi.sta.config("creationix","noderocks") | |
wifi.sta.connect() | |
tmr.alarm(0, 1000, 1, function () | |
local ip = wifi.sta.getip() | |
if ip then | |
tmr.stop(0) | |
print(ip) | |
dofile("websocket.lc") | |
dofile("main.lc") | |
else | |
print("Connecting to WIFI...") | |
end | |
end) |
websocket.createServer(80, function (socket) | |
local data | |
node.output(function (msg) | |
return socket.send(msg, 1) | |
end, 1) | |
print("New websocket client connected") | |
function socket.onmessage(payload, opcode) | |
if opcode == 1 then | |
if payload == "ls" then | |
local list = file.list() | |
local lines = {} | |
for k, v in pairs(list) do | |
lines[#lines + 1] = k .. "\0" .. v | |
end | |
socket.send(table.concat(lines, "\0"), 2) | |
return | |
end | |
local command, name = payload:match("^([a-z]+):(.*)$") | |
if command == "load" then | |
file.open(name, "r") | |
socket.send(file.read(), 2) | |
file.close() | |
elseif command == "save" then | |
file.open(name, "w") | |
file.write(data) | |
data = nil | |
file.close() | |
elseif command == "compile" then | |
node.compile(name) | |
elseif command == "run" then | |
dofile(name) | |
elseif command == "eval" then | |
local fn = loadstring(data, name) | |
data = nil | |
fn() | |
else | |
print("Invalid command: " .. command) | |
end | |
elseif opcode == 2 then | |
data = payload | |
end | |
end | |
end) |
do | |
local websocket = {} | |
_G.websocket = websocket | |
local band = bit.band | |
local bor = bit.bor | |
local rshift = bit.rshift | |
local lshift = bit.lshift | |
local char = string.char | |
local byte = string.byte | |
local sub = string.sub | |
local applyMask = crypto.mask | |
local toBase64 = crypto.toBase64 | |
local sha1 = crypto.sha1 | |
local function decode(chunk) | |
if #chunk < 2 then return end | |
local second = byte(chunk, 2) | |
local len = band(second, 0x7f) | |
local offset | |
if len == 126 then | |
if #chunk < 4 then return end | |
len = bor( | |
lshift(byte(chunk, 3), 8), | |
byte(chunk, 4)) | |
offset = 4 | |
elseif len == 127 then | |
if #chunk < 10 then return end | |
len = bor( | |
-- Ignore lengths longer than 32bit | |
lshift(byte(chunk, 7), 24), | |
lshift(byte(chunk, 8), 16), | |
lshift(byte(chunk, 9), 8), | |
byte(chunk, 10)) | |
offset = 10 | |
else | |
offset = 2 | |
end | |
local mask = band(second, 0x80) > 0 | |
if mask then | |
offset = offset + 4 | |
end | |
if #chunk < offset + len then return end | |
local first = byte(chunk, 1) | |
local payload = sub(chunk, offset + 1, offset + len) | |
assert(#payload == len, "Length mismatch") | |
if mask then | |
payload = applyMask(payload, sub(chunk, offset - 3, offset)) | |
end | |
local extra = sub(chunk, offset + len + 1) | |
local opcode = band(first, 0xf) | |
return extra, payload, opcode | |
end | |
local function encode(payload, opcode) | |
opcode = opcode or 2 | |
assert(type(opcode) == "number", "opcode must be number") | |
assert(type(payload) == "string", "payload must be string") | |
local len = #payload | |
local head = char( | |
bor(0x80, opcode), | |
len < 126 and len or (len < 0x10000) and 126 or 127 | |
) | |
if len >= 0x10000 then | |
head = head .. char( | |
0,0,0,0, -- 32 bit length is plenty, assume zero for rest | |
band(rshift(len, 24), 0xff), | |
band(rshift(len, 16), 0xff), | |
band(rshift(len, 8), 0xff), | |
band(len, 0xff) | |
) | |
elseif len >= 126 then | |
head = head .. char(band(rshift(len, 8), 0xff), band(len, 0xff)) | |
end | |
return head .. payload | |
end | |
local guid = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11" | |
local function acceptKey(key) | |
return toBase64(sha1(key .. guid)) | |
end | |
function websocket.createServer(port, callback) | |
net.createServer(net.TCP):listen(port, function(conn) | |
local buffer = false | |
local socket = {} | |
function socket.send(...) | |
return conn:send(encode(...)) | |
end | |
conn:on("receive", function(_, chunk) | |
if buffer then | |
buffer = buffer .. chunk | |
while true do | |
local extra, payload, opcode = decode(buffer) | |
if not extra then return end | |
buffer = extra | |
socket.onmessage(payload, opcode) | |
end | |
end | |
local _, e, method = string.find(chunk, "([A-Z]+) /[^\r]* HTTP/%d%.%d\r\n") | |
local key, name, value | |
while true do | |
_, e, name, value = string.find(chunk, "([^ ]+): *([^\r]+)\r\n", e + 1) | |
if not e then break end | |
if string.lower(name) == "sec-websocket-key" then | |
key = value | |
end | |
end | |
if method == "GET" and key then | |
conn:send("HTTP/1.1 101 Switching Protocols\r\n") | |
conn:send("Upgrade: websocket\r\n") | |
conn:send("Connection: Upgrade\r\n") | |
conn:send("Sec-WebSocket-Accept: " .. acceptKey(key) .. "\r\n\r\n") | |
buffer = "" | |
callback(socket) | |
else | |
conn:send("HTTP/1.1 404 Not Found\r\nConnection: Close\r\n\r\n") | |
conn:on("sent", conn.close) | |
end | |
end) | |
end) | |
end | |
end |
Remember people that gists don't notify anymore. If you have a question, ping me on twitter (@creationix), email or irc (@creationix). I also hang out in the nodemcu and esp8266/arduino channels channels on gitter.im.
conn:send("HTTP/1.1 101 Switching Protocols\r\n")
conn:send("Upgrade: websocket\r\n")
conn:send("Connection: Upgrade\r\n")
conn:send("Sec-WebSocket-Accept: " .. acceptKey(key) .. "\r\n\r\n")
Multiple conn:send won't work.
Change it to conn:send("HTTP/1.1 101 Switching Protocols\r\nUpgrade: websocket\r\nConnection: Upgrade\r\nSec-WebSocket-Accept: " .. acceptKey(key) .. "\r\n\r\n")
to get the correct frame header
I was able to get one way communication going with this - from browser to ESP8266, but not the other way around. Spoke to the author and he couldn't get it working reliably in the end.
I need also a TCP/UDP server with secure connection (SSH/SSL)