2 local coroutine = coroutine;
3 local tonumber = tonumber;
5 local deadroutine = coroutine.create(function() end);
6 coroutine.resume(deadroutine);
10 local function parser(data, success_cb)
11 local function readline()
12 local pos = data:find("\r\n", nil, true);
14 data = data..coroutine.yield();
15 pos = data:find("\r\n", nil, true);
17 local r = data:sub(1, pos-1);
18 data = data:sub(pos+2);
21 local function readlength(n)
23 data = data..coroutine.yield();
25 local r = data:sub(1, n);
26 data = data:sub(n + 1);
32 local status_line = readline();
33 local method, path, httpversion = status_line:match("^(%S+)%s+(%S+)%s+HTTP/(%S+)$");
34 if not method then coroutine.yield("invalid-status-line"); end
37 local headers = {}; -- read headers
39 local line = readline();
40 if line == "" then break; end -- headers done
41 local key, val = line:match("^([^%s:]+): *(.*)$");
42 if not key then coroutine.yield("invalid-header-line"); end -- TODO handle multi-line and invalid headers
44 headers[key] = headers[key] and headers[key]..","..val or val;
48 local len = tonumber(headers["content-length"]);
49 len = len or 0; -- TODO check for invalid len
50 local body = readlength(len);
55 httpversion = httpversion;
62 function new(success_cb, error_cb)
63 local co = coroutine.create(parser);
65 feed = function(self, data)
70 local success, result = coroutine.resume(co, data, success_cb);
73 return error_cb(result);