mirror of
https://github.com/lunarmodules/luasocket.git
synced 2024-11-14 01:08:21 +01:00
e5a090b01c
Adjusted socket.try.
312 lines
9.7 KiB
Lua
312 lines
9.7 KiB
Lua
-----------------------------------------------------------------------------
|
|
-- HTTP/1.1 client support for the Lua language.
|
|
-- LuaSocket toolkit.
|
|
-- Author: Diego Nehab
|
|
-- Conforming to RFC 2616
|
|
-- RCS ID: $Id$
|
|
-----------------------------------------------------------------------------
|
|
-- make sure LuaSocket is loaded
|
|
if not LUASOCKET_LIBNAME then error('module requires LuaSocket') end
|
|
-- get LuaSocket namespace
|
|
local socket = _G[LUASOCKET_LIBNAME]
|
|
if not socket then error('module requires LuaSocket') end
|
|
-- create namespace inside LuaSocket namespace
|
|
socket.http = socket.http or {}
|
|
-- make all module globals fall into namespace
|
|
setmetatable(socket.http, { __index = _G })
|
|
setfenv(1, socket.http)
|
|
|
|
-----------------------------------------------------------------------------
|
|
-- Program constants
|
|
-----------------------------------------------------------------------------
|
|
-- connection timeout in seconds
|
|
TIMEOUT = 60
|
|
-- default port for document retrieval
|
|
PORT = 80
|
|
-- user agent field sent in request
|
|
USERAGENT = socket.version
|
|
-- block size used in transfers
|
|
BLOCKSIZE = 2048
|
|
|
|
-----------------------------------------------------------------------------
|
|
-- Function return value selectors
|
|
-----------------------------------------------------------------------------
|
|
local function second(a, b)
|
|
return b
|
|
end
|
|
|
|
local function third(a, b, c)
|
|
return c
|
|
end
|
|
|
|
local function receive_headers(reqt, respt, tmp)
|
|
local sock = tmp.sock
|
|
local line, name, value, _
|
|
local headers = {}
|
|
-- store results
|
|
respt.headers = headers
|
|
-- get first line
|
|
line = socket.try(sock:receive())
|
|
-- headers go until a blank line is found
|
|
while line ~= "" do
|
|
-- get field-name and value
|
|
_, _, name, value = string.find(line, "^(.-):%s*(.*)")
|
|
socket.try(name and value, "malformed reponse headers")
|
|
name = string.lower(name)
|
|
-- get next line (value might be folded)
|
|
line = socket.try(sock:receive())
|
|
-- unfold any folded values
|
|
while string.find(line, "^%s") do
|
|
value = value .. line
|
|
line = socket.try(sock:receive())
|
|
end
|
|
-- save pair in table
|
|
if headers[name] then headers[name] = headers[name] .. ", " .. value
|
|
else headers[name] = value end
|
|
end
|
|
end
|
|
|
|
local function receive_body(reqt, respt, tmp)
|
|
local sink = reqt.sink or ltn12.sink.null()
|
|
local pump = reqt.pump or ltn12.pump
|
|
local source
|
|
local te = respt.headers["transfer-encoding"]
|
|
if te and te ~= "identity" then
|
|
-- get by chunked transfer-coding of message body
|
|
source = socket.source("http-chunked", tmp.sock)
|
|
elseif tonumber(respt.headers["content-length"]) then
|
|
-- get by content-length
|
|
local length = tonumber(respt.headers["content-length"])
|
|
source = socket.source("by-length", tmp.sock, length)
|
|
else
|
|
-- get it all until connection closes
|
|
source = socket.source("until-closed", tmp.sock)
|
|
end
|
|
socket.try(pump(source, sink))
|
|
end
|
|
|
|
local function send_headers(sock, headers)
|
|
-- send request headers
|
|
for i, v in pairs(headers) do
|
|
socket.try(sock:send(i .. ": " .. v .. "\r\n"))
|
|
end
|
|
-- mark end of request headers
|
|
socket.try(sock:send("\r\n"))
|
|
end
|
|
|
|
local function should_receive_body(reqt, respt, tmp)
|
|
if reqt.method == "HEAD" then return nil end
|
|
if respt.code == 204 or respt.code == 304 then return nil end
|
|
if respt.code >= 100 and respt.code < 200 then return nil end
|
|
return 1
|
|
end
|
|
|
|
local function receive_status(reqt, respt, tmp)
|
|
local status = socket.try(tmp.sock:receive())
|
|
local code = third(string.find(status, "HTTP/%d*%.%d* (%d%d%d)"))
|
|
-- store results
|
|
respt.code, respt.status = tonumber(code), status
|
|
end
|
|
|
|
local function request_uri(reqt, respt, tmp)
|
|
local url = tmp.parsed
|
|
if not reqt.proxy then
|
|
local parsed = tmp.parsed
|
|
url = {
|
|
path = parsed.path,
|
|
params = parsed.params,
|
|
query = parsed.query,
|
|
fragment = parsed.fragment
|
|
}
|
|
end
|
|
return socket.url.build(url)
|
|
end
|
|
|
|
local function send_request(reqt, respt, tmp)
|
|
local uri = request_uri(reqt, respt, tmp)
|
|
local headers = tmp.headers
|
|
local pump = reqt.pump or ltn12.pump
|
|
-- send request line
|
|
socket.try(tmp.sock:send((reqt.method or "GET")
|
|
.. " " .. uri .. " HTTP/1.1\r\n"))
|
|
if reqt.source and not headers["content-length"] then
|
|
headers["transfer-encoding"] = "chunked"
|
|
end
|
|
send_headers(tmp.sock, headers)
|
|
-- send request message body, if any
|
|
if not reqt.source then return end
|
|
if headers["content-length"] then
|
|
socket.try(pump(reqt.source, socket.sink(tmp.sock)))
|
|
else
|
|
socket.try(pump(reqt.source, socket.sink("http-chunked", tmp.sock)))
|
|
end
|
|
end
|
|
|
|
local function open(reqt, respt, tmp)
|
|
local proxy = reqt.proxy or PROXY
|
|
local host, port
|
|
if proxy then
|
|
local pproxy = socket.url.parse(proxy)
|
|
socket.try(pproxy.port and pproxy.host, "invalid proxy")
|
|
host, port = pproxy.host, pproxy.port
|
|
else
|
|
host, port = tmp.parsed.host, tmp.parsed.port
|
|
end
|
|
-- store results
|
|
tmp.sock = socket.try(socket.tcp())
|
|
socket.try(tmp.sock:settimeout(reqt.timeout or TIMEOUT))
|
|
socket.try(tmp.sock:connect(host, port))
|
|
end
|
|
|
|
local function adjust_headers(reqt, respt, tmp)
|
|
local lower = {}
|
|
local headers = reqt.headers or {}
|
|
-- set default headers
|
|
lower["user-agent"] = USERAGENT
|
|
-- override with user values
|
|
for i,v in headers do
|
|
lower[string.lower(i)] = v
|
|
end
|
|
lower["host"] = tmp.parsed.host
|
|
-- this cannot be overriden
|
|
lower["connection"] = "close"
|
|
-- store results
|
|
tmp.headers = lower
|
|
end
|
|
|
|
local function parse_url(reqt, respt, tmp)
|
|
-- parse url with default fields
|
|
local parsed = socket.url.parse(reqt.url, {
|
|
host = "",
|
|
port = PORT,
|
|
path ="/",
|
|
scheme = "http"
|
|
})
|
|
-- scheme has to be http
|
|
socket.try(parsed.scheme == "http",
|
|
string.format("unknown scheme '%s'", parsed.scheme))
|
|
-- explicit authentication info overrides that given by the URL
|
|
parsed.user = reqt.user or parsed.user
|
|
parsed.password = reqt.password or parsed.password
|
|
-- store results
|
|
tmp.parsed = parsed
|
|
end
|
|
|
|
-- forward declaration
|
|
local request_p
|
|
|
|
local function should_authorize(reqt, respt, tmp)
|
|
-- if there has been an authorization attempt, it must have failed
|
|
if reqt.headers and reqt.headers["authorization"] then return nil end
|
|
-- if last attempt didn't fail due to lack of authentication,
|
|
-- or we don't have authorization information, we can't retry
|
|
return respt.code == 401 and tmp.parsed.user and tmp.parsed.password
|
|
end
|
|
|
|
local function clone(headers)
|
|
if not headers then return nil end
|
|
local copy = {}
|
|
for i,v in pairs(headers) do
|
|
copy[i] = v
|
|
end
|
|
return copy
|
|
end
|
|
|
|
local function authorize(reqt, respt, tmp)
|
|
local headers = clone(reqt.headers) or {}
|
|
headers["authorization"] = "Basic " ..
|
|
(mime.b64(tmp.parsed.user .. ":" .. tmp.parsed.password))
|
|
local autht = {
|
|
method = reqt.method,
|
|
url = reqt.url,
|
|
source = reqt.source,
|
|
sink = reqt.sink,
|
|
headers = headers,
|
|
timeout = reqt.timeout,
|
|
proxy = reqt.proxy,
|
|
}
|
|
request_p(autht, respt, tmp)
|
|
end
|
|
|
|
local function should_redirect(reqt, respt, tmp)
|
|
return (reqt.redirect ~= false) and
|
|
(respt.code == 301 or respt.code == 302) and
|
|
(not reqt.method or reqt.method == "GET" or reqt.method == "HEAD")
|
|
and (not tmp.nredirects or tmp.nredirects < 5)
|
|
end
|
|
|
|
local function redirect(reqt, respt, tmp)
|
|
tmp.nredirects = (tmp.nredirects or 0) + 1
|
|
local redirt = {
|
|
method = reqt.method,
|
|
-- the RFC says the redirect URL has to be absolute, but some
|
|
-- servers do not respect that
|
|
url = socket.url.absolute(reqt.url, respt.headers["location"]),
|
|
source = reqt.source,
|
|
sink = reqt.sink,
|
|
headers = reqt.headers,
|
|
timeout = reqt.timeout,
|
|
proxy = reqt.proxy
|
|
}
|
|
request_p(redirt, respt, tmp)
|
|
-- we pass the location header as a clue we redirected
|
|
if respt.headers then respt.headers.location = redirt.url end
|
|
end
|
|
|
|
local function skip_continue(reqt, respt, tmp)
|
|
if respt.code == 100 then
|
|
receive_status(reqt, respt, tmp)
|
|
end
|
|
end
|
|
|
|
-- execute a request of through an exception
|
|
function request_p(reqt, respt, tmp)
|
|
parse_url(reqt, respt, tmp)
|
|
adjust_headers(reqt, respt, tmp)
|
|
open(reqt, respt, tmp)
|
|
send_request(reqt, respt, tmp)
|
|
receive_status(reqt, respt, tmp)
|
|
skip_continue(reqt, respt, tmp)
|
|
receive_headers(reqt, respt, tmp)
|
|
if should_redirect(reqt, respt, tmp) then
|
|
tmp.sock:close()
|
|
redirect(reqt, respt, tmp)
|
|
elseif should_authorize(reqt, respt, tmp) then
|
|
tmp.sock:close()
|
|
authorize(reqt, respt, tmp)
|
|
elseif should_receive_body(reqt, respt, tmp) then
|
|
receive_body(reqt, respt, tmp)
|
|
end
|
|
end
|
|
|
|
function request(reqt)
|
|
local respt, tmp = {}, {}
|
|
local s, e = pcall(request_p, reqt, respt, tmp)
|
|
if not s then respt.error = e end
|
|
if tmp.sock then tmp.sock:close() end
|
|
return respt
|
|
end
|
|
|
|
function get(url)
|
|
local t = {}
|
|
respt = request {
|
|
url = url,
|
|
sink = ltn12.sink.table(t)
|
|
}
|
|
return (table.getn(t) > 0 or nil) and table.concat(t), respt.headers,
|
|
respt.code, respt.error
|
|
end
|
|
|
|
function post(url, body)
|
|
local t = {}
|
|
respt = request {
|
|
url = url,
|
|
method = "POST",
|
|
source = ltn12.source.string(body),
|
|
sink = ltn12.sink.table(t),
|
|
headers = { ["content-length"] = string.len(body) }
|
|
}
|
|
return (table.getn(t) > 0 or nil) and table.concat(t),
|
|
respt.headers, respt.code, respt.error
|
|
end
|