summaryrefslogtreecommitdiffstats
path: root/lem/http.lua
blob: 1f972a961f194d58b8c6c1de34d4471ba9466278 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
--
-- This file is part of LEM, a Lua Event Machine.
-- Copyright 2011-2013 Emil Renner Berthing
--
-- LEM is free software: you can redistribute it and/or modify it
-- under the terms of the GNU Lesser General Public License as
-- published by the Free Software Foundation, either version 3 of
-- the License, or (at your option) any later version.
--
-- LEM is distributed in the hope that it will be useful, but
-- WITHOUT ANY WARRANTY; without even the implied warranty of
-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
-- GNU Lesser General Public License for more details.
--
-- You should have received a copy of the GNU Lesser General Public
-- License along with LEM.  If not, see <http://www.gnu.org/licenses/>.
--

local io   = require 'lem.io'
local http = require 'lem.http.core'

do
	local parsers = require 'lem.parsers'

	parsers.lookup['HTTPRequest'] = http.HTTPRequest
	http.HTTPRequest = nil
	parsers.lookup['HTTPResponse'] = http.HTTPResponse
	http.HTTPResponse = nil
end

local tonumber = tonumber
local concat = table.concat

function http.Request:body()
	local len, body = self.headers['content-length'], ''
	if not len then return body end

	len = tonumber(len)
	if len <= 0 then return body end

	if self.headers['expect'] == '100-continue' then
		local ok, err = self.client:write('HTTP/1.1 100 Continue\r\n\r\n')
		if not ok then return nil, err end
	end

	local err
	body, err = self.client:read(len)
	if not body then return nil, err end

	return body
end

function http.Response:body_chunked()
	local client = self.client
	local t, n = {}, 0
	local line, err
	while true do
		line, err = client:read('*l')
		if not line then return nil, err end

		local num = tonumber(line, 16)
		if not num then return nil, 'expectation failed' end
		if num == 0 then break end

		local data, err = client:read(num)
		if not data then return nil, err end

		n = n + 1
		t[n] = data

		line, err = client:read('*l')
		if not line then return nil, err end
	end

	line, err = client:read('*l')
	if not line then return nil, err end

	return concat(t)
end

function http.Response:body()
	if self.headers['transfer-encoding'] == 'chunked' then
		return self:body_chunked()
	end

	local num = self.headers['content-length']
	if not num then
		if self.headers['connection'] == 'close' then
			return self.client:read('*a')
		end
		return nil, 'no content length specified'
	end

	num = tonumber(num)
	if not num then return nil, 'invalid content length' end

	return self.client:read(num)
end

return http

-- vim: ts=2 sw=2 noet: