sml-twitter 0.7.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/History +226 -0
- data/License +20 -0
- data/Notes +33 -0
- data/README.rdoc +19 -0
- data/Rakefile +40 -0
- data/VERSION.yml +5 -0
- data/examples/connect.rb +30 -0
- data/examples/friendship_existance.rb +13 -0
- data/examples/helpers/config_store.rb +38 -0
- data/examples/httpauth.rb +11 -0
- data/examples/ids.rb +13 -0
- data/examples/lists.rb +11 -0
- data/examples/oauth.rb +27 -0
- data/examples/search.rb +15 -0
- data/examples/timeline.rb +19 -0
- data/examples/unauthorized.rb +16 -0
- data/examples/update.rb +11 -0
- data/examples/user.rb +5 -0
- data/lib/twitter.rb +63 -0
- data/lib/twitter/base.rb +249 -0
- data/lib/twitter/httpauth.rb +35 -0
- data/lib/twitter/oauth.rb +56 -0
- data/lib/twitter/request.rb +118 -0
- data/lib/twitter/search.rb +111 -0
- data/lib/twitter/trends.rb +29 -0
- data/test/fixtures/firehose.json +1 -0
- data/test/fixtures/follower_ids.json +1 -0
- data/test/fixtures/friend_ids.json +1 -0
- data/test/fixtures/friends_timeline.json +1 -0
- data/test/fixtures/friendship.json +1 -0
- data/test/fixtures/list.json +1 -0
- data/test/fixtures/list_statuses.json +1 -0
- data/test/fixtures/list_subscriptions.json +1 -0
- data/test/fixtures/list_users.json +1 -0
- data/test/fixtures/lists.json +1 -0
- data/test/fixtures/memberships.json +1 -0
- data/test/fixtures/mentions.json +1 -0
- data/test/fixtures/rate_limit_exceeded.json +1 -0
- data/test/fixtures/search.json +1 -0
- data/test/fixtures/search_from_jnunemaker.json +1 -0
- data/test/fixtures/status.json +1 -0
- data/test/fixtures/status_show.json +1 -0
- data/test/fixtures/trends_current.json +1 -0
- data/test/fixtures/trends_current_exclude.json +1 -0
- data/test/fixtures/trends_daily.json +1 -0
- data/test/fixtures/trends_daily_date.json +1 -0
- data/test/fixtures/trends_daily_exclude.json +1 -0
- data/test/fixtures/trends_weekly.json +1 -0
- data/test/fixtures/trends_weekly_date.json +1 -0
- data/test/fixtures/trends_weekly_exclude.json +1 -0
- data/test/fixtures/user.json +1 -0
- data/test/fixtures/user_timeline.json +1 -0
- data/test/test_helper.rb +51 -0
- data/test/twitter/base_test.rb +214 -0
- data/test/twitter/httpauth_test.rb +76 -0
- data/test/twitter/oauth_test.rb +108 -0
- data/test/twitter/request_test.rb +217 -0
- data/test/twitter/search_test.rb +159 -0
- data/test/twitter/trends_test.rb +95 -0
- data/test/twitter_test.rb +38 -0
- metadata +203 -0
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
require 'test_helper'
|
|
2
|
+
|
|
3
|
+
class BaseTest < Test::Unit::TestCase
|
|
4
|
+
context "base" do
|
|
5
|
+
setup do
|
|
6
|
+
oauth = Twitter::OAuth.new('token', 'secret')
|
|
7
|
+
@access_token = OAuth::AccessToken.new(oauth.consumer, 'atoken', 'asecret')
|
|
8
|
+
oauth.stubs(:access_token).returns(@access_token)
|
|
9
|
+
@twitter = Twitter::Base.new(oauth)
|
|
10
|
+
end
|
|
11
|
+
|
|
12
|
+
context "initialize" do
|
|
13
|
+
should "require a client" do
|
|
14
|
+
@twitter.client.should respond_to(:get)
|
|
15
|
+
@twitter.client.should respond_to(:post)
|
|
16
|
+
end
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
should "delegate get to the client" do
|
|
20
|
+
@access_token.expects(:get).with('/foo').returns(nil)
|
|
21
|
+
@twitter.get('/foo')
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
should "delegate post to the client" do
|
|
25
|
+
@access_token.expects(:post).with('/foo', {:bar => 'baz'}).returns(nil)
|
|
26
|
+
@twitter.post('/foo', {:bar => 'baz'})
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
context "hitting the api" do
|
|
30
|
+
should "be able to get friends timeline" do
|
|
31
|
+
stub_get('/statuses/friends_timeline.json', 'friends_timeline.json')
|
|
32
|
+
timeline = @twitter.friends_timeline
|
|
33
|
+
timeline.size.should == 20
|
|
34
|
+
first = timeline.first
|
|
35
|
+
first.source.should == '<a href="http://www.atebits.com/software/tweetie/">Tweetie</a>'
|
|
36
|
+
first.user.name.should == 'John Nunemaker'
|
|
37
|
+
first.user.url.should == 'http://railstips.org/about'
|
|
38
|
+
first.id.should == 1441588944
|
|
39
|
+
first.favorited.should be(false)
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
should "be able to get user timeline" do
|
|
43
|
+
stub_get('/statuses/user_timeline.json', 'user_timeline.json')
|
|
44
|
+
timeline = @twitter.user_timeline
|
|
45
|
+
timeline.size.should == 20
|
|
46
|
+
first = timeline.first
|
|
47
|
+
first.text.should == 'Colder out today than expected. Headed to the Beanery for some morning wakeup drink. Latte or coffee...hmmm...'
|
|
48
|
+
first.user.name.should == 'John Nunemaker'
|
|
49
|
+
end
|
|
50
|
+
|
|
51
|
+
should "be able to get a status" do
|
|
52
|
+
stub_get('/statuses/show/1441588944.json', 'status.json')
|
|
53
|
+
status = @twitter.status(1441588944)
|
|
54
|
+
status.user.name.should == 'John Nunemaker'
|
|
55
|
+
status.id.should == 1441588944
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
should "be able to update status" do
|
|
59
|
+
stub_post('/statuses/update.json', 'status.json')
|
|
60
|
+
status = @twitter.update('Rob Dyrdek is the funniest man alive. That is all.')
|
|
61
|
+
status.user.name.should == 'John Nunemaker'
|
|
62
|
+
status.text.should == 'Rob Dyrdek is the funniest man alive. That is all.'
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
should "be able to get mentions" do
|
|
66
|
+
stub_get('/statuses/mentions.json', 'mentions.json')
|
|
67
|
+
mentions = @twitter.mentions
|
|
68
|
+
mentions.size.should == 19
|
|
69
|
+
first = mentions.first
|
|
70
|
+
first.user.name.should == '-oAk-'
|
|
71
|
+
first.text.should == '@jnunemaker cold out today. cold yesterday. even colder today.'
|
|
72
|
+
end
|
|
73
|
+
|
|
74
|
+
should "be able to get follower ids" do
|
|
75
|
+
stub_get('/followers/ids.json', 'follower_ids.json')
|
|
76
|
+
follower_ids = @twitter.follower_ids
|
|
77
|
+
follower_ids.size.should == 1252
|
|
78
|
+
follower_ids.first.should == 613
|
|
79
|
+
end
|
|
80
|
+
|
|
81
|
+
should "be able to get friend ids" do
|
|
82
|
+
stub_get('/friends/ids.json', 'friend_ids.json')
|
|
83
|
+
friend_ids = @twitter.friend_ids
|
|
84
|
+
friend_ids.size.should == 161
|
|
85
|
+
friend_ids.first.should == 15323
|
|
86
|
+
end
|
|
87
|
+
|
|
88
|
+
should "correctly hash statuses" do
|
|
89
|
+
stub_get('/statuses/friends_timeline.json', 'friends_timeline.json')
|
|
90
|
+
hashes = @twitter.friends_timeline.map{ |s| s.hash }
|
|
91
|
+
hashes.should == @twitter.friends_timeline.map{ |s| s.hash }
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
should "be able to get a friendship" do
|
|
95
|
+
stub_get("/friendships/show.json?source_screen_name=dcrec1&target_screen_name=pengwynn", 'friendship.json')
|
|
96
|
+
@twitter.friendship_show(:source_screen_name => 'dcrec1', :target_screen_name => 'pengwynn').relationship.target.followed_by == false
|
|
97
|
+
end
|
|
98
|
+
end
|
|
99
|
+
|
|
100
|
+
context "when using lists" do
|
|
101
|
+
|
|
102
|
+
should "be able to create a new list" do
|
|
103
|
+
stub_post('/pengwynn/lists.json', 'list.json')
|
|
104
|
+
list = @twitter.list_create('pengwynn', {:name => 'Rubyists'})
|
|
105
|
+
list.name.should == 'Rubyists'
|
|
106
|
+
list.slug.should == 'rubyists'
|
|
107
|
+
list.mode.should == 'public'
|
|
108
|
+
end
|
|
109
|
+
|
|
110
|
+
should "be able to update a list" do
|
|
111
|
+
stub_put('/pengwynn/lists/rubyists.json', 'list.json')
|
|
112
|
+
list = @twitter.list_update('pengwynn', 'rubyists', {:name => 'Rubyists'})
|
|
113
|
+
list.name.should == 'Rubyists'
|
|
114
|
+
list.slug.should == 'rubyists'
|
|
115
|
+
list.mode.should == 'public'
|
|
116
|
+
end
|
|
117
|
+
|
|
118
|
+
should "be able to delete a list" do
|
|
119
|
+
stub_delete('/pengwynn/lists/rubyists.json', 'list.json')
|
|
120
|
+
list = @twitter.list_delete('pengwynn', 'rubyists')
|
|
121
|
+
list.name.should == 'Rubyists'
|
|
122
|
+
list.slug.should == 'rubyists'
|
|
123
|
+
list.mode.should == 'public'
|
|
124
|
+
end
|
|
125
|
+
|
|
126
|
+
should "be able to view lists to which a user belongs" do
|
|
127
|
+
stub_get('/pengwynn/lists/memberships.json', 'memberships.json')
|
|
128
|
+
lists = @twitter.memberships('pengwynn').lists
|
|
129
|
+
lists.size.should == 16
|
|
130
|
+
lists.first.name.should == 'web-dev'
|
|
131
|
+
lists.first.member_count.should == 38
|
|
132
|
+
end
|
|
133
|
+
|
|
134
|
+
should "be able to view lists for the authenticated user" do
|
|
135
|
+
stub_get('http://api.twitter.com/1/pengwynn/lists.json', 'lists.json')
|
|
136
|
+
lists = @twitter.lists('pengwynn').lists
|
|
137
|
+
lists.size.should == 1
|
|
138
|
+
lists.first.name.should == 'Rubyists'
|
|
139
|
+
lists.first.slug.should == 'rubyists'
|
|
140
|
+
end
|
|
141
|
+
|
|
142
|
+
should "be able to view list details" do
|
|
143
|
+
stub_get('/pengwynn/lists/rubyists.json', 'list.json')
|
|
144
|
+
list = @twitter.list('pengwynn', 'rubyists')
|
|
145
|
+
list.name.should == 'Rubyists'
|
|
146
|
+
list.subscriber_count.should == 0
|
|
147
|
+
end
|
|
148
|
+
|
|
149
|
+
should "be able to view list timeline" do
|
|
150
|
+
stub_get('/pengwynn/lists/rubyists/statuses.json', 'list_statuses.json')
|
|
151
|
+
tweets = @twitter.list_timeline('pengwynn', 'rubyists')
|
|
152
|
+
tweets.size.should == 20
|
|
153
|
+
tweets.first.id.should == 5272535583
|
|
154
|
+
tweets.first.user.name.should == 'John Nunemaker'
|
|
155
|
+
end
|
|
156
|
+
|
|
157
|
+
should "be able to view list members" do
|
|
158
|
+
stub_get('/pengwynn/rubyists/members.json', 'list_users.json')
|
|
159
|
+
members = @twitter.list_members('pengwynn', 'rubyists').users
|
|
160
|
+
members.size.should == 2
|
|
161
|
+
members.first.name.should == 'John Nunemaker'
|
|
162
|
+
members.first.screen_name.should == 'jnunemaker'
|
|
163
|
+
end
|
|
164
|
+
|
|
165
|
+
should "be able to add a member to a list" do
|
|
166
|
+
stub_post('/pengwynn/rubyists/members.json', 'user.json')
|
|
167
|
+
user = @twitter.list_add_member('pengwynn', 'rubyists', 4243)
|
|
168
|
+
user.screen_name.should == 'jnunemaker'
|
|
169
|
+
end
|
|
170
|
+
|
|
171
|
+
should "be able to remove a member from a list" do
|
|
172
|
+
stub_delete('/pengwynn/rubyists/members.json', 'user.json')
|
|
173
|
+
user = @twitter.list_remove_member('pengwynn', 'rubyists', 4243)
|
|
174
|
+
user.screen_name.should == 'jnunemaker'
|
|
175
|
+
end
|
|
176
|
+
|
|
177
|
+
should "be able to check if a user is member of a list" do
|
|
178
|
+
stub_get('/pengwynn/rubyists/members/4243.json', 'user.json')
|
|
179
|
+
@twitter.is_list_member?('pengwynn', 'rubyists', 4243).should == true
|
|
180
|
+
end
|
|
181
|
+
|
|
182
|
+
should "be able to view list subscribers" do
|
|
183
|
+
stub_get('/pengwynn/rubyists/subscribers.json', 'list_users.json')
|
|
184
|
+
subscribers = @twitter.list_subscribers('pengwynn', 'rubyists').users
|
|
185
|
+
subscribers.size.should == 2
|
|
186
|
+
subscribers.first.name.should == 'John Nunemaker'
|
|
187
|
+
subscribers.first.screen_name.should == 'jnunemaker'
|
|
188
|
+
end
|
|
189
|
+
|
|
190
|
+
should "be able to subscribe to a list" do
|
|
191
|
+
stub_post('/pengwynn/rubyists/subscribers.json', 'user.json')
|
|
192
|
+
user = @twitter.list_subscribe('pengwynn', 'rubyists')
|
|
193
|
+
user.screen_name.should == 'jnunemaker'
|
|
194
|
+
end
|
|
195
|
+
|
|
196
|
+
should "be able to unsubscribe from a list" do
|
|
197
|
+
stub_delete('/pengwynn/rubyists/subscribers.json', 'user.json')
|
|
198
|
+
user = @twitter.list_unsubscribe('pengwynn', 'rubyists')
|
|
199
|
+
user.screen_name.should == 'jnunemaker'
|
|
200
|
+
end
|
|
201
|
+
|
|
202
|
+
should "be able to view a members list subscriptions" do
|
|
203
|
+
stub_get('/pengwynn/lists/subscriptions.json', 'list_subscriptions.json')
|
|
204
|
+
subscriptions = @twitter.list_subscriptions('pengwynn').lists
|
|
205
|
+
subscriptions.size.should == 1
|
|
206
|
+
subscriptions.first.full_name.should == '@chriseppstein/sass-users'
|
|
207
|
+
subscriptions.first.slug.should == 'sass-users'
|
|
208
|
+
end
|
|
209
|
+
|
|
210
|
+
end
|
|
211
|
+
|
|
212
|
+
|
|
213
|
+
end
|
|
214
|
+
end
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
require 'test_helper'
|
|
2
|
+
|
|
3
|
+
class HTTPAuthTest < Test::Unit::TestCase
|
|
4
|
+
context "Creating new instance" do
|
|
5
|
+
should "should take user and password" do
|
|
6
|
+
twitter = Twitter::HTTPAuth.new('username', 'password')
|
|
7
|
+
twitter.username.should == 'username'
|
|
8
|
+
twitter.password.should == 'password'
|
|
9
|
+
end
|
|
10
|
+
|
|
11
|
+
should "accept options" do
|
|
12
|
+
twitter = Twitter::HTTPAuth.new('username', 'password', :ssl => true)
|
|
13
|
+
twitter.options.should == {:ssl => true}
|
|
14
|
+
end
|
|
15
|
+
|
|
16
|
+
should "default ssl to false" do
|
|
17
|
+
twitter = Twitter::HTTPAuth.new('username', 'password')
|
|
18
|
+
twitter.options[:ssl].should be(false)
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
should "use https if ssl is true" do
|
|
22
|
+
Twitter::HTTPAuth.expects(:base_uri).with('https://twitter.com')
|
|
23
|
+
twitter = Twitter::HTTPAuth.new('username', 'password', :ssl => true)
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
should "use http if ssl is false" do
|
|
27
|
+
Twitter::HTTPAuth.expects(:base_uri).with('http://twitter.com')
|
|
28
|
+
twitter = Twitter::HTTPAuth.new('username', 'password', :ssl => false)
|
|
29
|
+
end
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
context "Client methods" do
|
|
33
|
+
setup do
|
|
34
|
+
@twitter = Twitter::HTTPAuth.new('username', 'password')
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
should "not throw error when accessing response message" do
|
|
38
|
+
stub_get('http://twitter.com:80/statuses/user_timeline.json', 'user_timeline.json')
|
|
39
|
+
response = @twitter.get('/statuses/user_timeline.json')
|
|
40
|
+
response.message.should == 'OK'
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
should "be able to get" do
|
|
44
|
+
stub_get('http://username:password@twitter.com:80/statuses/user_timeline.json', 'user_timeline.json')
|
|
45
|
+
response = @twitter.get('/statuses/user_timeline.json')
|
|
46
|
+
response.should == fixture_file('user_timeline.json')
|
|
47
|
+
end
|
|
48
|
+
|
|
49
|
+
should "be able to get with headers" do
|
|
50
|
+
@twitter.class.expects(:get).with(
|
|
51
|
+
'/statuses/user_timeline.json', {
|
|
52
|
+
:basic_auth => {:username => 'username', :password => 'password'},
|
|
53
|
+
:headers => {'Foo' => 'Bar'}
|
|
54
|
+
}
|
|
55
|
+
).returns(fixture_file('user_timeline.json'))
|
|
56
|
+
@twitter.get('/statuses/user_timeline.json', {'Foo' => 'Bar'})
|
|
57
|
+
end
|
|
58
|
+
|
|
59
|
+
should "be able to post" do
|
|
60
|
+
stub_post('http://username:password@twitter.com:80/statuses/update.json', 'status.json')
|
|
61
|
+
response = @twitter.post('/statuses/update.json', :text => 'My update.')
|
|
62
|
+
response.should == fixture_file('status.json')
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
should "be able to post with headers" do
|
|
66
|
+
@twitter.class.expects(:post).with(
|
|
67
|
+
'/statuses/update.json', {
|
|
68
|
+
:headers => {'Foo' => 'Bar'},
|
|
69
|
+
:body => {:text => 'My update.'},
|
|
70
|
+
:basic_auth => {:username => 'username', :password => 'password'}
|
|
71
|
+
}
|
|
72
|
+
).returns(fixture_file('status.json'))
|
|
73
|
+
@twitter.post('/statuses/update.json', {:text => 'My update.'}, {'Foo' => 'Bar'})
|
|
74
|
+
end
|
|
75
|
+
end
|
|
76
|
+
end
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
require 'test_helper'
|
|
2
|
+
|
|
3
|
+
class OAuthTest < Test::Unit::TestCase
|
|
4
|
+
should "initialize with consumer token and secret" do
|
|
5
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
6
|
+
|
|
7
|
+
twitter.ctoken.should == 'token'
|
|
8
|
+
twitter.csecret.should == 'secret'
|
|
9
|
+
end
|
|
10
|
+
|
|
11
|
+
should "set autorization path to '/oauth/authorize' by default" do
|
|
12
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
13
|
+
twitter.consumer.options[:authorize_path].should == '/oauth/authorize'
|
|
14
|
+
end
|
|
15
|
+
|
|
16
|
+
should "set autorization path to '/oauth/authenticate' if sign_in_with_twitter" do
|
|
17
|
+
twitter = Twitter::OAuth.new('token', 'secret', :sign_in => true)
|
|
18
|
+
twitter.consumer.options[:authorize_path].should == '/oauth/authenticate'
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
should "have a consumer" do
|
|
22
|
+
consumer = mock('oauth consumer')
|
|
23
|
+
OAuth::Consumer.expects(:new).with('token', 'secret', {:site => 'http://twitter.com'}).returns(consumer)
|
|
24
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
25
|
+
|
|
26
|
+
twitter.consumer.should == consumer
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
should "have a request token from the consumer" do
|
|
30
|
+
consumer = mock('oauth consumer')
|
|
31
|
+
request_token = mock('request token')
|
|
32
|
+
consumer.expects(:get_request_token).returns(request_token)
|
|
33
|
+
OAuth::Consumer.expects(:new).with('token', 'secret', {:site => 'http://twitter.com'}).returns(consumer)
|
|
34
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
35
|
+
|
|
36
|
+
twitter.request_token.should == request_token
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
context "set_callback_url" do
|
|
40
|
+
should "clear request token and set the callback url" do
|
|
41
|
+
consumer = mock('oauth consumer')
|
|
42
|
+
request_token = mock('request token')
|
|
43
|
+
|
|
44
|
+
OAuth::Consumer.
|
|
45
|
+
expects(:new).
|
|
46
|
+
with('token', 'secret', {:site => 'http://twitter.com'}).
|
|
47
|
+
returns(consumer)
|
|
48
|
+
|
|
49
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
50
|
+
|
|
51
|
+
consumer.
|
|
52
|
+
expects(:get_request_token).
|
|
53
|
+
with({:oauth_callback => 'http://myapp.com/oauth_callback'})
|
|
54
|
+
|
|
55
|
+
twitter.set_callback_url('http://myapp.com/oauth_callback')
|
|
56
|
+
end
|
|
57
|
+
end
|
|
58
|
+
|
|
59
|
+
should "be able to create access token from request token, request secret and verifier" do
|
|
60
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
61
|
+
consumer = OAuth::Consumer.new('token', 'secret', {:site => 'http://twitter.com'})
|
|
62
|
+
twitter.stubs(:consumer).returns(consumer)
|
|
63
|
+
|
|
64
|
+
access_token = mock('access token', :token => 'atoken', :secret => 'asecret')
|
|
65
|
+
request_token = mock('request token')
|
|
66
|
+
request_token.
|
|
67
|
+
expects(:get_access_token).
|
|
68
|
+
with(:oauth_verifier => 'verifier').
|
|
69
|
+
returns(access_token)
|
|
70
|
+
|
|
71
|
+
OAuth::RequestToken.
|
|
72
|
+
expects(:new).
|
|
73
|
+
with(consumer, 'rtoken', 'rsecret').
|
|
74
|
+
returns(request_token)
|
|
75
|
+
|
|
76
|
+
twitter.authorize_from_request('rtoken', 'rsecret', 'verifier')
|
|
77
|
+
twitter.access_token.class.should be(OAuth::AccessToken)
|
|
78
|
+
twitter.access_token.token.should == 'atoken'
|
|
79
|
+
twitter.access_token.secret.should == 'asecret'
|
|
80
|
+
end
|
|
81
|
+
|
|
82
|
+
should "be able to create access token from access token and secret" do
|
|
83
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
84
|
+
consumer = OAuth::Consumer.new('token', 'secret', {:site => 'http://twitter.com'})
|
|
85
|
+
twitter.stubs(:consumer).returns(consumer)
|
|
86
|
+
|
|
87
|
+
twitter.authorize_from_access('atoken', 'asecret')
|
|
88
|
+
twitter.access_token.class.should be(OAuth::AccessToken)
|
|
89
|
+
twitter.access_token.token.should == 'atoken'
|
|
90
|
+
twitter.access_token.secret.should == 'asecret'
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
should "delegate get to access token" do
|
|
94
|
+
access_token = mock('access token')
|
|
95
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
96
|
+
twitter.stubs(:access_token).returns(access_token)
|
|
97
|
+
access_token.expects(:get).returns(nil)
|
|
98
|
+
twitter.get('/foo')
|
|
99
|
+
end
|
|
100
|
+
|
|
101
|
+
should "delegate post to access token" do
|
|
102
|
+
access_token = mock('access token')
|
|
103
|
+
twitter = Twitter::OAuth.new('token', 'secret')
|
|
104
|
+
twitter.stubs(:access_token).returns(access_token)
|
|
105
|
+
access_token.expects(:post).returns(nil)
|
|
106
|
+
twitter.post('/foo')
|
|
107
|
+
end
|
|
108
|
+
end
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
require 'test_helper'
|
|
2
|
+
|
|
3
|
+
class RequestTest < Test::Unit::TestCase
|
|
4
|
+
context "new get request" do
|
|
5
|
+
setup do
|
|
6
|
+
@client = mock('twitter client')
|
|
7
|
+
@request = Twitter::Request.new(@client, :get, '/statuses/user_timeline.json', {:query => {:since_id => 1234}})
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
should "have client" do
|
|
11
|
+
@request.client.should == @client
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
should "have method" do
|
|
15
|
+
@request.method.should == :get
|
|
16
|
+
end
|
|
17
|
+
|
|
18
|
+
should "have path" do
|
|
19
|
+
@request.path.should == '/statuses/user_timeline.json'
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
should "have options" do
|
|
23
|
+
@request.options[:query].should == {:since_id => 1234}
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
should "have uri" do
|
|
27
|
+
@request.uri.should == '/statuses/user_timeline.json?since_id=1234'
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
context "performing request for collection" do
|
|
31
|
+
setup do
|
|
32
|
+
response = mock('response') do
|
|
33
|
+
stubs(:body).returns(fixture_file('user_timeline.json'))
|
|
34
|
+
stubs(:code).returns('200')
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
@client.expects(:get).returns(response)
|
|
38
|
+
@object = @request.perform
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
should "return array of mashes" do
|
|
42
|
+
@object.size.should == 20
|
|
43
|
+
@object.each { |obj| obj.class.should be(Mash) }
|
|
44
|
+
@object.first.text.should == 'Colder out today than expected. Headed to the Beanery for some morning wakeup drink. Latte or coffee...hmmm...'
|
|
45
|
+
end
|
|
46
|
+
end
|
|
47
|
+
|
|
48
|
+
context "performing a request for a single object" do
|
|
49
|
+
setup do
|
|
50
|
+
response = mock('response') do
|
|
51
|
+
stubs(:body).returns(fixture_file('status.json'))
|
|
52
|
+
stubs(:code).returns('200')
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
@client.expects(:get).returns(response)
|
|
56
|
+
@object = @request.perform
|
|
57
|
+
end
|
|
58
|
+
|
|
59
|
+
should "return a single mash" do
|
|
60
|
+
@object.class.should be(Mash)
|
|
61
|
+
@object.text.should == 'Rob Dyrdek is the funniest man alive. That is all.'
|
|
62
|
+
end
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
context "with no query string" do
|
|
66
|
+
should "not have any query string" do
|
|
67
|
+
request = Twitter::Request.new(@client, :get, '/statuses/user_timeline.json')
|
|
68
|
+
request.uri.should == '/statuses/user_timeline.json'
|
|
69
|
+
end
|
|
70
|
+
end
|
|
71
|
+
|
|
72
|
+
context "with blank query string" do
|
|
73
|
+
should "not have any query string" do
|
|
74
|
+
request = Twitter::Request.new(@client, :get, '/statuses/user_timeline.json', :query => {})
|
|
75
|
+
request.uri.should == '/statuses/user_timeline.json'
|
|
76
|
+
end
|
|
77
|
+
end
|
|
78
|
+
|
|
79
|
+
should "have get shortcut to initialize and perform all in one" do
|
|
80
|
+
Twitter::Request.any_instance.expects(:perform).returns(nil)
|
|
81
|
+
Twitter::Request.get(@client, '/foo')
|
|
82
|
+
end
|
|
83
|
+
|
|
84
|
+
should "allow setting query string and headers" do
|
|
85
|
+
response = mock('response') do
|
|
86
|
+
stubs(:body).returns('')
|
|
87
|
+
stubs(:code).returns('200')
|
|
88
|
+
end
|
|
89
|
+
|
|
90
|
+
@client.expects(:get).with('/statuses/friends_timeline.json?since_id=1234', {'Foo' => 'Bar'}).returns(response)
|
|
91
|
+
Twitter::Request.get(@client, '/statuses/friends_timeline.json?since_id=1234', :headers => {'Foo' => 'Bar'})
|
|
92
|
+
end
|
|
93
|
+
end
|
|
94
|
+
|
|
95
|
+
context "new post request" do
|
|
96
|
+
setup do
|
|
97
|
+
@client = mock('twitter client')
|
|
98
|
+
@request = Twitter::Request.new(@client, :post, '/statuses/update.json', {:body => {:status => 'Woohoo!'}})
|
|
99
|
+
end
|
|
100
|
+
|
|
101
|
+
should "allow setting body and headers" do
|
|
102
|
+
response = mock('response') do
|
|
103
|
+
stubs(:body).returns('')
|
|
104
|
+
stubs(:code).returns('200')
|
|
105
|
+
end
|
|
106
|
+
|
|
107
|
+
@client.expects(:post).with('/statuses/update.json', {:status => 'Woohoo!'}, {'Foo' => 'Bar'}).returns(response)
|
|
108
|
+
Twitter::Request.post(@client, '/statuses/update.json', :body => {:status => 'Woohoo!'}, :headers => {'Foo' => 'Bar'})
|
|
109
|
+
end
|
|
110
|
+
|
|
111
|
+
context "performing request" do
|
|
112
|
+
setup do
|
|
113
|
+
response = mock('response') do
|
|
114
|
+
stubs(:body).returns(fixture_file('status.json'))
|
|
115
|
+
stubs(:code).returns('200')
|
|
116
|
+
end
|
|
117
|
+
|
|
118
|
+
@client.expects(:post).returns(response)
|
|
119
|
+
@object = @request.perform
|
|
120
|
+
end
|
|
121
|
+
|
|
122
|
+
should "return a mash of the object" do
|
|
123
|
+
@object.text.should == 'Rob Dyrdek is the funniest man alive. That is all.'
|
|
124
|
+
end
|
|
125
|
+
end
|
|
126
|
+
|
|
127
|
+
should "have post shortcut to initialize and perform all in one" do
|
|
128
|
+
Twitter::Request.any_instance.expects(:perform).returns(nil)
|
|
129
|
+
Twitter::Request.post(@client, '/foo')
|
|
130
|
+
end
|
|
131
|
+
end
|
|
132
|
+
|
|
133
|
+
context "error raising" do
|
|
134
|
+
setup do
|
|
135
|
+
oauth = Twitter::OAuth.new('token', 'secret')
|
|
136
|
+
oauth.authorize_from_access('atoken', 'asecret')
|
|
137
|
+
@client = Twitter::Base.new(oauth)
|
|
138
|
+
end
|
|
139
|
+
|
|
140
|
+
should "not raise error for 200" do
|
|
141
|
+
stub_get('http://twitter.com:80/foo', '', ['200'])
|
|
142
|
+
lambda {
|
|
143
|
+
Twitter::Request.get(@client, '/foo')
|
|
144
|
+
}.should_not raise_error
|
|
145
|
+
end
|
|
146
|
+
|
|
147
|
+
should "not raise error for 304" do
|
|
148
|
+
stub_get('http://twitter.com:80/foo', '', ['304'])
|
|
149
|
+
lambda {
|
|
150
|
+
Twitter::Request.get(@client, '/foo')
|
|
151
|
+
}.should_not raise_error
|
|
152
|
+
end
|
|
153
|
+
|
|
154
|
+
should "raise RateLimitExceeded for 400" do
|
|
155
|
+
stub_get('http://twitter.com:80/foo', 'rate_limit_exceeded.json', ['400'])
|
|
156
|
+
lambda {
|
|
157
|
+
Twitter::Request.get(@client, '/foo')
|
|
158
|
+
}.should raise_error(Twitter::RateLimitExceeded)
|
|
159
|
+
end
|
|
160
|
+
|
|
161
|
+
should "raise Unauthorized for 401" do
|
|
162
|
+
stub_get('http://twitter.com:80/foo', '', ['401'])
|
|
163
|
+
lambda {
|
|
164
|
+
Twitter::Request.get(@client, '/foo')
|
|
165
|
+
}.should raise_error(Twitter::Unauthorized)
|
|
166
|
+
end
|
|
167
|
+
|
|
168
|
+
should "raise General for 403" do
|
|
169
|
+
stub_get('http://twitter.com:80/foo', '', ['403'])
|
|
170
|
+
lambda {
|
|
171
|
+
Twitter::Request.get(@client, '/foo')
|
|
172
|
+
}.should raise_error(Twitter::General)
|
|
173
|
+
end
|
|
174
|
+
|
|
175
|
+
should "raise NotFound for 404" do
|
|
176
|
+
stub_get('http://twitter.com:80/foo', '', ['404'])
|
|
177
|
+
lambda {
|
|
178
|
+
Twitter::Request.get(@client, '/foo')
|
|
179
|
+
}.should raise_error(Twitter::NotFound)
|
|
180
|
+
end
|
|
181
|
+
|
|
182
|
+
should "raise InformTwitter for 500" do
|
|
183
|
+
stub_get('http://twitter.com:80/foo', '', ['500'])
|
|
184
|
+
lambda {
|
|
185
|
+
Twitter::Request.get(@client, '/foo')
|
|
186
|
+
}.should raise_error(Twitter::InformTwitter)
|
|
187
|
+
end
|
|
188
|
+
|
|
189
|
+
should "raise Unavailable for 502" do
|
|
190
|
+
stub_get('http://twitter.com:80/foo', '', ['502'])
|
|
191
|
+
lambda {
|
|
192
|
+
Twitter::Request.get(@client, '/foo')
|
|
193
|
+
}.should raise_error(Twitter::Unavailable)
|
|
194
|
+
end
|
|
195
|
+
|
|
196
|
+
should "raise Unavailable for 503" do
|
|
197
|
+
stub_get('http://twitter.com:80/foo', '', ['503'])
|
|
198
|
+
lambda {
|
|
199
|
+
Twitter::Request.get(@client, '/foo')
|
|
200
|
+
}.should raise_error(Twitter::Unavailable)
|
|
201
|
+
end
|
|
202
|
+
end
|
|
203
|
+
|
|
204
|
+
context "Making request with mash option set to false" do
|
|
205
|
+
setup do
|
|
206
|
+
oauth = Twitter::OAuth.new('token', 'secret')
|
|
207
|
+
oauth.authorize_from_access('atoken', 'asecret')
|
|
208
|
+
@client = Twitter::Base.new(oauth)
|
|
209
|
+
end
|
|
210
|
+
|
|
211
|
+
should "not attempt to create mash of return object" do
|
|
212
|
+
stub_get('http://twitter.com:80/foo', 'friend_ids.json')
|
|
213
|
+
object = Twitter::Request.get(@client, '/foo', :mash => false)
|
|
214
|
+
object.class.should_not be(Mash)
|
|
215
|
+
end
|
|
216
|
+
end
|
|
217
|
+
end
|