twitter-jruby 0.9.5.2010050701

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.
Files changed (88) hide show
  1. data/History +282 -0
  2. data/License +20 -0
  3. data/Notes +33 -0
  4. data/README.rdoc +27 -0
  5. data/Rakefile +39 -0
  6. data/VERSION.yml +5 -0
  7. data/examples/connect.rb +30 -0
  8. data/examples/friendship_existance.rb +13 -0
  9. data/examples/helpers/config_store.rb +38 -0
  10. data/examples/httpauth.rb +11 -0
  11. data/examples/ids.rb +13 -0
  12. data/examples/lists.rb +11 -0
  13. data/examples/oauth.rb +27 -0
  14. data/examples/search.rb +15 -0
  15. data/examples/timeline.rb +19 -0
  16. data/examples/tumblr.rb +9 -0
  17. data/examples/unauthorized.rb +16 -0
  18. data/examples/update.rb +11 -0
  19. data/examples/user.rb +5 -0
  20. data/lib/twitter.rb +146 -0
  21. data/lib/twitter/base.rb +390 -0
  22. data/lib/twitter/httpauth.rb +39 -0
  23. data/lib/twitter/local_trends.rb +15 -0
  24. data/lib/twitter/oauth.rb +58 -0
  25. data/lib/twitter/request.rb +71 -0
  26. data/lib/twitter/search.rb +159 -0
  27. data/lib/twitter/trends.rb +41 -0
  28. data/test/fixtures/blocking.json +1632 -0
  29. data/test/fixtures/firehose.json +1 -0
  30. data/test/fixtures/follower_ids.json +1 -0
  31. data/test/fixtures/followers.json +1 -0
  32. data/test/fixtures/friend_ids.json +1 -0
  33. data/test/fixtures/friends_timeline.json +1 -0
  34. data/test/fixtures/friendship.json +1 -0
  35. data/test/fixtures/friendship_exists.json +1 -0
  36. data/test/fixtures/home_timeline.json +1 -0
  37. data/test/fixtures/ids.json +1 -0
  38. data/test/fixtures/list.json +1 -0
  39. data/test/fixtures/list_statuses.json +1 -0
  40. data/test/fixtures/list_statuses_1_1.json +1 -0
  41. data/test/fixtures/list_statuses_2_1.json +1 -0
  42. data/test/fixtures/list_subscriptions.json +1 -0
  43. data/test/fixtures/list_users.json +1 -0
  44. data/test/fixtures/lists.json +1 -0
  45. data/test/fixtures/memberships.json +1 -0
  46. data/test/fixtures/mentions.json +1 -0
  47. data/test/fixtures/not_found.json +1 -0
  48. data/test/fixtures/people_search.json +39 -0
  49. data/test/fixtures/rate_limit_exceeded.json +1 -0
  50. data/test/fixtures/report_spam.json +41 -0
  51. data/test/fixtures/retweet.json +1 -0
  52. data/test/fixtures/retweeted_by_me.json +1 -0
  53. data/test/fixtures/retweeted_to_me.json +1 -0
  54. data/test/fixtures/retweeters_of_tweet.json +166 -0
  55. data/test/fixtures/retweets.json +1 -0
  56. data/test/fixtures/retweets_of_me.json +1 -0
  57. data/test/fixtures/sample-image.png +0 -0
  58. data/test/fixtures/saved_search.json +7 -0
  59. data/test/fixtures/saved_searches.json +16 -0
  60. data/test/fixtures/search.json +1 -0
  61. data/test/fixtures/search_from_jnunemaker.json +1 -0
  62. data/test/fixtures/status.json +1 -0
  63. data/test/fixtures/status_show.json +1 -0
  64. data/test/fixtures/trends_available.json +253 -0
  65. data/test/fixtures/trends_current.json +1 -0
  66. data/test/fixtures/trends_current_exclude.json +1 -0
  67. data/test/fixtures/trends_daily.json +1925 -0
  68. data/test/fixtures/trends_daily_date.json +1 -0
  69. data/test/fixtures/trends_daily_exclude.json +1 -0
  70. data/test/fixtures/trends_location.json +57 -0
  71. data/test/fixtures/trends_weekly.json +1 -0
  72. data/test/fixtures/trends_weekly_date.json +1 -0
  73. data/test/fixtures/trends_weekly_exclude.json +1 -0
  74. data/test/fixtures/unauthorized.json +1 -0
  75. data/test/fixtures/update_profile_background_image.json +1 -0
  76. data/test/fixtures/update_profile_image.json +1 -0
  77. data/test/fixtures/user.json +1 -0
  78. data/test/fixtures/user_timeline.json +710 -0
  79. data/test/fixtures/users.json +1 -0
  80. data/test/test_helper.rb +46 -0
  81. data/test/twitter/base_test.rb +426 -0
  82. data/test/twitter/httpauth_test.rb +76 -0
  83. data/test/twitter/oauth_test.rb +108 -0
  84. data/test/twitter/request_test.rb +217 -0
  85. data/test/twitter/search_test.rb +208 -0
  86. data/test/twitter/trends_test.rb +112 -0
  87. data/test/twitter_test.rb +106 -0
  88. metadata +280 -0
