作者:shihuaping0918@163.com,转载请注明作者
skynet中有http server和http client的代码,用来支持http访问。在分析http之前,要先看一下url库。文件位于skyet/lualib/http/url.lua。
回顾一下http协议,http协议有消息头,消息体,消息头和消息体以\r\n分隔,消息头以\r\n结束。对于较长数据块,以chunk的形式传输。http发送请求有get/post两种方式。但不管是哪种方式都会用到url,而url是可以带参数的,这个参数部分有个专门的名字叫做query string。
就是下面这一行,"?"后面的query。
http_URL = "http:" "//" host [ ":" port ] [ abs_path [ "?" query ]]
设计过字符串协议的同学应该都很清楚,字符串协议是需要有保留字符或者保留字符串的。对于url也不例外,所以query string需要对保留字做特殊处理,这个处理就叫做percent-encoding。将保留字转换为%xx,也就是16进制的字符串表示方式。当服务器收到这个query string时,需要对它做一次解码。
local url = {}
--将16进制字符转为数字,再将数字转为ascii码
local function decode_func(c)
return string.char(tonumber(c, 16))
end
-- 将字符串中的+替换为空格
-- 将%xx转换为字符
local function decode(str)
local str = str:gsub('+', ' ')
return str:gsub("%%(..)", decode_func)
end
--将url提取为absolute path和query string
function url.parse(u)
local path,query = u:match "([^?]*)%??(.*)"
if path then
path = decode(path)
end
return path, query
end
--解析query string
--query string格式为a=b&c=d&e=f
function url.parse_query(q)
local r = {}
-- .- 表示匹配多个字符
-- = 表示匹配等于号
-- ()用于分组,以对应k,v
for k,v in q:gmatch "(.-)=([^&]*)&?" do
r[decode(k)] = decode(v)
end
return r
end
return url