+ @bot.path and datafile methods
[rbot] / data / rbot / plugins / twitter.rb
1 #-- vim:sw=2:et
2 #++
3 #
4 # :title: Twitter Status Update for rbot
5 #
6 # Author:: Carter Parks (carterparks) <carter@carterparks.com>
7 # Author:: Giuseppe "Oblomov" Bilotta <giuseppe.bilotta@gmail.com>
8 #
9 # Copyright:: (C) 2007 Carter Parks
10 # Copyright:: (C) 2007 Giuseppe Bilotta
11 #
12 # Users can setup their twitter username and password and then begin updating
13 # twitter whenever
14
15 require 'rexml/rexml'
16 require 'cgi'
17
18 class TwitterPlugin < Plugin
19   Config.register Config::IntegerValue.new('twitter.status_count',
20     :default => 1, :validate => Proc.new { |v| v > 0 && v <= 10},
21     :desc => "Maximum number of status updates shown by 'twitter status'")
22   Config.register Config::IntegerValue.new('twitter.friends_status_count',
23     :default => 3, :validate => Proc.new { |v| v > 0 && v <= 10},
24     :desc => "Maximum number of status updates shown by 'twitter friends status'")
25
26   def initialize
27     super
28
29     class << @registry
30       def store(val)
31         val
32       end
33       def restore(val)
34         val
35       end
36     end
37
38     @header = {
39       'X-Twitter-Client' => 'rbot twitter plugin'
40     }
41   end
42
43   # return a help string when the bot is asked for help on this plugin
44   def help(plugin, topic="")
45     return "twitter status [nick] => show nick's (or your) status, use 'twitter friends status [nick]' to also show the friends' timeline | twitter update [status] => updates your status on twitter | twitter identify [username] [password] => ties your nick to your twitter username and password | twitter actions [on|off] => enable/disable twitting of actions (/me does ...)"
46   end
47
48   # update the status on twitter
49   def get_status(m, params)
50
51     nick = params[:nick] || @registry[m.sourcenick + "_username"]
52
53     friends = params[:friends]
54
55     if not nick
56       m.reply "you should specify the username of the twitter touse, or identify using 'twitter identify [username] [password]'"
57       return false
58     end
59
60     user = URI.escape(nick)
61
62     count = @bot.config['twitter.status_count']
63     unless friends
64       uri = "http://twitter.com/statuses/user_timeline/#{user}.xml?count=#{count}"
65     else
66       count = @bot.config['twitter.friends_status_count']
67       auth = ""
68       if m.private?
69         auth << URI.escape(@registry[m.sourcenick + "_username"])
70         auth << ":"
71         auth << URI.escape(@registry[m.sourcenick + "_password"])
72         auth << "@"
73       end
74       uri = "http://#{auth}twitter.com/statuses/friends_timeline/#{user}.xml"
75     end
76
77     response = @bot.httputil.get(uri, :headers => @header, :cache => false)
78     debug response
79
80     texts = []
81
82     if response
83       begin
84         rex = REXML::Document.new(response)
85         rex.root.elements.each("status") { |st|
86           # month, day, hour, min, sec, year = st.elements['created_at'].text.match(/\w+ (\w+) (\d+) (\d+):(\d+):(\d+) \S+ (\d+)/)[1..6]
87           # debug [year, month, day, hour, min, sec].inspect
88           # time = Time.local(year.to_i, month, day.to_i, hour.to_i, min.to_i, sec.to_i)
89           time = Time.parse(st.elements['created_at'].text)
90           now = Time.now
91           # Sometimes, time can be in the future; invert the relation in this case
92           delta = ((time > now) ? time - now : now - time)
93           msg = st.elements['text'].to_s + " (#{Utils.secs_to_string(delta.to_i)} ago via #{st.elements['source'].to_s})"
94           author = ""
95           if friends
96             author = Utils.decode_html_entities(st.elements['user'].elements['name'].text) + ": " rescue ""
97           end
98           texts << author+Utils.decode_html_entities(msg).ircify_html
99         }
100         if friends
101           # friends always return the latest 20 updates, so we clip the count
102           texts[count..-1]=nil
103         end
104       rescue
105         error $!
106         if friends
107           m.reply "could not parse status for #{nick}'s friends"
108         else
109           m.reply "could not parse status for #{nick}"
110         end
111         return false
112       end
113       m.reply texts.reverse.join("\n")
114       return true
115     else
116       if friends
117         rep = "could not get status for #{nick}'s friends"
118         rep << ", try asking in private" unless m.private?
119       else
120         rep = "could not get status for #{nick}"
121       end
122       m.reply rep
123       return false
124     end
125   end
126
127   # update the status on twitter
128   def update_status(m, params)
129
130
131     unless @registry.has_key?(m.sourcenick + "_password") && @registry.has_key?(m.sourcenick + "_username")
132       m.reply "you must identify using 'twitter identify [username] [password]'"
133       return false
134     end
135
136     user = URI.escape(@registry[m.sourcenick + "_username"])
137     pass = URI.escape(@registry[m.sourcenick + "_password"])
138     uri = "http://#{user}:#{pass}@twitter.com/statuses/update.xml"
139
140     msg = params[:status].to_s
141
142     if msg.length > 160
143       m.reply "your status message update is too long, please keep it under 140 characters if possible, 160 characters maximum"
144       return
145     end
146
147     if msg.length > 140
148       m.reply "your status message is longer than 140 characters, which is not optimal, but I'm going to update anyway"
149     end
150
151     source = "source=rbot"
152     msg = "status=#{CGI.escape(msg)}"
153     body = [source,msg].join("&")
154
155     response = @bot.httputil.post(uri, body, :headers => @header)
156     debug response
157
158     reply_method = params[:notify] ? :notify : :reply
159     if response.class == Net::HTTPOK
160       m.__send__(reply_method, "status updated")
161     else
162       m.__send__(reply_method, "could not update status")
163     end
164   end
165
166   # ties a nickname to a twitter username and password
167   def identify(m, params)
168     @registry[m.sourcenick + "_username"] = params[:username].to_s
169     @registry[m.sourcenick + "_password"] = params[:password].to_s
170     m.reply "you're all setup!"
171   end
172
173   # update on ACTION if the user has enabled the option
174   def ctcp_listen(m)
175     return unless m.action?
176     return unless @registry[m.sourcenick + "_actions"]
177     update_status(m, :status => m.message, :notify => true)
178   end
179
180   # show or toggle action twitting
181   def actions(m, params)
182     case params[:toggle]
183     when 'on'
184       @registry[m.sourcenick + "_actions"] = true
185       m.okay
186     when 'off'
187       @registry.delete(m.sourcenick + "_actions")
188       m.okay
189     else
190       if @registry[m.sourcenick + "_actions"]
191         m.reply _("actions will be twitted")
192       else
193         m.reply _("actions will not be twitted")
194       end
195     end
196   end
197 end
198
199 # create an instance of our plugin class and register for the "length" command
200 plugin = TwitterPlugin.new
201 plugin.map 'twitter identify :username :password', :action => "identify", :public => false
202 plugin.map 'twitter update *status', :action => "update_status", :threaded => true
203 plugin.map 'twitter status [:nick]', :action => "get_status", :threaded => true
204 plugin.map 'twitter actions [:toggle]', :action => "actions", :requirements => { :toggle => /^on|off$/ }
205 plugin.map 'twitter :friends [status] [:nick]', :action => "get_status", :requirements => { :friends => /^friends?$/ }, :threaded => true
206