Welcome to mirror list, hosted at ThFree Co, Russian Federation.

multipart.lua « upload « luarocks « src « luarocks - github.com/torch/luajit-rocks.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: aad2e4398d371284f3419f4575499655e821a4c1 (plain)
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
103
104
105
106
107
108
109
110
111

local multipart = {}

local File = {}

local unpack = unpack or table.unpack

math.randomseed(os.time())

-- socket.url.escape(s) from LuaSocket 3.0rc1
function multipart.url_escape(s)
   return (string.gsub(s, "([^A-Za-z0-9_])", function(c)
      return string.format("%%%02x", string.byte(c))
   end))
end

function File:mime()
   if not self.mimetype then
      local mimetypes_ok, mimetypes = pcall(require, "mimetypes")
      if mimetypes_ok then
         self.mimetype = mimetypes.guess(self.fname)
      end
      self.mimetype = self.mimetype or "application/octet-stream"
   end
   return self.mimetype
end

function File:content()
   local fd = io.open(self.fname, "rb")
   if not fd then
      return nil, "Failed to open file: "..self.fname
   end
   local data = fd:read("*a")
   fd:close()
   return data
end

local function rand_string(len)
   local shuffled = {}
   for i = 1, len do
      local r = math.random(97, 122)
      if math.random() >= 0.5 then
        r = r - 32
      end
      shuffled[i] = r
   end
   return string.char(unpack(shuffled))
end

-- multipart encodes params
-- returns encoded string,boundary
-- params is an a table of tuple tables:
-- params = {
--   {key1, value2},
--   {key2, value2},
--   key3: value3
-- }
function multipart.encode(params)
   local tuples = { }
   for i = 1, #params do
      tuples[i] = params[i]
   end
   for k,v in pairs(params) do
      if type(k) == "string" then
         table.insert(tuples, {k, v})
      end
   end
   local chunks = {}
   for _, tuple in ipairs(tuples) do
      local k,v = unpack(tuple)
      k = multipart.url_escape(k)
      local buffer = { 'Content-Disposition: form-data; name="' .. k .. '"' }
      local content
      if type(v) == "table" and v.__class == File then
         buffer[1] = buffer[1] .. ('; filename="' .. v.fname:gsub(".*/", "") .. '"')
         table.insert(buffer, "Content-type: " .. v:mime())
         content = v:content()
      else
         content = v
      end
      table.insert(buffer, "")
      table.insert(buffer, content)
      table.insert(chunks, table.concat(buffer, "\r\n"))
   end
   local boundary
   while not boundary do
      boundary = "Boundary" .. rand_string(16)
      for _, chunk in ipairs(chunks) do
         if chunk:find(boundary) then
            boundary = nil
            break
         end
      end
   end
  local inner = "\r\n--" .. boundary .. "\r\n"
  return table.concat({ "--", boundary, "\r\n",
                        table.concat(chunks, inner),
                        "\r\n", "--", boundary, "--", "\r\n" }), boundary
end

function multipart.new_file(fname, mime)
   local self = {}
   setmetatable(self, { __index = File })
   self.__class = File
   self.fname = fname
   self.mimetype = mime
   return self
end

return multipart