Skip to content

Commit 9a3aa90

Browse files
committed
Multipart support.
Adds helper code to encode multipart/form-data requests. This, for instances, allows to post images to Twitter. The multipart encoder was adapted from @catwell's [lua-multipart-post](https://github.com/catwell/lua-multipart-post). fixes #6
1 parent c057c93 commit 9a3aa90

File tree

7 files changed

+240
-28
lines changed

7 files changed

+240
-28
lines changed

LICENSE

+27-1
Original file line numberDiff line numberDiff line change
@@ -18,4 +18,30 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
1818
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
1919
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
2020
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21-
THE SOFTWARE.
21+
THE SOFTWARE.
22+
23+
24+
-----------------
25+
Contains code from lua-multipart-post
26+
27+
https://github.com/catwell/lua-multipart-post
28+
29+
Copyright (C) 2012-2013 by Moodstocks SAS
30+
31+
Permission is hereby granted, free of charge, to any person obtaining a copy
32+
of this software and associated documentation files (the "Software"), to deal
33+
in the Software without restriction, including without limitation the rights
34+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
35+
copies of the Software, and to permit persons to whom the Software is
36+
furnished to do so, subject to the following conditions:
37+
38+
The above copyright notice and this permission notice shall be included in
39+
all copies or substantial portions of the Software.
40+
41+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
42+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
43+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
44+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
45+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
46+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
47+
THE SOFTWARE.

README.md

+52-16
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ This is an adaptation of Jeffrey Friedl's [Twitter OAuth Authentication Routines
88
with Lightroom's code replaced by other libraries (i.e. [LuaSec][2], [LuaSocket][3], etc) and with HMAC-SHA1 calculations
99
done with [LuaCrypto][4] instead of [plain Lua][6].
1010