@@ -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://api.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://api.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://api.twitter.com:80/1/statuses/user_timeline.json', 'user_timeline.json')
39
+ response = @twitter.get('/1/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@api.twitter.com:80/1/statuses/user_timeline.json', 'user_timeline.json')
45
+ response = @twitter.get('/1/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
+ '/1/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('/1/statuses/user_timeline.json', {'Foo' => 'Bar'})
57
+ end
58
+
59
+ should "be able to post" do
60
+ stub_post('http://username:password@api.twitter.com:80/1/statuses/update.json', 'status.json')
61
+ response = @twitter.post('/1/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
+ '/1/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('/1/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://api.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://api.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://api.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://api.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://api.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, '/1/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 == '/1/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 == '/1/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(Hashie::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(Hashie::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, '/1/statuses/user_timeline.json')
68
+ request.uri.should == '/1/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, '/1/statuses/user_timeline.json', :query => {})
75
+ request.uri.should == '/1/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('/1/statuses/friends_timeline.json?since_id=1234', {'Foo' => 'Bar'}).returns(response)
91
+ Twitter::Request.get(@client, '/1/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, '/1/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('/1/statuses/update.json', {:status => 'Woohoo!'}, {'Foo' => 'Bar'}).returns(response)
108
+ Twitter::Request.post(@client, '/1/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://api.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://api.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://api.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://api.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://api.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://api.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://api.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://api.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://api.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://api.twitter.com:80/foo', 'friend_ids.json')
213
+ object = Twitter::Request.get(@client, '/foo', :mash => false)
214
+ object.class.should_not be(Hashie::Mash)
215
+ end
216
+ end
217
+ end
@@ -0,0 +1,208 @@
1
+ require 'test_helper'
2
+
3
+ class SearchTest < Test::Unit::TestCase
4
+ context "searching" do
5
+ setup do
6
+ @search = Twitter::Search.new
7
+ end
8
+
9
+ should "should be able to initialize with a search term" do
10
+ Twitter::Search.new('httparty').query[:q].should include('httparty')
11
+ end
12
+
13
+ should "default user agent to Ruby Twitter Gem" do
14
+ search = Twitter::Search.new('foo')
15
+ search.user_agent.should == 'Ruby Twitter Gem'
16
+ end
17
+
18
+ should "allow overriding default user agent" do
19
+ search = Twitter::Search.new('foo', :user_agent => 'Foobar')
20
+ search.user_agent.should == 'Foobar'
21
+ end
22
+
23
+ should "pass user agent along with headers when making request" do
24
+ Twitter::Search.expects(:get).with('http://search.twitter.com/search.json', {:format => :json, :query => {:q => 'foo'}, :headers => {'User-Agent' => 'Foobar'}})
25
+ Twitter::Search.new('foo', :user_agent => 'Foobar').fetch()
26
+ end
27
+
28
+ should "should be able to specify from" do
29
+ @search.from('jnunemaker').query[:q].should include('from:jnunemaker')
30
+ end
31
+
32
+ should "should be able to specify not from" do
33
+ @search.from('jnunemaker',true).query[:q].should include('-from:jnunemaker')
34
+ end
35
+
36
+ should "should be able to specify to" do
37
+ @search.to('jnunemaker').query[:q].should include('to:jnunemaker')
38
+ end
39
+
40
+ should "should be able to specify not to" do
41
+ @search.to('jnunemaker',true).query[:q].should include('-to:jnunemaker')
42
+ end
43
+
44
+ should "should be able to specify not referencing" do
45
+ @search.referencing('jnunemaker',true).query[:q].should include('-@jnunemaker')
46
+ end
47
+
48
+ should "should alias references to referencing" do
49
+ @search.references('jnunemaker').query[:q].should include('@jnunemaker')
50
+ end
51
+
52
+ should "should alias ref to referencing" do
53
+ @search.ref('jnunemaker').query[:q].should include('@jnunemaker')
54
+ end
55
+
56
+ should "should be able to specify containing" do
57
+ @search.containing('milk').query[:q].should include('milk')
58
+ end
59
+
60
+ should "should be able to specify not containing" do
61
+ @search.containing('milk',true).query[:q].should include('-milk')
62
+ end
63
+
64
+ should "should alias contains to containing" do
65
+ @search.contains('milk').query[:q].should include('milk')
66
+ end
67
+
68
+ should "should be able to specify hashed" do
69
+ @search.hashed('twitter').query[:q].should include('#twitter')
70
+ end
71
+
72
+ should "should be able to specify not hashed" do
73
+ @search.hashed('twitter',true).query[:q].should include('-#twitter')
74
+ end
75
+
76
+ should "should be able to specify the language" do
77
+ @search.lang('en')
78
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:lang => 'en', :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
79
+ @search.fetch()
80
+ end
81
+
82
+ should "should be able to specify the number of results per page" do
83
+ @search.per_page(25)
84
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:rpp => 25, :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
85
+ @search.fetch()
86
+ end
87
+
88
+ should "should be able to specify the page number" do
89
+ @search.page(20)
90
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:page => 20, :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
91
+ @search.fetch()
92
+ end
93
+
94
+ should "should be able to specify only returning results greater than an id" do
95
+ @search.since(1234)
96
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:since_id => 1234, :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
97
+ @search.fetch()
98
+ end
99
+
100
+ should "should be able to specify since a date" do
101
+ @search.since_date('2009-04-14')
102
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => { :since => '2009-04-14', :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({ 'foo' => 'bar'})
103
+ @search.fetch
104
+ end
105
+
106
+ should "should be able to specify until a date" do
107
+ @search.until_date('2009-04-14')
108
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => { :until => '2009-04-14', :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({ 'foo' => 'bar'})
109
+ @search.fetch
110
+ end
111
+
112
+ should "should be able to specify geo coordinates" do
113
+ @search.geocode('40.757929', '-73.985506', '25mi')
114
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:geocode => '40.757929,-73.985506,25mi', :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
115
+ @search.fetch()
116
+ end
117
+
118
+ should "should be able to specify max id" do
119
+ @search.max(1234)
120
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:max_id => 1234, :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
121
+ @search.fetch()
122
+ end
123
+
124
+ should "should be able to set the phrase" do
125
+ @search.phrase("Who Dat")
126
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:phrase => "Who Dat", :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
127
+ @search.fetch()
128
+ end
129
+
130
+ should "should be able to set the result type" do
131
+ @search.result_type("popular")
132
+ @search.class.expects(:get).with('http://search.twitter.com/search.json', :query => {:result_type => 'popular', :q => ''}, :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
133
+ @search.fetch()
134
+ end
135
+
136
+ should "should be able to clear the filters set" do
137
+ @search.from('jnunemaker').to('oaknd1')
138
+ @search.clear.query.should == {:q => []}
139
+ end
140
+
141
+ should "should be able to chain methods together" do
142
+ @search.from('jnunemaker').to('oaknd1').referencing('orderedlist').containing('milk').hashed('twitter').lang('en').per_page(20).since(1234).geocode('40.757929', '-73.985506', '25mi')
143
+ @search.query[:q].should == ['from:jnunemaker', 'to:oaknd1', '@orderedlist', 'milk', '#twitter']
144
+ @search.query[:lang].should == 'en'
145
+ @search.query[:rpp].should == 20
146
+ @search.query[:since_id].should == 1234
147
+ @search.query[:geocode].should == '40.757929,-73.985506,25mi'
148
+ end
149
+
150
+ context "fetching" do
151
+ setup do
152
+ stub_get('http://search.twitter.com:80/search.json?q=%40jnunemaker', 'search.json')
153
+ @search = Twitter::Search.new('@jnunemaker')
154
+ @response = @search.fetch
155
+ end
156
+
157
+ should "should return results" do
158
+ @response.results.size.should == 15
159
+ end
160
+
161
+ should "should support dot notation" do
162
+ first = @response.results.first
163
+ first.text.should == %q(Someone asked about a tweet reader. Easy to do in ruby with @jnunemaker's twitter gem and the win32-sapi gem, if you are on windows.)
164
+ first.from_user.should == 'PatParslow'
165
+ end
166
+
167
+ should "cache fetched results so multiple fetches don't keep hitting api" do
168
+ Twitter::Search.expects(:get).never
169
+ @search.fetch
170
+ end
171
+
172
+ should "rehit api if fetch is called with true" do
173
+ Twitter::Search.expects(:get).once
174
+ @search.fetch(true)
175
+ end
176
+
177
+ should "tell if another page is available" do
178
+ @search.next_page?.should == true
179
+ end
180
+
181
+ should "be able to fetch the next page" do
182
+ Twitter::Search.expects(:get).with('http://search.twitter.com/search.json', :query => 'page=2&max_id=1446791544&q=%40jnunemaker', :format => :json, :headers => {'User-Agent' => 'Ruby Twitter Gem'}).returns({'foo' => 'bar'})
183
+ @search.fetch_next_page
184
+ end
185
+ end
186
+
187
+ context "iterating over results" do
188
+ setup do
189
+ stub_get('http://search.twitter.com:80/search.json?q=from%3Ajnunemaker', 'search_from_jnunemaker.json')
190
+ @search.from('jnunemaker')
191
+ end
192
+
193
+ should "work" do
194
+ @search.each { |result| result.should_not be(nil) }
195
+ end
196
+
197
+ should "work multiple times in a row" do
198
+ @search.each { |result| result.should_not be(nil) }
199
+ @search.each { |result| result.should_not be(nil) }
200
+ end
201
+ end
202
+
203
+ should "should be able to iterate over results" do
204
+ @search.respond_to?(:each).should be(true)
205
+ end
206
+ end
207
+
208
+ end