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);
29 local function readheaders()
30 local headers = {}; -- read headers
32 local line = readline();
33 if line == "" then break; end -- headers done
34 local key, val = line:match("^([^%s:]+): *(.*)$");
35 if not key then coroutine.yield("invalid-header-line"); end -- TODO handle multi-line and invalid headers
37 headers[key] = headers[key] and headers[key]..","..val or val;
43 local status_line = readline();
44 local method, path, httpversion = status_line:match("^(%S+)%s+(%S+)%s+HTTP/(%S+)$");
45 if not method then coroutine.yield("invalid-status-line"); end
47 local headers = readheaders();
50 local len = tonumber(headers["content-length"]);
51 len = len or 0; -- TODO check for invalid len
52 local body = readlength(len);
57 httpversion = httpversion;
64 function new(success_cb, error_cb)
65 local co = coroutine.create(parser);
67 feed = function(self, data)
72 local success, result = coroutine.resume(co, data, success_cb);
75 return error_cb(result);