11-
Most of the code was taken from [Jeffrey Friedl's blog][1].
11+
Most of the code was taken from [Jeffrey Friedl's blog][1]. Multipart encoding was adapted from @catwell's [lua-multipart-post][9].
1212

1313
LuaOAuth supports two modes of operation. A "synchronous" mode were you block while you wait for the results or an
1414
"asynchronous" mode where you must supply "callbacks" in order to receive the results. LuaOAuth will behave asynchronously
@@ -76,9 +76,9 @@ Copy the following in a script and run it from the console. Follow the instructi
7676
``` lua
7777
local OAuth = require "OAuth"
7878
local client = OAuth.new("consumer_key", "consumer_secret", {
79-
RequestToken = "http://api.twitter.com/oauth/request_token",
80-
AuthorizeUser = {"http://api.twitter.com/oauth/authorize", method = "GET"},
81-
AccessToken = "http://api.twitter.com/oauth/access_token"
79+
RequestToken = "https://api.twitter.com/oauth/request_token",
80+
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
81+
AccessToken = "https://api.twitter.com/oauth/access_token"
8282
})
8383
local callback_url = "oob"
8484
local values = client:RequestToken({ oauth_callback = callback_url })
@@ -97,9 +97,9 @@ oauth_verifier = tostring(oauth_verifier) -- must be a string
9797

9898
-- now we'll use the tokens we got in the RequestToken call, plus our PIN
9999
local client = OAuth.new("consumer_key", "consumer_secret", {
100-
RequestToken = "http://api.twitter.com/oauth/request_token",
101-
AuthorizeUser = {"http://api.twitter.com/oauth/authorize", method = "GET"},
102-
AccessToken = "http://api.twitter.com/oauth/access_token"
100+
RequestToken = "https://api.twitter.com/oauth/request_token",
101+
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
102+
AccessToken = "https://api.twitter.com/oauth/access_token"
103103
}, {
104104
OAuthToken = oauth_token,
105105
OAuthVerifier = oauth_verifier
@@ -129,41 +129,76 @@ local oauth_token = "<the value from last step>"
129129
local oauth_token_secret = "<the value from last step>"
130130

131131
local client = OAuth.new("consumer_key", "consumer_secret", {
132-
RequestToken = "http://api.twitter.com/oauth/request_token",
133-
AuthorizeUser = {"http://api.twitter.com/oauth/authorize", method = "GET"},
134-
AccessToken = "http://api.twitter.com/oauth/access_token"
132+
RequestToken = "https://api.twitter.com/oauth/request_token",
133+
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
134+
AccessToken = "https://api.twitter.com/oauth/access_token"
135135
}, {
136136
OAuthToken = oauth_token,
137137
OAuthTokenSecret = oauth_token_secret
138138
})
139139

140140
-- the mandatory "Hello World" example...
141141
local response_code, response_headers, response_status_line, response_body =
142-
client:PerformRequest("POST", "http://api.twitter.com/1/statuses/update.json", {status = "Hello World From Lua!" .. os.time()})
142+
client:PerformRequest("POST", "https://api.twitter.com/1.1/statuses/update.json", {status = "Hello World From Lua!" .. os.time()})
143143
print("response_code", response_code)
144144
print("response_status_line", response_status_line)
145145
for k,v in pairs(response_headers) do print(k,v) end
146146
print("response_body", response_body)
147147
```
148148

149-
Now, let's try to request my Twitts.
149+
Now, let's try to request my Tweets.
150150

151151
``` lua
152152
local oauth_token = "<the value from last step>"
153153
local oauth_token_secret = "<the value from last step>"
154154

155155
local client = OAuth.new("consumer_key", "consumer_secret", {
156-
RequestToken = "http://api.twitter.com/oauth/request_token",
157-
AuthorizeUser = {"http://api.twitter.com/oauth/authorize", method = "GET"},
158-
AccessToken = "http://api.twitter.com/oauth/access_token"
156+
RequestToken = "https://api.twitter.com/oauth/request_token",
157+
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
158+
AccessToken = "https://api.twitter.com/oauth/access_token"
159159
}, {
160160
OAuthToken = oauth_token,
161161
OAuthTokenSecret = oauth_token_secret
162162
})
163163

164164
-- the mandatory "Hello World" example...
165165
local response_code, response_headers, response_status_line, response_body =
166-
client:PerformRequest("GET", "http://api.twitter.com/1/statuses/user_timeline.json", {screen_name = "iburgueno"})
166+
client:PerformRequest("GET", "https://api.twitter.com/1.1/statuses/user_timeline.json", {screen_name = "iburgueno"})
167+
print("response_code", response_code)
168+
print("response_status_line", response_status_line)
169+
for k,v in pairs(response_headers) do print(k,v) end
170+
print("response_body", response_body)
171+
```
172+
173+
And now, let's post an image:
174+
175+
``` lua
176+
local oauth_token = "<the value from last step>"
177+
local oauth_token_secret = "<the value from last step>"
178+
179+
local client = OAuth.new("consumer_key", "consumer_secret", {
180+
RequestToken = "https://api.twitter.com/oauth/request_token",
181+
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
182+
AccessToken = "https://api.twitter.com/oauth/access_token"
183+
}, {
184+
OAuthToken = oauth_token,
185+
OAuthTokenSecret = oauth_token_secret
186+
})
187+
188+
local helpers = require "OAuth.helpers"
189+
190+
local req = helpers.multipart.Request{
191+
status = "Hello World From Lua!",
192+
["media[]"] = {
193+
filename = "@picture.jpg",
194+
data = picture_data -- some picture file you have read
195+
}
196+
}
197+
local response_code, response_headers, response_status_line, response_body =
198+
client:PerformRequest("POST",
199+
"https://api.twitter.com/1.1/statuses/update_with_media.json",
200+
req.body, req.headers
201+
)
167202
print("response_code", response_code)
168203
print("response_status_line", response_status_line)
169204
for k,v in pairs(response_headers) do print(k,v) end
@@ -179,3 +214,4 @@ print("response_body", response_body)
179214
[6]: http://regex.info/blog/lua/sha1
180215
[7]: http://dev.twitter.com/apps
181216
[8]: https://github.com/ignacio/luanode
217+
[9]: https://github.com/catwell/lua-multipart-post

src/OAuth.lua

+4
Original file line numberDiff line numberDiff line change
@@ -465,6 +465,8 @@ end
465465
-- @param url is the url to request
466466
-- @param arguments is an optional table whose keys and values will be encoded as "application/x-www-form-urlencoded"
467467
-- (when doing a POST) or encoded and sent in the query string (when doing a GET).
468+
-- It can also be a string with the body to be sent in the request (usually a POST). In that case, you need to supply
469+
-- a valid Content-Type header.
468470
-- @param headers is an optional table with http headers to be sent in the request
469471
-- @param callback is only required if running under LuaNode. It is a function to be called with an (optional) error object and the result of the request.
470472
-- @return nothing if running under Luanode (the callback will be called instead). Else, the http status code
@@ -506,6 +508,8 @@ end
506508
-- @param url is the url to request
507509
-- @param arguments is an optional table whose keys and values will be encoded as "application/x-www-form-urlencoded"
508510
-- (when doing a POST) or encoded and sent in the query string (when doing a GET).
511+
-- It can also be a string with the body to be sent in the request (usually a POST). In that case, you need to supply
512+
-- a valid Content-Type header.
509513
-- @param headers is an optional table with http headers to be sent in the request
510514
-- @return a table with the headers, a table with the (cleaned up) arguments and the request body.
511515
function BuildRequest(self, method, url, arguments, headers)

src/OAuth/helpers.lua

+97-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
1-
local pairs, table, tostring = pairs, table, tostring
1+
local pairs, table, tostring, select = pairs, table, tostring, select
2+
local type, assert = type, assert
3+
local string = require "string"
4+
local math = require "math"
25
local Url, Qs
36
local isLuaNode
47

@@ -21,3 +24,96 @@ url_encode_arguments = (isLuaNode and Qs.url_encode_arguments) or function(argum
2124
end
2225
return table.concat(body, "&")
2326
end
27+
28+
29+
---
30+
-- Multipart form-data helper.
31+
--
32+
-- Taken from https://github.com/catwell/lua-multipart-post
33+
--
34+
do -- Create a scope to avoid these local helpers to escape
35+
36+
local function fmt(p, ...)
37+
if select('#',...) == 0 then
38+
return p
39+
else
40+
return string.format(p, ...)
41+
end
42+
end
43+
44+
local function tprintf(t, p, ...)
45+
t[#t + 1] = fmt(p, ...)
46+
end
47+
48+
local function append_data(r, k, data, extra)
49+
tprintf(r, "content-disposition: form-data; name=\"%s\"", k)
50+
if extra.filename then
51+
tprintf(r, "; filename=\"%s\"", extra.filename)
52+
end
53+
if extra.content_type then
54+
tprintf(r, "\r\ncontent-type: %s", extra.content_type)
55+
end
56+
if extra.content_transfer_encoding then
57+
tprintf(r, "\r\ncontent-transfer-encoding: %s", extra.content_transfer_encoding)
58+
end
59+
tprintf(r, "\r\n\r\n")
60+
tprintf(r, data)
61+
tprintf(r, "\r\n")
62+
end
63+
64+
local function gen_boundary()
65+
local t = {"BOUNDARY-"}
66+
for i = 2, 17 do
67+
t[i] = string.char(math.random(65, 90))
68+
end
69+
t[18] = "-BOUNDARY"
70+
return table.concat(t)
71+
end
72+
73+
local function encode(t, boundary)
74+
local r = {}
75+
local _t
76+
77+
-- generate a boundary if none was supplied
78+
boundary = boundary or gen_boundary()
79+
80+
for k,v in pairs(t) do
81+
tprintf(r,"--%s\r\n",boundary)
82+
_t = type(v)
83+
if _t == "string" then
84+
append_data(r, k, v, {})
85+
elseif _t == "table" then
86+
assert(v.data, "invalid input")
87+
local extra = {
88+
filename = v.filename or v.name,
89+
content_type = v.content_type or v.mimetype or "application/octet-stream",
90+
content_transfer_encoding = v.content_transfer_encoding or "binary",
91+
}
92+
append_data(r, k, v.data, extra)
93+
else
94+
error(string.format("unexpected type %s", _t))
95+
end
96+
end
97+
tprintf(r, "--%s--\r\n", boundary)
98+
return table.concat(r)
99+
end
100+
101+
102+
multipart = {
103+
---
104+
-- t is a table with the data to be encoded as multipart/form-data
105+
-- TODO: improve docs
106+
Request = function(t)
107+
local boundary = gen_boundary()
108+
local body = encode(t, boundary)
109+
return {
110+
body = body,
111+
headers = {
112+
["Content-Length"] = #body,
113+
["Content-Type"] = ("multipart/form-data; boundary=%s"):format(boundary),
114+
},
115+
}
116+
end
117+
}
118+
119+
end -- end of multipart scope

unittest/run.lua

-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,7 @@ require "lunit"
22

33
package.path = "../src/?.lua;../src/?/init.lua;".. package.path
44

5-
local OAuth = require "OAuth"
65
local console = require "lunit-console"
7-
86
require "echo_lab_madgex_com"
97
--require "twitter"
108
require "termie"

unittest/test_twitter_multipart.jpg

28.5 KB
Loading

unittest/twitter.lua

+60-8
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,72 @@
11
module(..., lunit.testcase, package.seeall)
22

3-
local consumer_key = "my consumer key (you need to change this)"
4-
local consumer_secret = "your consumer secret (you need to change this)"
3+
local OAuth = require "OAuth"
54

6-
function test()
5+
local consumer_key = "<your consumer key>"
6+
local consumer_secret = "<your consumer secret>"
7+
8+
---
9+
-- Teste requesting a token from Twitter and build the authorization url.
10+
--
11+
function testAuthorize()
12+
local OAuth = require "OAuth"
13+
714
local client = OAuth.new(consumer_key, consumer_secret, {
8-
RequestToken = "http://api.twitter.com/oauth/request_token",
15+
RequestToken = "https://api.twitter.com/oauth/request_token",
916
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
10-
AccessToken = "http://api.twitter.com/oauth/access_token"
17+
AccessToken = "https://api.twitter.com/oauth/access_token"
1118
})
19+
1220
print("requesting token")
1321
local values = client:RequestToken()
14-
local auth_url = client:BuildAuthorizationUrl()
15-
print("Test authorization at the following URL")
16-
print(auth_url)
1722
assert_table(values)
1823
assert_string(values.oauth_token)
1924
assert_string(values.oauth_token_secret)
25+
--for k,v in pairs(values) do print(k,v) end
26+
27+
local auth_url = client:BuildAuthorizationUrl()
28+
assert_string(auth_url)
29+
30+
print("Test authorization at the following URL: " .. auth_url)
31+
end
32+
33+
34+
---
35+
-- Uses the update_with_media Twitter endpoint to test a POST request with multipart/form-data encoding.
36+
--
37+
function testMultipartPost()
38+
39+
-- helper
40+
local function read_image()
41+
local f = assert(io.open([[test_twitter_multipart.jpg]], "rb"))
42+
local image_data = f:read("*a")
43+
f:close()
44+
return image_data
45+
end
46+
47+
local oauth_token = "<a valid oauth token>"
48+
local oauth_token_secret = "<and its corresponding token secret>"
49+
50+
local client = OAuth.new(consumer_key, consumer_secret, {
51+
RequestToken = "https://api.twitter.com/oauth/request_token",
52+
AuthorizeUser = {"https://api.twitter.com/oauth/authorize", method = "GET"},
53+
AccessToken = "https://api.twitter.com/oauth/access_token"
54+
}, {
55+
OAuthToken = oauth_token,
56+
OAuthTokenSecret = oauth_token_secret
57+
})
58+
59+
local helpers = require "OAuth.helpers"
60+
61+
local req = helpers.multipart.Request{
62+
status = "Hello World From Lua!" .. os.time(),
63+
["media[]"] = {
64+
filename = "@test_multipart.jpg",
65+
data = read_image()
66+
}
67+
}
68+
69+
local response_code, response_headers, response_status_line, response_body =
70+
client:PerformRequest("POST", "https://api.twitter.com/1.1/statuses/update_with_media.json", req.body, req.headers)
71+
assert_equal(200, response_code)
2072
end

0 commit comments

Comments
 (0)