https.lua 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. ----------------------------------------------------------------------------
  2. -- LuaSec 0.6
  3. -- Copyright (C) 2009-2016 PUC-Rio
  4. --
  5. -- Author: Pablo Musa
  6. -- Author: Tomas Guisasola
  7. ---------------------------------------------------------------------------
  8. local socket = require("socket")
  9. local ssl = require("ssl")
  10. local ltn12 = require("ltn12")
  11. local http = require("socket.http")
  12. local url = require("socket.url")
  13. local try = socket.try
  14. --
  15. -- Module
  16. --
  17. local _M = {
  18. _VERSION = "0.6",
  19. _COPYRIGHT = "LuaSec 0.6 - Copyright (C) 2009-2016 PUC-Rio",
  20. PORT = 443,
  21. }
  22. -- TLS configuration
  23. local cfg = {
  24. protocol = "any",
  25. options = {"all", "no_sslv2", "no_sslv3"},
  26. verify = "none",
  27. }
  28. --------------------------------------------------------------------
  29. -- Auxiliar Functions
  30. --------------------------------------------------------------------
  31. -- Insert default HTTPS port.
  32. local function default_https_port(u)
  33. return url.build(url.parse(u, {port = _M.PORT}))
  34. end
  35. -- Convert an URL to a table according to Luasocket needs.
  36. local function urlstring_totable(url, body, result_table)
  37. url = {
  38. url = default_https_port(url),
  39. method = body and "POST" or "GET",
  40. sink = ltn12.sink.table(result_table)
  41. }
  42. if body then
  43. url.source = ltn12.source.string(body)
  44. url.headers = {
  45. ["content-length"] = #body,
  46. ["content-type"] = "application/x-www-form-urlencoded",
  47. }
  48. end
  49. return url
  50. end
  51. -- Forward calls to the real connection object.
  52. local function reg(conn)
  53. local mt = getmetatable(conn.sock).__index
  54. for name, method in pairs(mt) do
  55. if type(method) == "function" then
  56. conn[name] = function (self, ...)
  57. return method(self.sock, ...)
  58. end
  59. end
  60. end
  61. end
  62. -- Return a function which performs the SSL/TLS connection.
  63. local function tcp(params)
  64. params = params or {}
  65. -- Default settings
  66. for k, v in pairs(cfg) do
  67. params[k] = params[k] or v
  68. end
  69. -- Force client mode
  70. params.mode = "client"
  71. -- 'create' function for LuaSocket
  72. return function ()
  73. local conn = {}
  74. conn.sock = try(socket.tcp())
  75. local st = getmetatable(conn.sock).__index.settimeout
  76. function conn:settimeout(...)
  77. return st(self.sock, ...)
  78. end
  79. -- Replace TCP's connection function
  80. function conn:connect(host, port)
  81. try(self.sock:connect(host, port))
  82. self.sock = try(ssl.wrap(self.sock, params))
  83. try(self.sock:dohandshake())
  84. reg(self, getmetatable(self.sock))
  85. return 1
  86. end
  87. return conn
  88. end
  89. end
  90. --------------------------------------------------------------------
  91. -- Main Function
  92. --------------------------------------------------------------------
  93. -- Make a HTTP request over secure connection. This function receives
  94. -- the same parameters of LuaSocket's HTTP module (except 'proxy' and
  95. -- 'redirect') plus LuaSec parameters.
  96. --
  97. -- @param url mandatory (string or table)
  98. -- @param body optional (string)
  99. -- @return (string if url == string or 1), code, headers, status
  100. --
  101. local function request(url, body)
  102. local result_table = {}
  103. local stringrequest = type(url) == "string"
  104. if stringrequest then
  105. url = urlstring_totable(url, body, result_table)
  106. else
  107. url.url = default_https_port(url.url)
  108. end
  109. if http.PROXY or url.proxy then
  110. return nil, "proxy not supported"
  111. elseif url.redirect then
  112. return nil, "redirect not supported"
  113. elseif url.create then
  114. return nil, "create function not permitted"
  115. end
  116. -- New 'create' function to establish a secure connection
  117. url.create = tcp(url)
  118. local res, code, headers, status = http.request(url)
  119. if res and stringrequest then
  120. return table.concat(result_table), code, headers, status
  121. end
  122. return res, code, headers, status
  123. end
  124. --------------------------------------------------------------------------------
  125. -- Export module
  126. --
  127. _M.request = request
  128. return _M