#-- vim:sw=2:et #++ # # :title: RSS feed plugin for rbot # # Author:: Stanislav Karchebny # Author:: Ian Monroe # Author:: Mark Kretschmann # Author:: Giuseppe Bilotta # # Copyright:: (C) 2004 Stanislav Karchebny # Copyright:: (C) 2005 Ian Monroe, Mark Kretschmann # Copyright:: (C) 2006-2007 Giuseppe Bilotta # # License:: MIT license require 'rss' # Try to load rss/content/2.0 so we can access the data in # tags. begin require 'rss/content/2.0' rescue LoadError end module ::RSS # Add support for Slashdot namespace in RDF. The code is just an adaptation # of the DublinCore code. unless defined?(SLASH_PREFIX) SLASH_PREFIX = 'slash' SLASH_URI = "http://purl.org/rss/1.0/modules/slash/" RDF.install_ns(SLASH_PREFIX, SLASH_URI) module BaseSlashModel def append_features(klass) super return if klass.instance_of?(Module) SlashModel::ELEMENT_NAME_INFOS.each do |name, plural_name| plural = plural_name || "#{name}s" full_name = "#{SLASH_PREFIX}_#{name}" full_plural_name = "#{SLASH_PREFIX}_#{plural}" klass_name = "Slash#{Utils.to_class_name(name)}" # This will fail with older version of the Ruby RSS module begin klass.install_have_children_element(name, SLASH_URI, "*", full_name, full_plural_name) klass.install_must_call_validator(SLASH_PREFIX, SLASH_URI) rescue ArgumentError klass.module_eval("install_have_children_element(#{full_name.dump}, #{full_plural_name.dump})") end klass.module_eval(<<-EOC, *get_file_and_line_from_caller(0)) remove_method :#{full_name} if method_defined? :#{full_name} remove_method :#{full_name}= if method_defined? :#{full_name}= remove_method :set_#{full_name} if method_defined? :set_#{full_name} def #{full_name} @#{full_name}.first and @#{full_name}.first.value end def #{full_name}=(new_value) @#{full_name}[0] = Utils.new_with_value_if_need(#{klass_name}, new_value) end alias set_#{full_name} #{full_name}= EOC end end end module SlashModel extend BaseModel extend BaseSlashModel TEXT_ELEMENTS = { "department" => nil, "section" => nil, "comments" => nil, "hit_parade" => nil } ELEMENT_NAME_INFOS = SlashModel::TEXT_ELEMENTS.to_a ELEMENTS = TEXT_ELEMENTS.keys ELEMENTS.each do |name, plural_name| module_eval(<<-EOC, *get_file_and_line_from_caller(0)) class Slash#{Utils.to_class_name(name)} < Element include RSS10 content_setup class << self def required_prefix SLASH_PREFIX end def required_uri SLASH_URI end end @tag_name = #{name.dump} alias_method(:value, :content) alias_method(:value=, :content=) def initialize(*args) begin if Utils.element_initialize_arguments?(args) super else super() self.content = args[0] end # Older Ruby RSS module rescue NoMethodError super() self.content = args[0] end end def full_name tag_name_with_prefix(SLASH_PREFIX) end def maker_target(target) target.new_#{name} end def setup_maker_attributes(#{name}) #{name}.content = content end end EOC end end class RDF class Item; include SlashModel; end end SlashModel::ELEMENTS.each do |name| class_name = Utils.to_class_name(name) BaseListener.install_class_name(SLASH_URI, name, "Slash#{class_name}") end SlashModel::ELEMENTS.collect! {|name| "#{SLASH_PREFIX}_#{name}"} end if self.const_defined? :Atom # There are improper Atom feeds around that use the non-standard # 'modified' element instead of the correct 'updated' one. Let's # support it too. module Atom class Feed class Modified < RSS::Element include CommonModel include DateConstruct end __send__("install_have_child_element", "modified", URI, nil, "modified", :content) class Entry Modified = Feed::Modified __send__("install_have_child_element", "modified", URI, nil, "modified", :content) end end end end class Element class << self def def_bang(name, chain) class_eval %< def #{name}! blank2nil { #{chain.join(' rescue ')} rescue nil } end >, *get_file_and_line_from_caller(0) end end # Atom categories are squashed to their label only { :link => %w{link.href link}, :guid => %w{guid.content guid}, :content => %w{content.content content}, :description => %w{description.content description}, :title => %w{title.content title}, :category => %w{category.content category.label category}, :dc_subject => %w{dc_subject}, :author => %w{author.name.content author.name author}, :dc_creator => %w{dc_creator} }.each { |name, chain| def_bang name, chain } def categories! return nil unless self.respond_to? :categories cats = categories.map do |c| blank2nil { c.content rescue c.label rescue c rescue nil } end.compact cats.empty? ? nil : cats end protected def blank2nil(&block) x = yield (x && !x.empty?) ? x : nil end end end class ::RssBlob attr_accessor :url, :handle, :type, :refresh_rate, :xml, :title, :items, :mutex, :watchers, :last_fetched, :http_cache, :last_success def initialize(url,handle=nil,type=nil,watchers=[], xml=nil, lf = nil) @url = url if handle @handle = handle else @handle = url end @type = type @watchers=[] @refresh_rate = nil @http_cache = false @xml = xml @title = nil @items = nil @mutex = Mutex.new @last_fetched = lf @last_success = nil sanitize_watchers(watchers) end def dup @mutex.synchronize do self.class.new(@url, @handle, @type ? @type.dup : nil, @watchers.dup, @xml ? @xml.dup : nil, @last_fetched) end end # Downcase all watchers, possibly turning them into Strings if they weren't def sanitize_watchers(list=@watchers) ls = list.dup @watchers.clear ls.each { |w| add_watch(w) } end def watched? !@watchers.empty? end def watched_by?(who) @watchers.include?(who.downcase) end def add_watch(who) if watched_by?(who) return nil end @mutex.synchronize do @watchers << who.downcase end return who end def rm_watch(who) @mutex.synchronize do @watchers.delete(who.downcase) end end def to_a [@handle,@url,@type,@refresh_rate,@watchers] end def to_s(watchers=false) if watchers a = self.to_a.flatten else a = self.to_a[0,3] end a.compact.join(" | ") end end class RSSFeedsPlugin < Plugin Config.register Config::IntegerValue.new('rss.head_max', :default => 100, :validate => Proc.new{|v| v > 0 && v < 200}, :desc => "How many characters to use of a RSS item header") Config.register Config::IntegerValue.new('rss.text_max', :default => 200, :validate => Proc.new{|v| v > 0 && v < 400}, :desc => "How many characters to use of a RSS item text") Config.register Config::IntegerValue.new('rss.thread_sleep', :default => 300, :validate => Proc.new{|v| v > 30}, :desc => "How many seconds to sleep before checking RSS feeds again") Config.register Config::IntegerValue.new('rss.announce_timeout', :default => 0, :desc => "Don't announce watched feed if these many seconds elapsed since the last successful update") Config.register Config::IntegerValue.new('rss.announce_max', :default => 3, :desc => "Maximum number of new items to announce when a watched feed is updated") Config.register Config::BooleanValue.new('rss.show_updated', :default => true, :desc => "Whether feed items for which the description was changed should be shown as new") Config.register Config::BooleanValue.new('rss.show_links', :default => true, :desc => "Whether to display links from the text of a feed item.") Config.register Config::EnumValue.new('rss.announce_method', :values => ['say', 'notice'], :default => 'say', :desc => "Whether to display links from the text of a feed item.") # Make an 'unique' ID for a given item, based on appropriate bot options # Currently only supported is bot.config['rss.show_updated']: when false, # only the guid/link is accounted for. def make_uid(item) uid = [item.guid! || item.link!] if @bot.config['rss.show_updated'] uid.push(item.content! || item.description!) uid.unshift item.title! end # debug "taking hash of #{uid.inspect}" uid.hash end # We used to save the Mutex with the RssBlob, which was idiotic. And # since Mutexes dumped in one version might not be restorable in another, # we need a few tricks to be able to restore data from other versions of Ruby # # When migrating 1.8.6 => 1.8.5, all we need to do is define an empty # #marshal_load() method for Mutex. For 1.8.5 => 1.8.6 we need something # dirtier, as seen later on in the initialization code. unless Mutex.new.respond_to?(:marshal_load) class ::Mutex def marshal_load(str) return end end end # Auxiliary method used to collect two lines for rss output filters, # running substitutions against DataStream _s_ optionally joined # with hash _h_. # # For substitutions, *_wrap keys can be used to alter the content of # other nonempty keys. If the value of *_wrap is a String, it will be # put before and after the corresponding key; if it's an Array, the first # and second elements will be used for wrapping; if it's nil, no wrapping # will be done (useful to override a default wrapping). # # For example: # :handle_wrap => '::':: # will wrap s[:handle] by prefixing and postfixing it with '::' # :date_wrap => [nil, ' :: ']:: # will put ' :: ' after s[:date] def make_stream(line1, line2, s, h={}) ss = s.merge(h) subs = {} wraps = {} ss.each do |k, v| kk = k.to_s.chomp!('_wrap') if kk nk = kk.intern case v when String wraps[nk] = ss[nk].wrap_nonempty(v, v) when Array wraps[nk] = ss[nk].wrap_nonempty(*v) when nil # do nothing else warning "ignoring #{v.inspect} wrapping of unknown class" end unless ss[nk].nil? else subs[k] = v end end subs.merge! wraps DataStream.new([line1, line2].compact.join("\n") % subs, ss) end # Auxiliary method used to define rss output filters def rss_type(key, &block) @bot.register_filter(key, @outkey, &block) end # Define default output filters (rss types), and load custom ones. # Custom filters are looked for in the plugin's default filter locations # and in rss/types.rb under botclass. # Preferably, the rss_type method should be used in these files, e.g.: # rss_type :my_type do |s| # line1 = "%{handle} and some %{author} info" # make_stream(line1, nil, s) # end # to define the new type 'my_type'. The keys available in the DataStream # are: # item:: # the actual rss item # handle:: # the item handle # date:: # the item date # title:: # the item title # desc, link, category, author:: # the item description, link, category, author # at:: # the string ' @ ' if the item has both an title and a link # handle_wrap, date_wrap, title_wrap, ...:: # these keys can be defined to wrap the corresponding elements if they # are nonempty. By default handle is wrapped with '::', date has a ' ::' # appended and title is enbolden # def define_filters @outkey ||= :"rss.out" # Define an HTML info filter @bot.register_filter(:rss, :htmlinfo) { |s| htmlinfo_filter(s) } # This is the output format used by the input filter rss_type :htmlinfo do |s| line1 = "%{title}%{at}%{link}" make_stream(line1, nil, s) end # the default filter rss_type :default do |s| line1 = "%{handle}%{date}%{title}%{at}%{link}" line1 << " (by %{author})" if s[:author] make_stream(line1, nil, s) end @user_types ||= datafile 'types.rb' load_filters load_filters :path => @user_types end FEED_NS = %r{xmlns.*http://(purl\.org/rss|www.w3c.org/1999/02/22-rdf)} def htmlinfo_filter(s) return nil unless s[:headers] and s[:headers]['x-rbot-location'] return nil unless s[:headers]['content-type'].first.match(/xml|rss|atom|rdf/i) or (s[:text].include?(" blob.title, :content => output.join(" | ")} end # Display the known rss types def rss_types(m, params) ar = @bot.filter_names(@outkey) ar.delete(:default) m.reply ar.map { |k| k.to_s }.sort!.join(", ") end attr_reader :feeds def initialize super define_filters if @registry.has_key?(:feeds) # When migrating from Ruby 1.8.5 to 1.8.6, dumped Mutexes may render the # data unrestorable. If this happens, we patch the data, thus allowing # the restore to work. # # This is actually pretty safe for a number of reasons: # * the code is only called if standard marshalling fails # * the string we look for is quite unlikely to appear randomly # * if the string appears somewhere and the patched string isn't recoverable # either, we'll get another (unrecoverable) error, which makes the rss # plugin unsable, just like it was if no recovery was attempted # * if the string appears somewhere and the patched string is recoverable, # we may get a b0rked feed, which is eventually overwritten by a clean # one, so the worst thing that can happen is that a feed update spams # the watchers once @registry.recovery = Proc.new { |val| patched = val.sub(":\v@mutexo:\nMutex", ":\v@mutexo:\vObject") ret = Marshal.restore(patched) ret.each_value { |blob| blob.mutex = nil blob } } @feeds = @registry[:feeds] raise LoadError, "corrupted feed database" unless @feeds @registry.recovery = nil @feeds.keys.grep(/[A-Z]/) { |k| @feeds[k.downcase] = @feeds[k] @feeds.delete(k) } @feeds.each { |k, f| f.mutex = Mutex.new f.sanitize_watchers parseRss(f) if f.xml } else @feeds = Hash.new end @watch = Hash.new rewatch_rss end def name "rss" end def watchlist @feeds.select { |h, f| f.watched? } end def cleanup stop_watches super end def save unparsed = Hash.new() @feeds.each { |k, f| unparsed[k] = f.dup # we don't want to save the mutex unparsed[k].mutex = nil } @registry[:feeds] = unparsed end def stop_watch(handle) if @watch.has_key?(handle) begin debug "Stopping watch #{handle}" @bot.timer.remove(@watch[handle]) @watch.delete(handle) rescue Exception => e report_problem("Failed to stop watch for #{handle}", e, nil) end end end def stop_watches @watch.each_key { |k| stop_watch(k) } end def help(plugin,topic="") case topic when "show" "rss show #{Bold}handle#{Bold} [#{Bold}limit#{Bold}] : show #{Bold}limit#{Bold} (default: 5, max: 15) entries from rss #{Bold}handle#{Bold}; #{Bold}limit#{Bold} can also be in the form a..b, to display a specific range of items" when "list" "rss list [#{Bold}handle#{Bold}] : list all rss feeds (matching #{Bold}handle#{Bold})" when "watched" "rss watched [#{Bold}handle#{Bold}] [in #{Bold}chan#{Bold}]: list all watched rss feeds (matching #{Bold}handle#{Bold}) (in channel #{Bold}chan#{Bold})" when "who", "watches", "who watches" "rss who watches [#{Bold}handle#{Bold}]]: list all watchers for rss feeds (matching #{Bold}handle#{Bold})" when "add" "rss add #{Bold}handle#{Bold} #{Bold}url#{Bold} [#{Bold}type#{Bold}] : add a new rss called #{Bold}handle#{Bold} from url #{Bold}url#{Bold} (of type #{Bold}type#{Bold})" when "change" "rss change #{Bold}what#{Bold} of #{Bold}handle#{Bold} to #{Bold}new#{Bold} : change the #{Underline}handle#{Underline}, #{Underline}url#{Underline}, #{Underline}type#{Underline} or #{Underline}refresh#{Underline} rate of rss called #{Bold}handle#{Bold} to value #{Bold}new#{Bold}" when /^(del(ete)?|rm)$/ "rss del(ete)|rm #{Bold}handle#{Bold} : delete rss feed #{Bold}handle#{Bold}" when "replace" "rss replace #{Bold}handle#{Bold} #{Bold}url#{Bold} [#{Bold}type#{Bold}] : try to replace the url of rss called #{Bold}handle#{Bold} with #{Bold}url#{Bold} (of type #{Bold}type#{Bold}); only works if nobody else is watching it" when "forcereplace" "rss forcereplace #{Bold}handle#{Bold} #{Bold}url#{Bold} [#{Bold}type#{Bold}] : replace the url of rss called #{Bold}handle#{Bold} with #{Bold}url#{Bold} (of type #{Bold}type#{Bold})" when "watch" "rss watch #{Bold}handle#{Bold} [#{Bold}url#{Bold} [#{Bold}type#{Bold}]] [in #{Bold}chan#{Bold}]: watch rss #{Bold}handle#{Bold} for changes (in channel #{Bold}chan#{Bold}); when the other parameters are present, the feed will be created if it doesn't exist yet" when /(un|rm)watch/ "rss unwatch|rmwatch #{Bold}handle#{Bold} [in #{Bold}chan#{Bold}]: stop watching rss #{Bold}handle#{Bold} (in channel #{Bold}chan#{Bold}) for changes" when /who(?: watche?s?)?/ "rss who watches #{Bold}handle#{Bold}: lists watches for rss #{Bold}handle#{Bold}" when "rewatch" "rss rewatch : restart threads that watch for changes in watched rss" when "types" "rss types : show the rss types for which an output format exist (all other types will use the default one)" else "manage RSS feeds: rss types|show|list|watched|add|change|del(ete)|rm|(force)replace|watch|unwatch|rmwatch|rewatch|who watches" end end def report_problem(report, e=nil, m=nil) if m && m.respond_to?(:reply) m.reply report else warning report end if e debug e.inspect debug e.backtrace.join("\n") if e.respond_to?(:backtrace) end end def show_rss(m, params) handle = params[:handle] lims = params[:limit].to_s.match(/(\d+)(?:..(\d+))?/) debug lims.to_a.inspect if lims[2] ll = [[lims[1].to_i-1,lims[2].to_i-1].min, 0].max ul = [[lims[1].to_i-1,lims[2].to_i-1].max, 14].min rev = lims[1].to_i > lims[2].to_i else ll = 0 ul = [[lims[1].to_i-1, 0].max, 14].min rev = false end feed = @feeds.fetch(handle.downcase, nil) unless feed m.reply "I don't know any feeds named #{handle}" return end m.reply "lemme fetch it..." title = items = nil we_were_watching = false if @watch.key?(feed.handle) # If a feed is being watched, we run the watcher thread # so that all watchers can be informed of changes to # the feed. Before we do that, though, we remove the # show requester from the watchlist, if present, lest # he gets the update twice. if feed.watched_by?(m.replyto) we_were_watching = true feed.rm_watch(m.replyto) end @bot.timer.reschedule(@watch[feed.handle], 0) if we_were_watching feed.add_watch(m.replyto) end else fetched = fetchRss(feed, m, false) end return unless fetched or feed.xml if fetched or not feed.items parsed = parseRss(feed, m) end return unless feed.items m.reply "using old data" unless fetched and parsed and parsed > 0 title = feed.title items = feed.items # We sort the feeds in freshness order (newer ones first) items = freshness_sort(items) disp = items[ll..ul] disp.reverse! if rev m.reply "Channel : #{title}" disp.each do |item| printFormattedRss(feed, item, { :places => [m.replyto], :handle => nil, :date => true, :announce_method => :say }) end end def itemDate(item,ex=nil) return item.pubDate if item.respond_to?(:pubDate) and item.pubDate return item.date if item.respond_to?(:date) and item.date return ex end def freshness_sort(items) notime = Time.at(0) items.sort { |a, b| itemDate(b, notime) <=> itemDate(a, notime) } end def list_rss(m, params) wanted = params[:handle] listed = @feeds.keys if wanted wanted_rx = Regexp.new(wanted, true) listed.reject! { |handle| !handle.match(wanted_rx) } end listed.sort! debug listed if @bot.config['send.max_lines'] > 0 and listed.size > @bot.config['send.max_lines'] reply = listed.inject([]) do |ar, handle| feed = @feeds[handle] string = handle.dup (string << " (#{feed.type})") if feed.type (string << " (watched)") if feed.watched_by?(m.replyto) ar << string end.join(', ') elsif listed.size > 0 reply = listed.inject([]) do |ar, handle| feed = @feeds[handle] string = "#{feed.handle}: #{feed.url} (in format: #{feed.type ? feed.type : 'default'})" (string << " refreshing every #{Utils.secs_to_string(feed.refresh_rate)}") if feed.refresh_rate (string << " (watched)") if feed.watched_by?(m.replyto) ar << string end.join("\n") else reply = "no feeds found" reply << " matching #{wanted}" if wanted end m.reply reply, :max_lines => 0 end def watched_rss(m, params) wanted = params[:handle] chan = params[:chan] || m.replyto reply = String.new watchlist.each { |handle, feed| next if wanted and !handle.match(/#{wanted}/i) next unless feed.watched_by?(chan) reply << "#{feed.handle}: #{feed.url} (in format: #{feed.type ? feed.type : 'default'})" (reply << " refreshing every #{Utils.secs_to_string(feed.refresh_rate)}") if feed.refresh_rate reply << "\n" } if reply.empty? reply = "no watched feeds" reply << " matching #{wanted}" if wanted end m.reply reply end def who_watches(m, params) wanted = params[:handle] reply = String.new watchlist.each { |handle, feed| next if wanted and !handle.match(/#{wanted}/i) reply << "#{feed.handle}: #{feed.url} (in format: #{feed.type ? feed.type : 'default'})" (reply << " refreshing every #{Utils.secs_to_string(feed.refresh_rate)}") if feed.refresh_rate reply << ": watched by #{feed.watchers.join(', ')}" reply << "\n" } if reply.empty? reply = "no watched feeds" reply << " matching #{wanted}" if wanted end m.reply reply end def add_rss(m, params, force=false) handle = params[:handle] url = params[:url] unless url.match(/https?/) m.reply "I only deal with feeds from HTTP sources, so I can't use #{url} (maybe you forgot the handle?)" return end type = params[:type] if @feeds.fetch(handle.downcase, nil) && !force m.reply "There is already a feed named #{handle} (URL: #{@feeds[handle.downcase].url})" return end unless url m.reply "You must specify both a handle and an url to add an RSS feed" return end @feeds[handle.downcase] = RssBlob.new(url,handle,type) reply = "Added RSS #{url} named #{handle}" if type reply << " (format: #{type})" end m.reply reply return handle end def change_rss(m, params) handle = params[:handle].downcase feed = @feeds.fetch(handle, nil) unless feed m.reply "No such feed with handle #{handle}" return end case params[:what].intern when :handle # preserve rename case, but beware of key realnew = params[:new] new = realnew.downcase if feed.handle.downcase == new if feed.handle == realnew m.reply _("You want me to rename %{handle} to itself?") % { :handle => feed.handle } return false else feed.mutex.synchronize do feed.handle = realnew end end elsif @feeds.key?(new) and @feeds[new] m.reply "There already is a feed with handle #{new}" return else feed.mutex.synchronize do @feeds[new] = feed @feeds.delete(handle) feed.handle = realnew end handle = new end when :url new = params[:new] feed.mutex.synchronize do feed.url = new end when :format, :type new = params[:new] new = nil if new == 'default' feed.mutex.synchronize do feed.type = new end when :refresh new = params[:new].to_i new = nil if new == 0 feed.mutex.synchronize do feed.refresh_rate = new end else m.reply "Don't know how to change #{params[:what]} for feeds" return end m.reply "Feed changed:" list_rss(m, {:handle => handle}) end def del_rss(m, params, pass=false) feed = unwatch_rss(m, params, true) return unless feed if feed.watched? m.reply "someone else is watching #{feed.handle}, I won't remove it from my list" return end @feeds.delete(feed.handle.downcase) m.okay unless pass return end def replace_rss(m, params) handle = params[:handle] if @feeds.key?(handle.downcase) del_rss(m, {:handle => handle}, true) end if @feeds.key?(handle.downcase) m.reply "can't replace #{feed.handle}" else add_rss(m, params, true) end end def forcereplace_rss(m, params) add_rss(m, params, true) end def watch_rss(m, params) handle = params[:handle] chan = params[:chan] || m.replyto url = params[:url] type = params[:type] if url add_rss(m, params) end feed = @feeds.fetch(handle.downcase, nil) if feed if feed.add_watch(chan) watchRss(feed, m) m.okay else m.reply "Already watching #{feed.handle} in #{chan}" end else m.reply "Couldn't watch feed #{handle} (no such feed found)" end end def unwatch_rss(m, params, pass=false) handle = params[:handle].downcase chan = params[:chan] || m.replyto unless @feeds.has_key?(handle) m.reply("dunno that feed") return end feed = @feeds[handle] if feed.rm_watch(chan) m.reply "#{chan} has been removed from the watchlist for #{feed.handle}" else m.reply("#{chan} wasn't watching #{feed.handle}") unless pass end if !feed.watched? stop_watch(handle) end return feed end def rewatch_rss(m=nil, params=nil) if params and handle = params[:handle] feed = @feeds.fetch(handle.downcase, nil) if feed feed.http_cache = false @bot.timer.reschedule(@watch[feed.handle], (params[:delay] || 0).to_f) m.okay if m else m.reply _("no such feed %{handle}") % { :handle => handle } if m end else stop_watches # Read watches from list. watchlist.each{ |hndl, fd| watchRss(fd, m) } m.okay if m end end private def watchRss(feed, m=nil) if @watch.has_key?(feed.handle) # report_problem("watcher thread for #{feed.handle} is already running", nil, m) return end status = Hash.new status[:failures] = 0 tmout = 0 if feed.last_fetched tmout = feed.last_fetched + calculate_timeout(feed) - Time.now tmout = 0 if tmout < 0 end debug "scheduling a watcher for #{feed} in #{tmout} seconds" @watch[feed.handle] = @bot.timer.add(tmout) { debug "watcher for #{feed} wakes up" failures = status[:failures] begin debug "fetching #{feed}" first_run = !feed.last_success if (!first_run && @bot.config['rss.announce_timeout'] > 0 && (Time.now - feed.last_success > @bot.config['rss.announce_timeout'])) debug "#{feed} wasn't polled for too long, supressing output" first_run = true end oldxml = feed.xml ? feed.xml.dup : nil unless fetchRss(feed, nil, feed.http_cache) failures += 1 else feed.http_cache = true if first_run debug "first run for #{feed}, getting items" parseRss(feed) elsif oldxml and oldxml == feed.xml debug "xml for #{feed} didn't change" failures -= 1 if failures > 0 else # This one is used for debugging otxt = [] if feed.items.nil? oids = [] else # These are used for checking new items vs old ones oids = Set.new feed.items.map { |item| uid = make_uid item otxt << item.to_s debug [uid, item].inspect debug [uid, otxt.last].inspect uid } end nitems = parseRss(feed) if nitems.nil? failures += 1 elsif nitems == 0 debug "no items in feed #{feed}" else debug "Checking if new items are available for #{feed}" failures -= 1 if failures > 0 # debug "Old:" # debug oldxml # debug "New:" # debug feed.xml dispItems = feed.items.reject { |item| uid = make_uid item txt = item.to_s if oids.include?(uid) debug "rejecting old #{uid} #{item.inspect}" debug [uid, txt].inspect true else debug "accepting new #{uid} #{item.inspect}" debug [uid, txt].inspect warning "same text! #{txt}" if otxt.include?(txt) false end } if dispItems.length > 0 max = @bot.config['rss.announce_max'] debug "Found #{dispItems.length} new items in #{feed}" if max > 0 and dispItems.length > max debug "showing only the latest #{dispItems.length}" feed.watchers.each do |loc| @bot.say loc, (_("feed %{feed} had %{num} updates, showing the latest %{max}") % { :feed => feed.handle, :num => dispItems.length, :max => max }) end dispItems.slice!(max..-1) end # When displaying watched feeds, publish them from older to newer dispItems.reverse.each { |item| printFormattedRss(feed, item) } else debug "No new items found in #{feed}" end end end end rescue Exception => e error "Error watching #{feed}: #{e.inspect}" debug e.backtrace.join("\n") failures += 1 end status[:failures] = failures seconds = calculate_timeout(feed, failures) debug "watcher for #{feed} going to sleep #{seconds} seconds.." begin @bot.timer.reschedule(@watch[feed.handle], seconds) rescue warning "watcher for #{feed} failed to reschedule: #{$!.inspect}" end } debug "watcher for #{feed} added" end def calculate_timeout(feed, failures = 0) seconds = @bot.config['rss.thread_sleep'] feed.mutex.synchronize do seconds = feed.refresh_rate if feed.refresh_rate end seconds *= failures + 1 seconds += seconds * (rand(100)-50)/100 return seconds end def make_date(obj) if obj.kind_of? Time obj.strftime("%Y/%m/%d %H:%M") else obj.to_s end end def printFormattedRss(feed, item, options={}) # debug item opts = { :places => feed.watchers, :handle => feed.handle, :date => false, :announce_method => @bot.config['rss.announce_method'] }.merge options places = opts[:places] announce_method = opts[:announce_method] handle = opts[:handle].to_s date = \ if opts[:date] if item.respond_to?(:updated) and item.updated make_date(item.updated.content) elsif item.respond_to?(:modified) and item.modified make_date(item.modified.content) elsif item.respond_to?(:source) and item.source.respond_to?(:updated) make_date(item.source.updated.content) elsif item.respond_to?(:pubDate) make_date(item.pubDate) elsif item.respond_to?(:date) make_date(item.date) else "(no date)" end else String.new end tit_opt = {} # Twitters don't need a cap on the title length since they have a hard # limit to 160 characters, and most of them are under 140 characters tit_opt[:limit] = @bot.config['rss.head_max'] unless feed.type == 'twitter' if item.title base_title = item.title.to_s.dup # git changesets are SHA1 hashes (40 hex digits), way too long, get rid of them, as they are # visible in the URL anyway # TODO make this optional? base_title.sub!(/^Changeset \[([\da-f]{40})\]:/) { |c| "(git commit)"} if feed.type == 'trac' title = base_title.ircify_html(tit_opt) end desc_opt = {} desc_opt[:limit] = @bot.config['rss.text_max'] desc_opt[:a_href] = :link_out if @bot.config['rss.show_links'] # We prefer content_encoded here as it tends to provide more html formatting # for use with ircify_html. if item.respond_to?(:content_encoded) && item.content_encoded desc = item.content_encoded.ircify_html(desc_opt) elsif item.respond_to?(:description) && item.description desc = item.description.ircify_html(desc_opt) elsif item.respond_to?(:content) && item.content if item.content.type == "html" desc = item.content.content.ircify_html(desc_opt) else desc = item.content.content if desc.size > desc_opt[:limit] desc = desc.slice(0, desc_opt[:limit]) + "#{Reverse}...#{Reverse}" end end else desc = "(?)" end link = item.link! link.strip! if link categories = item.categories! category = item.category! || item.dc_subject! category.strip! if category author = item.dc_creator! || item.author! author.strip! if author line1 = nil line2 = nil at = ((item.title && item.link) ? ' @ ' : '') key = @bot.global_filter_name(feed.type, @outkey) key = @bot.global_filter_name(:default, @outkey) unless @bot.has_filter?(key) stream_hash = { :item => item, :handle => handle, :handle_wrap => ['::', ':: '], :date => date, :date_wrap => [nil, ' :: '], :title => title, :title_wrap => Bold, :desc => desc, :link => link, :categories => categories, :category => category, :author => author, :at => at } output = @bot.filter(key, stream_hash) return output if places.empty? places.each { |loc| output.to_s.each_line { |line| @bot.__send__(announce_method, loc, line, :overlong => :truncate) } } end def fetchRss(feed, m=nil, cache=true) feed.last_fetched = Time.now begin # Use 60 sec timeout, cause the default is too low xml = @bot.httputil.get(feed.url, :read_timeout => 60, :open_timeout => 60, :cache => cache) rescue URI::InvalidURIError, URI::BadURIError => e report_problem("invalid rss feed #{feed.url}", e, m) return nil rescue => e report_problem("error getting #{feed.url}", e, m) return nil end debug "fetched #{feed}" unless xml report_problem("reading feed #{feed} failed", nil, m) return nil end # Ok, 0.9 feeds are not supported, maybe because # Netscape happily removed the DTD. So what we do is just to # reassign the 0.9 RDFs to 1.0, and hope it goes right. xml.gsub!("xmlns=\"http://my.netscape.com/rdf/simple/0.9/\"", "xmlns=\"http://purl.org/rss/1.0/\"") # make sure the parser doesn't double-convert in case the feed is not UTF-8 xml.sub!(/<\?xml (.*?)\?>/) do |match| if /\bencoding=(['"])(.*?)\1/.match(match) match.sub!(/\bencoding=(['"])(?:.*?)\1/,'encoding="UTF-8"') end match end feed.mutex.synchronize do feed.xml = xml feed.last_success = Time.now end return true end def parseRss(feed, m=nil) return nil unless feed.xml feed.mutex.synchronize do xml = feed.xml rss = nil errors = [] RSS::AVAILABLE_PARSERS.each do |parser| begin ## do validate parse rss = RSS::Parser.parse(xml, true, true, parser) debug "parsed and validated #{feed} with #{parser}" break rescue RSS::InvalidRSSError begin ## do non validate parse for invalid RSS 1.0 rss = RSS::Parser.parse(xml, false, true, parser) debug "parsed but not validated #{feed} with #{parser}" break rescue RSS::Error => e errors << [parser, e, "parsing rss stream failed, whoops =("] end rescue RSS::Error => e errors << [parser, e, "parsing rss stream failed, oioi"] rescue => e errors << [parser, e, "processing error occured, sorry =("] end end unless errors.empty? debug errors self.send(:report_problem, errors.last[2], errors.last[1], m) return nil unless rss end items = [] if rss.nil? if xml.match(/xmlns\s*=\s*(['"])http:\/\/www.w3.org\/2005\/Atom\1/) and not defined?(RSS::Atom) report_problem("#{feed.handle} @ #{feed.url} looks like an Atom feed, but your Ruby/RSS library doesn't seem to support it. Consider getting the latest version from http://raa.ruby-lang.org/project/rss/", nil, m) else report_problem("#{feed.handle} @ #{feed.url} doesn't seem to contain an RSS or Atom feed I can read", nil, m) end return nil else begin rss.output_encoding = 'UTF-8' rescue RSS::UnknownConvertMethod => e report_problem("bah! something went wrong =(", e, m) return nil end if rss.respond_to? :channel rss.channel.title ||= "(?)" title = rss.channel.title else title = rss.title.content end rss.items.each do |item| item.title ||= "(?)" items << item end end if items.empty? report_problem("no items found in the feed, maybe try weed?", e, m) else feed.title = title.strip feed.items = items end return items.length end end end plugin = RSSFeedsPlugin.new plugin.default_auth( 'edit', false ) plugin.default_auth( 'edit:add', true) plugin.map 'rss show :handle :limit', :action => 'show_rss', :requirements => {:limit => /^\d+(?:\.\.\d+)?$/}, :defaults => {:limit => 5} plugin.map 'rss list :handle', :action => 'list_rss', :defaults => {:handle => nil} plugin.map 'rss watched :handle [in :chan]', :action => 'watched_rss', :defaults => {:handle => nil} plugin.map 'rss who watches :handle', :action => 'who_watches', :defaults => {:handle => nil} plugin.map 'rss add :handle :url :type', :action => 'add_rss', :auth_path => 'edit', :defaults => {:type => nil} plugin.map 'rss change :what of :handle to :new', :action => 'change_rss', :auth_path => 'edit', :requirements => { :what => /handle|url|format|type|refresh/ } plugin.map 'rss change :what for :handle to :new', :action => 'change_rss', :auth_path => 'edit', :requirements => { :what => /handle|url|format|type|refesh/ } plugin.map 'rss del :handle', :auth_path => 'edit:rm!', :action => 'del_rss' plugin.map 'rss delete :handle', :auth_path => 'edit:rm!', :action => 'del_rss' plugin.map 'rss rm :handle', :auth_path => 'edit:rm!', :action => 'del_rss' plugin.map 'rss replace :handle :url :type', :auth_path => 'edit', :action => 'replace_rss', :defaults => {:type => nil} plugin.map 'rss forcereplace :handle :url :type', :auth_path => 'edit', :action => 'forcereplace_rss', :defaults => {:type => nil} plugin.map 'rss watch :handle [in :chan]', :action => 'watch_rss', :defaults => {:url => nil, :type => nil} plugin.map 'rss watch :handle :url :type [in :chan]', :action => 'watch_rss', :defaults => {:url => nil, :type => nil} plugin.map 'rss unwatch :handle [in :chan]', :action => 'unwatch_rss' plugin.map 'rss rmwatch :handle [in :chan]', :action => 'unwatch_rss' plugin.map 'rss rewatch [:handle] [:delay]', :action => 'rewatch_rss' plugin.map 'rss types', :action => 'rss_types'