rbot 0.9.10 → 0.9.12
Sign up to get free protection for your applications and to get access to all the features.
- data/AUTHORS +22 -4
- data/COPYING +5 -18
- data/ChangeLog +396 -0
- data/README +23 -25
- data/REQUIREMENTS +58 -12
- data/Rakefile +215 -0
- data/bin/rbot +47 -25
- data/bin/rbot-remote +76 -0
- data/data/rbot/languages/french.lang +5 -5
- data/data/rbot/languages/german.lang +4 -4
- data/data/rbot/languages/italian.lang +53 -0
- data/data/rbot/languages/japanese.lang +43 -0
- data/data/rbot/languages/russian.lang +60 -60
- data/data/rbot/languages/traditional_chinese.lang +75 -0
- data/data/rbot/plugins/alias.rb +192 -0
- data/data/rbot/plugins/autoop.rb +74 -37
- data/data/rbot/plugins/autorejoin.rb +15 -5
- data/data/rbot/plugins/bans.rb +454 -150
- data/data/rbot/plugins/bash.rb +173 -31
- data/data/rbot/plugins/botsnack.rb +37 -0
- data/data/rbot/plugins/cal.rb +11 -2
- data/data/rbot/plugins/chanserv.rb +132 -0
- data/data/rbot/plugins/chucknorris.rb +2 -2
- data/data/rbot/plugins/chucknorris.yml.gz +0 -0
- data/data/rbot/plugins/debugger.rb +136 -0
- data/data/rbot/plugins/deepthoughts.rb +89 -95
- data/data/rbot/plugins/delicious.rb +109 -0
- data/data/rbot/plugins/dice.rb +77 -25
- data/data/rbot/plugins/dict.rb +243 -0
- data/data/rbot/plugins/dictclient.rb +211 -0
- data/data/rbot/plugins/digg.rb +2 -2
- data/data/rbot/plugins/excuse.rb +4 -4
- data/data/rbot/plugins/factoids.rb +506 -0
- data/data/rbot/plugins/figlet.rb +127 -18
- data/data/rbot/plugins/fish.rb +93 -33
- data/data/rbot/plugins/forecast.rb +40 -27
- data/data/rbot/plugins/fortune.rb +86 -11
- data/data/rbot/plugins/freshmeat.rb +46 -12
- data/data/rbot/plugins/games/azgame.rb +565 -0
- data/data/rbot/plugins/games/quiz.rb +961 -0
- data/data/rbot/plugins/games/roshambo.rb +62 -0
- data/data/rbot/plugins/{roulette.rb → games/roulette.rb} +58 -8
- data/data/rbot/plugins/games/shiritori.rb +485 -0
- data/data/rbot/plugins/games/uno.rb +1205 -0
- data/data/rbot/plugins/games/wheelfortune.rb +615 -0
- data/data/rbot/plugins/grouphug.rb +76 -22
- data/data/rbot/plugins/hl2.rb +97 -0
- data/data/rbot/plugins/host.rb +10 -1
- data/data/rbot/plugins/httpd.rb.disabled +1 -0
- data/data/rbot/plugins/imdb.rb +511 -47
- data/data/rbot/plugins/insult.rb +32 -37
- data/data/rbot/plugins/iplookup.rb +224 -224
- data/data/rbot/plugins/karma.rb +45 -21
- data/data/rbot/plugins/keywords.rb +258 -160
- data/data/rbot/plugins/lart.rb +126 -95
- data/data/rbot/plugins/lastfm.rb +421 -12
- data/data/rbot/plugins/linkbot.rb +75 -0
- data/data/rbot/plugins/markov.rb +103 -46
- data/data/rbot/plugins/math.rb +6 -5
- data/data/rbot/plugins/modes.rb +114 -0
- data/data/rbot/plugins/nickrecover.rb +97 -0
- data/data/rbot/plugins/nickserv.rb +111 -48
- data/data/rbot/plugins/nslookup.rb +15 -19
- data/data/rbot/plugins/plugin.header +11 -0
- data/data/rbot/plugins/quakeauth.rb +74 -11
- data/data/rbot/plugins/quotes.rb +189 -233
- data/data/rbot/plugins/reaction.rb +415 -0
- data/data/rbot/plugins/remind.rb +11 -15
- data/data/rbot/plugins/remotectl.rb +29 -0
- data/data/rbot/plugins/ri.rb +74 -0
- data/data/rbot/plugins/rot13.rb +16 -1
- data/data/rbot/plugins/rss.rb +1219 -585
- data/data/rbot/plugins/salut.rb +226 -0
- data/data/rbot/plugins/script.rb +184 -0
- data/data/rbot/plugins/search.rb +233 -0
- data/data/rbot/plugins/seen.rb +44 -40
- data/data/rbot/plugins/shortenurls.rb +93 -0
- data/data/rbot/plugins/slashdot.rb +49 -11
- data/data/rbot/plugins/spell.rb +37 -27
- data/data/rbot/plugins/theyfightcrime.rb +36 -39
- data/data/rbot/plugins/threat.rb +25 -22
- data/data/rbot/plugins/time.rb +135 -0
- data/data/rbot/plugins/topic.rb +107 -50
- data/data/rbot/plugins/translator.rb +360 -0
- data/data/rbot/plugins/tube.rb +28 -40
- data/data/rbot/plugins/twitter.rb +187 -0
- data/data/rbot/plugins/urban.rb +62 -53
- data/data/rbot/plugins/url.rb +193 -409
- data/data/rbot/plugins/usermodes.rb +26 -0
- data/data/rbot/plugins/wall.rb +60 -0
- data/data/rbot/plugins/weather.rb +254 -598
- data/data/rbot/plugins/wow.rb +41 -37
- data/data/rbot/plugins/wserver.rb +17 -25
- data/data/rbot/plugins/youtube.rb +237 -0
- data/data/rbot/templates/lart/{larts → larts-english} +0 -0
- data/data/rbot/templates/lart/larts-italian +25 -0
- data/data/rbot/templates/lart/larts-japanese +10 -0
- data/data/rbot/templates/lart/{praises → praises-english} +0 -0
- data/data/rbot/templates/lart/praises-italian +4 -0
- data/data/rbot/templates/lart/praises-japanese +3 -0
- data/data/rbot/templates/quiz/win_messages +11 -0
- data/data/rbot/templates/salut/salut-english +40 -0
- data/data/rbot/templates/salut/salut-french +29 -0
- data/data/rbot/templates/salut/salut-italian +51 -0
- data/data/rbot/templates/salut/salut-japanese +31 -0
- data/lib/rbot/botuser.rb +940 -0
- data/lib/rbot/config-compat.rb +23 -0
- data/lib/rbot/config.rb +211 -276
- data/lib/rbot/core/auth.rb +1063 -0
- data/lib/rbot/core/basics.rb +214 -0
- data/lib/rbot/core/config.rb +330 -0
- data/lib/rbot/core/filters_ui.rb +64 -0
- data/lib/rbot/core/irclog.rb +292 -0
- data/lib/rbot/core/remote.rb +401 -0
- data/lib/rbot/core/unicode.rb +91 -0
- data/lib/rbot/core/userdata.rb +200 -0
- data/lib/rbot/core/utils/extends.rb +444 -0
- data/lib/rbot/core/utils/filters.rb +154 -0
- data/lib/rbot/core/utils/httputil.rb +689 -0
- data/lib/rbot/core/utils/utils.rb +717 -0
- data/lib/rbot/dbhash.rb +30 -9
- data/lib/rbot/irc.rb +1966 -0
- data/lib/rbot/ircbot.rb +816 -675
- data/lib/rbot/ircsocket.rb +246 -232
- data/lib/rbot/language.rb +86 -8
- data/lib/rbot/load-gettext.rb +154 -0
- data/lib/rbot/maskdb.rb +162 -0
- data/lib/rbot/message.rb +392 -65
- data/lib/rbot/messagemapper.rb +438 -94
- data/lib/rbot/pkgconfig.rb +6 -0
- data/lib/rbot/plugins.rb +713 -172
- data/lib/rbot/post-config.rb +1 -0
- data/lib/rbot/rbotconfig.rb +31 -12
- data/lib/rbot/registry.rb +79 -48
- data/lib/rbot/rfc2812.rb +927 -519
- data/lib/rbot/timer.rb +232 -167
- data/po/en_US/rbot-alias.po +82 -0
- data/po/en_US/rbot-autoop.po +0 -0
- data/po/en_US/rbot-autorejoin.po +0 -0
- data/po/en_US/rbot-azgame.po +200 -0
- data/po/en_US/rbot-bans.po +0 -0
- data/po/en_US/rbot-bash.po +0 -0
- data/po/en_US/rbot-botsnack.po +0 -0
- data/po/en_US/rbot-cal.po +3 -0
- data/po/en_US/rbot-chanserv.po +0 -0
- data/po/en_US/rbot-chucknorris.po +0 -0
- data/po/en_US/rbot-debugger.po +0 -0
- data/po/en_US/rbot-deepthoughts.po +0 -0
- data/po/en_US/rbot-delicious.po +0 -0
- data/po/en_US/rbot-dice.po +0 -0
- data/po/en_US/rbot-dict.po +0 -0
- data/po/en_US/rbot-dictclient.po +124 -0
- data/po/en_US/rbot-digg.po +0 -0
- data/po/en_US/rbot-eightball.po +0 -0
- data/po/en_US/rbot-excuse.po +0 -0
- data/po/en_US/rbot-factoids.po +107 -0
- data/po/en_US/rbot-figlet.po +36 -0
- data/po/en_US/rbot-fish.po +0 -0
- data/po/en_US/rbot-forecast.po +0 -0
- data/po/en_US/rbot-fortune.po +0 -0
- data/po/en_US/rbot-freshmeat.po +0 -0
- data/po/en_US/rbot-grouphug.po +18 -0
- data/po/en_US/rbot-hl2.po +0 -0
- data/po/en_US/rbot-host.po +3 -0
- data/po/en_US/rbot-imdb.po +0 -0
- data/po/en_US/rbot-insult.po +0 -0
- data/po/en_US/rbot-iplookup.po +0 -0
- data/po/en_US/rbot-karma.po +0 -0
- data/po/en_US/rbot-keywords.po +24 -0
- data/po/en_US/rbot-lart.po +0 -0
- data/po/en_US/rbot-lastfm.po +172 -0
- data/po/en_US/rbot-linkbot.po +0 -0
- data/po/en_US/rbot-markov.po +20 -0
- data/po/en_US/rbot-math.po +0 -0
- data/po/en_US/rbot-modes.po +0 -0
- data/po/en_US/rbot-nickrecover.po +40 -0
- data/po/en_US/rbot-nickserv.po +104 -0
- data/po/en_US/rbot-nslookup.po +0 -0
- data/po/en_US/rbot-quakeauth.po +0 -0
- data/po/en_US/rbot-quiz.po +0 -0
- data/po/en_US/rbot-quotes.po +108 -0
- data/po/en_US/rbot-reaction.po +0 -0
- data/po/en_US/rbot-remind.po +0 -0
- data/po/en_US/rbot-remotectl.po +0 -0
- data/po/en_US/rbot-ri.po +0 -0
- data/po/en_US/rbot-roshambo.po +0 -0
- data/po/en_US/rbot-rot13.po +0 -0
- data/po/en_US/rbot-roulette.po +0 -0
- data/po/en_US/rbot-rss.po +20 -0
- data/po/en_US/rbot-salut.po +0 -0
- data/po/en_US/rbot-script.po +0 -0
- data/po/en_US/rbot-search.po +0 -0
- data/po/en_US/rbot-seen.po +0 -0
- data/po/en_US/rbot-shiritori.po +113 -0
- data/po/en_US/rbot-shortenurls.po +0 -0
- data/po/en_US/rbot-slashdot.po +0 -0
- data/po/en_US/rbot-spell.po +54 -0
- data/po/en_US/rbot-theyfightcrime.po +0 -0
- data/po/en_US/rbot-threat.po +0 -0
- data/po/en_US/rbot-time.po +0 -0
- data/po/en_US/rbot-topic.po +0 -0
- data/po/en_US/rbot-translator.po +77 -0
- data/po/en_US/rbot-tube.po +0 -0
- data/po/en_US/rbot-twitter.po +24 -0
- data/po/en_US/rbot-uno.po +512 -0
- data/po/en_US/rbot-urban.po +0 -0
- data/po/en_US/rbot-url.po +0 -0
- data/po/en_US/rbot-usermodes.po +0 -0
- data/po/en_US/rbot-wall.po +33 -0
- data/po/en_US/rbot-weather.po +0 -0
- data/po/en_US/rbot-wheelfortune.po +205 -0
- data/po/en_US/rbot-wow.po +0 -0
- data/po/en_US/rbot-wserver.po +0 -0
- data/po/en_US/rbot-youtube.po +58 -0
- data/po/en_US/rbot.po +1097 -0
- data/po/fr/rbot-alias.po +97 -0
- data/po/fr/rbot-autoop.po +0 -0
- data/po/fr/rbot-autorejoin.po +0 -0
- data/po/fr/rbot-azgame.po +204 -0
- data/po/fr/rbot-bans.po +0 -0
- data/po/fr/rbot-bash.po +0 -0
- data/po/fr/rbot-botsnack.po +0 -0
- data/po/fr/rbot-cal.po +3 -0
- data/po/fr/rbot-chanserv.po +0 -0
- data/po/fr/rbot-chucknorris.po +0 -0
- data/po/fr/rbot-debugger.po +0 -0
- data/po/fr/rbot-deepthoughts.po +0 -0
- data/po/fr/rbot-delicious.po +0 -0
- data/po/fr/rbot-dice.po +0 -0
- data/po/fr/rbot-dict.po +0 -0
- data/po/fr/rbot-dictclient.po +134 -0
- data/po/fr/rbot-digg.po +0 -0
- data/po/fr/rbot-eightball.po +0 -0
- data/po/fr/rbot-excuse.po +0 -0
- data/po/fr/rbot-factoids.po +111 -0
- data/po/fr/rbot-figlet.po +37 -0
- data/po/fr/rbot-fish.po +0 -0
- data/po/fr/rbot-forecast.po +0 -0
- data/po/fr/rbot-fortune.po +0 -0
- data/po/fr/rbot-freshmeat.po +0 -0
- data/po/fr/rbot-grouphug.po +18 -0
- data/po/fr/rbot-hl2.po +0 -0
- data/po/fr/rbot-host.po +3 -0
- data/po/fr/rbot-imdb.po +0 -0
- data/po/fr/rbot-insult.po +0 -0
- data/po/fr/rbot-iplookup.po +0 -0
- data/po/fr/rbot-karma.po +0 -0
- data/po/fr/rbot-keywords.po +24 -0
- data/po/fr/rbot-lart.po +0 -0
- data/po/fr/rbot-lastfm.po +172 -0
- data/po/fr/rbot-linkbot.po +0 -0
- data/po/fr/rbot-markov.po +20 -0
- data/po/fr/rbot-math.po +0 -0
- data/po/fr/rbot-modes.po +0 -0
- data/po/fr/rbot-nickrecover.po +36 -0
- data/po/fr/rbot-nickserv.po +116 -0
- data/po/fr/rbot-nslookup.po +0 -0
- data/po/fr/rbot-quakeauth.po +0 -0
- data/po/fr/rbot-quiz.po +0 -0
- data/po/fr/rbot-quotes.po +138 -0
- data/po/fr/rbot-reaction.po +0 -0
- data/po/fr/rbot-remind.po +0 -0
- data/po/fr/rbot-remotectl.po +0 -0
- data/po/fr/rbot-ri.po +0 -0
- data/po/fr/rbot-roshambo.po +0 -0
- data/po/fr/rbot-rot13.po +0 -0
- data/po/fr/rbot-roulette.po +0 -0
- data/po/fr/rbot-rss.po +20 -0
- data/po/fr/rbot-salut.po +0 -0
- data/po/fr/rbot-script.po +0 -0
- data/po/fr/rbot-search.po +0 -0
- data/po/fr/rbot-seen.po +0 -0
- data/po/fr/rbot-shiritori.po +119 -0
- data/po/fr/rbot-shortenurls.po +0 -0
- data/po/fr/rbot-slashdot.po +0 -0
- data/po/fr/rbot-spell.po +56 -0
- data/po/fr/rbot-theyfightcrime.po +0 -0
- data/po/fr/rbot-threat.po +0 -0
- data/po/fr/rbot-time.po +0 -0
- data/po/fr/rbot-topic.po +0 -0
- data/po/fr/rbot-translator.po +90 -0
- data/po/fr/rbot-tube.po +0 -0
- data/po/fr/rbot-twitter.po +24 -0
- data/po/fr/rbot-uno.po +521 -0
- data/po/fr/rbot-urban.po +0 -0
- data/po/fr/rbot-url.po +0 -0
- data/po/fr/rbot-usermodes.po +0 -0
- data/po/fr/rbot-wall.po +33 -0
- data/po/fr/rbot-weather.po +0 -0
- data/po/fr/rbot-wheelfortune.po +255 -0
- data/po/fr/rbot-wow.po +0 -0
- data/po/fr/rbot-wserver.po +0 -0
- data/po/fr/rbot-youtube.po +64 -0
- data/po/fr/rbot.po +1133 -0
- data/po/it/rbot-alias.po +82 -0
- data/po/it/rbot-autoop.po +0 -0
- data/po/it/rbot-autorejoin.po +0 -0
- data/po/it/rbot-azgame.po +186 -0
- data/po/it/rbot-bans.po +0 -0
- data/po/it/rbot-bash.po +0 -0
- data/po/it/rbot-botsnack.po +0 -0
- data/po/it/rbot-cal.po +3 -0
- data/po/it/rbot-chanserv.po +0 -0
- data/po/it/rbot-chucknorris.po +0 -0
- data/po/it/rbot-debugger.po +0 -0
- data/po/it/rbot-deepthoughts.po +0 -0
- data/po/it/rbot-delicious.po +0 -0
- data/po/it/rbot-dice.po +0 -0
- data/po/it/rbot-dict.po +0 -0
- data/po/it/rbot-dictclient.po +111 -0
- data/po/it/rbot-digg.po +0 -0
- data/po/it/rbot-eightball.po +0 -0
- data/po/it/rbot-excuse.po +0 -0
- data/po/it/rbot-factoids.po +110 -0
- data/po/it/rbot-figlet.po +36 -0
- data/po/it/rbot-fish.po +0 -0
- data/po/it/rbot-forecast.po +0 -0
- data/po/it/rbot-fortune.po +0 -0
- data/po/it/rbot-freshmeat.po +0 -0
- data/po/it/rbot-grouphug.po +18 -0
- data/po/it/rbot-hl2.po +0 -0
- data/po/it/rbot-host.po +3 -0
- data/po/it/rbot-imdb.po +0 -0
- data/po/it/rbot-insult.po +0 -0
- data/po/it/rbot-iplookup.po +0 -0
- data/po/it/rbot-karma.po +0 -0
- data/po/it/rbot-keywords.po +24 -0
- data/po/it/rbot-lart.po +0 -0
- data/po/it/rbot-lastfm.po +172 -0
- data/po/it/rbot-linkbot.po +0 -0
- data/po/it/rbot-markov.po +20 -0
- data/po/it/rbot-math.po +0 -0
- data/po/it/rbot-modes.po +0 -0
- data/po/it/rbot-nickrecover.po +36 -0
- data/po/it/rbot-nickserv.po +104 -0
- data/po/it/rbot-nslookup.po +0 -0
- data/po/it/rbot-quakeauth.po +0 -0
- data/po/it/rbot-quiz.po +0 -0
- data/po/it/rbot-quotes.po +108 -0
- data/po/it/rbot-reaction.po +0 -0
- data/po/it/rbot-remind.po +0 -0
- data/po/it/rbot-remotectl.po +0 -0
- data/po/it/rbot-ri.po +0 -0
- data/po/it/rbot-roshambo.po +0 -0
- data/po/it/rbot-rot13.po +0 -0
- data/po/it/rbot-roulette.po +0 -0
- data/po/it/rbot-rss.po +20 -0
- data/po/it/rbot-salut.po +0 -0
- data/po/it/rbot-script.po +0 -0
- data/po/it/rbot-search.po +0 -0
- data/po/it/rbot-seen.po +0 -0
- data/po/it/rbot-shiritori.po +102 -0
- data/po/it/rbot-shortenurls.po +0 -0
- data/po/it/rbot-slashdot.po +0 -0
- data/po/it/rbot-spell.po +54 -0
- data/po/it/rbot-theyfightcrime.po +0 -0
- data/po/it/rbot-threat.po +0 -0
- data/po/it/rbot-time.po +0 -0
- data/po/it/rbot-topic.po +0 -0
- data/po/it/rbot-translator.po +77 -0
- data/po/it/rbot-tube.po +0 -0
- data/po/it/rbot-twitter.po +24 -0
- data/po/it/rbot-uno.po +564 -0
- data/po/it/rbot-urban.po +0 -0
- data/po/it/rbot-url.po +0 -0
- data/po/it/rbot-usermodes.po +0 -0
- data/po/it/rbot-wall.po +33 -0
- data/po/it/rbot-weather.po +0 -0
- data/po/it/rbot-wheelfortune.po +251 -0
- data/po/it/rbot-wow.po +0 -0
- data/po/it/rbot-wserver.po +0 -0
- data/po/it/rbot-youtube.po +64 -0
- data/po/it/rbot.po +1127 -0
- data/po/ja/rbot-alias.po +82 -0
- data/po/ja/rbot-autoop.po +0 -0
- data/po/ja/rbot-autorejoin.po +0 -0
- data/po/ja/rbot-azgame.po +194 -0
- data/po/ja/rbot-bans.po +0 -0
- data/po/ja/rbot-bash.po +0 -0
- data/po/ja/rbot-botsnack.po +0 -0
- data/po/ja/rbot-cal.po +3 -0
- data/po/ja/rbot-chanserv.po +0 -0
- data/po/ja/rbot-chucknorris.po +0 -0
- data/po/ja/rbot-debugger.po +0 -0
- data/po/ja/rbot-deepthoughts.po +0 -0
- data/po/ja/rbot-delicious.po +0 -0
- data/po/ja/rbot-dice.po +0 -0
- data/po/ja/rbot-dict.po +0 -0
- data/po/ja/rbot-dictclient.po +111 -0
- data/po/ja/rbot-digg.po +0 -0
- data/po/ja/rbot-eightball.po +0 -0
- data/po/ja/rbot-excuse.po +0 -0
- data/po/ja/rbot-factoids.po +107 -0
- data/po/ja/rbot-figlet.po +36 -0
- data/po/ja/rbot-fish.po +0 -0
- data/po/ja/rbot-forecast.po +0 -0
- data/po/ja/rbot-fortune.po +0 -0
- data/po/ja/rbot-freshmeat.po +0 -0
- data/po/ja/rbot-grouphug.po +18 -0
- data/po/ja/rbot-hl2.po +0 -0
- data/po/ja/rbot-host.po +3 -0
- data/po/ja/rbot-imdb.po +0 -0
- data/po/ja/rbot-insult.po +0 -0
- data/po/ja/rbot-iplookup.po +0 -0
- data/po/ja/rbot-karma.po +0 -0
- data/po/ja/rbot-keywords.po +24 -0
- data/po/ja/rbot-lart.po +0 -0
- data/po/ja/rbot-lastfm.po +172 -0
- data/po/ja/rbot-linkbot.po +0 -0
- data/po/ja/rbot-markov.po +20 -0
- data/po/ja/rbot-math.po +0 -0
- data/po/ja/rbot-modes.po +0 -0
- data/po/ja/rbot-nickrecover.po +36 -0
- data/po/ja/rbot-nickserv.po +104 -0
- data/po/ja/rbot-nslookup.po +0 -0
- data/po/ja/rbot-quakeauth.po +0 -0
- data/po/ja/rbot-quiz.po +0 -0
- data/po/ja/rbot-quotes.po +108 -0
- data/po/ja/rbot-reaction.po +0 -0
- data/po/ja/rbot-remind.po +0 -0
- data/po/ja/rbot-remotectl.po +0 -0
- data/po/ja/rbot-ri.po +0 -0
- data/po/ja/rbot-roshambo.po +0 -0
- data/po/ja/rbot-rot13.po +0 -0
- data/po/ja/rbot-roulette.po +0 -0
- data/po/ja/rbot-rss.po +20 -0
- data/po/ja/rbot-salut.po +0 -0
- data/po/ja/rbot-script.po +0 -0
- data/po/ja/rbot-search.po +0 -0
- data/po/ja/rbot-seen.po +0 -0
- data/po/ja/rbot-shiritori.po +115 -0
- data/po/ja/rbot-shortenurls.po +0 -0
- data/po/ja/rbot-slashdot.po +0 -0
- data/po/ja/rbot-spell.po +54 -0
- data/po/ja/rbot-theyfightcrime.po +0 -0
- data/po/ja/rbot-threat.po +0 -0
- data/po/ja/rbot-time.po +0 -0
- data/po/ja/rbot-topic.po +0 -0
- data/po/ja/rbot-translator.po +77 -0
- data/po/ja/rbot-tube.po +0 -0
- data/po/ja/rbot-twitter.po +24 -0
- data/po/ja/rbot-uno.po +512 -0
- data/po/ja/rbot-urban.po +0 -0
- data/po/ja/rbot-url.po +0 -0
- data/po/ja/rbot-usermodes.po +0 -0
- data/po/ja/rbot-wall.po +33 -0
- data/po/ja/rbot-weather.po +0 -0
- data/po/ja/rbot-wheelfortune.po +205 -0
- data/po/ja/rbot-wow.po +0 -0
- data/po/ja/rbot-wserver.po +0 -0
- data/po/ja/rbot-youtube.po +58 -0
- data/po/ja/rbot.po +999 -0
- data/po/rbot-alias.pot +83 -0
- data/po/rbot-autoop.pot +0 -0
- data/po/rbot-autorejoin.pot +0 -0
- data/po/rbot-azgame.pot +187 -0
- data/po/rbot-bans.pot +0 -0
- data/po/rbot-bash.pot +0 -0
- data/po/rbot-botsnack.pot +0 -0
- data/po/rbot-cal.pot +21 -0
- data/po/rbot-chanserv.pot +0 -0
- data/po/rbot-chucknorris.pot +0 -0
- data/po/rbot-debugger.pot +0 -0
- data/po/rbot-deepthoughts.pot +0 -0
- data/po/rbot-delicious.pot +0 -0
- data/po/rbot-dice.pot +0 -0
- data/po/rbot-dict.pot +0 -0
- data/po/rbot-dictclient.pot +112 -0
- data/po/rbot-digg.pot +0 -0
- data/po/rbot-eightball.pot +0 -0
- data/po/rbot-excuse.pot +0 -0
- data/po/rbot-factoids.pot +108 -0
- data/po/rbot-figlet.pot +37 -0
- data/po/rbot-fish.pot +0 -0
- data/po/rbot-forecast.pot +0 -0
- data/po/rbot-fortune.pot +0 -0
- data/po/rbot-freshmeat.pot +0 -0
- data/po/rbot-grouphug.pot +36 -0
- data/po/rbot-hl2.pot +0 -0
- data/po/rbot-host.pot +21 -0
- data/po/rbot-imdb.pot +0 -0
- data/po/rbot-insult.pot +0 -0
- data/po/rbot-iplookup.pot +0 -0
- data/po/rbot-karma.pot +0 -0
- data/po/rbot-keywords.pot +25 -0
- data/po/rbot-lart.pot +0 -0
- data/po/rbot-lastfm.pot +190 -0
- data/po/rbot-linkbot.pot +0 -0
- data/po/rbot-markov.pot +21 -0
- data/po/rbot-math.pot +0 -0
- data/po/rbot-modes.pot +0 -0
- data/po/rbot-nickrecover.pot +37 -0
- data/po/rbot-nickserv.pot +105 -0
- data/po/rbot-nslookup.pot +0 -0
- data/po/rbot-quakeauth.pot +0 -0
- data/po/rbot-quiz.pot +0 -0
- data/po/rbot-quotes.pot +109 -0
- data/po/rbot-reaction.pot +0 -0
- data/po/rbot-remind.pot +0 -0
- data/po/rbot-remotectl.pot +0 -0
- data/po/rbot-ri.pot +0 -0
- data/po/rbot-roshambo.pot +0 -0
- data/po/rbot-rot13.pot +0 -0
- data/po/rbot-roulette.pot +0 -0
- data/po/rbot-rss.pot +21 -0
- data/po/rbot-salut.pot +0 -0
- data/po/rbot-script.pot +0 -0
- data/po/rbot-search.pot +0 -0
- data/po/rbot-seen.pot +0 -0
- data/po/rbot-shiritori.pot +103 -0
- data/po/rbot-shortenurls.pot +0 -0
- data/po/rbot-slashdot.pot +0 -0
- data/po/rbot-spell.pot +55 -0
- data/po/rbot-theyfightcrime.pot +0 -0
- data/po/rbot-threat.pot +0 -0
- data/po/rbot-time.pot +0 -0
- data/po/rbot-topic.pot +0 -0
- data/po/rbot-translator.pot +78 -0
- data/po/rbot-tube.pot +0 -0
- data/po/rbot-twitter.pot +25 -0
- data/po/rbot-uno.pot +513 -0
- data/po/rbot-urban.pot +0 -0
- data/po/rbot-url.pot +0 -0
- data/po/rbot-usermodes.pot +0 -0
- data/po/rbot-wall.pot +33 -0
- data/po/rbot-weather.pot +0 -0
- data/po/rbot-wheelfortune.pot +206 -0
- data/po/rbot-wow.pot +0 -0
- data/po/rbot-wserver.pot +0 -0
- data/po/rbot-youtube.pot +59 -0
- data/po/rbot.pot +1004 -0
- data/po/zh_CN/rbot-alias.po +82 -0
- data/po/zh_CN/rbot-autoop.po +0 -0
- data/po/zh_CN/rbot-autorejoin.po +0 -0
- data/po/zh_CN/rbot-azgame.po +186 -0
- data/po/zh_CN/rbot-bans.po +0 -0
- data/po/zh_CN/rbot-bash.po +0 -0
- data/po/zh_CN/rbot-botsnack.po +0 -0
- data/po/zh_CN/rbot-cal.po +3 -0
- data/po/zh_CN/rbot-chanserv.po +0 -0
- data/po/zh_CN/rbot-chucknorris.po +0 -0
- data/po/zh_CN/rbot-debugger.po +0 -0
- data/po/zh_CN/rbot-deepthoughts.po +0 -0
- data/po/zh_CN/rbot-delicious.po +0 -0
- data/po/zh_CN/rbot-dice.po +0 -0
- data/po/zh_CN/rbot-dict.po +0 -0
- data/po/zh_CN/rbot-dictclient.po +111 -0
- data/po/zh_CN/rbot-digg.po +0 -0
- data/po/zh_CN/rbot-eightball.po +0 -0
- data/po/zh_CN/rbot-excuse.po +0 -0
- data/po/zh_CN/rbot-factoids.po +107 -0
- data/po/zh_CN/rbot-figlet.po +36 -0
- data/po/zh_CN/rbot-fish.po +0 -0
- data/po/zh_CN/rbot-forecast.po +0 -0
- data/po/zh_CN/rbot-fortune.po +0 -0
- data/po/zh_CN/rbot-freshmeat.po +0 -0
- data/po/zh_CN/rbot-grouphug.po +18 -0
- data/po/zh_CN/rbot-hl2.po +0 -0
- data/po/zh_CN/rbot-host.po +3 -0
- data/po/zh_CN/rbot-imdb.po +0 -0
- data/po/zh_CN/rbot-insult.po +0 -0
- data/po/zh_CN/rbot-iplookup.po +0 -0
- data/po/zh_CN/rbot-karma.po +0 -0
- data/po/zh_CN/rbot-keywords.po +24 -0
- data/po/zh_CN/rbot-lart.po +0 -0
- data/po/zh_CN/rbot-lastfm.po +172 -0
- data/po/zh_CN/rbot-linkbot.po +0 -0
- data/po/zh_CN/rbot-markov.po +20 -0
- data/po/zh_CN/rbot-math.po +0 -0
- data/po/zh_CN/rbot-modes.po +0 -0
- data/po/zh_CN/rbot-nickrecover.po +36 -0
- data/po/zh_CN/rbot-nickserv.po +104 -0
- data/po/zh_CN/rbot-nslookup.po +0 -0
- data/po/zh_CN/rbot-quakeauth.po +0 -0
- data/po/zh_CN/rbot-quiz.po +0 -0
- data/po/zh_CN/rbot-quotes.po +108 -0
- data/po/zh_CN/rbot-reaction.po +0 -0
- data/po/zh_CN/rbot-remind.po +0 -0
- data/po/zh_CN/rbot-remotectl.po +0 -0
- data/po/zh_CN/rbot-ri.po +0 -0
- data/po/zh_CN/rbot-roshambo.po +0 -0
- data/po/zh_CN/rbot-rot13.po +0 -0
- data/po/zh_CN/rbot-roulette.po +0 -0
- data/po/zh_CN/rbot-rss.po +20 -0
- data/po/zh_CN/rbot-salut.po +0 -0
- data/po/zh_CN/rbot-script.po +0 -0
- data/po/zh_CN/rbot-search.po +0 -0
- data/po/zh_CN/rbot-seen.po +0 -0
- data/po/zh_CN/rbot-shiritori.po +102 -0
- data/po/zh_CN/rbot-shortenurls.po +0 -0
- data/po/zh_CN/rbot-slashdot.po +0 -0
- data/po/zh_CN/rbot-spell.po +54 -0
- data/po/zh_CN/rbot-theyfightcrime.po +0 -0
- data/po/zh_CN/rbot-threat.po +0 -0
- data/po/zh_CN/rbot-time.po +0 -0
- data/po/zh_CN/rbot-topic.po +0 -0
- data/po/zh_CN/rbot-translator.po +77 -0
- data/po/zh_CN/rbot-tube.po +0 -0
- data/po/zh_CN/rbot-twitter.po +24 -0
- data/po/zh_CN/rbot-uno.po +512 -0
- data/po/zh_CN/rbot-urban.po +0 -0
- data/po/zh_CN/rbot-url.po +0 -0
- data/po/zh_CN/rbot-usermodes.po +0 -0
- data/po/zh_CN/rbot-wall.po +33 -0
- data/po/zh_CN/rbot-weather.po +0 -0
- data/po/zh_CN/rbot-wheelfortune.po +205 -0
- data/po/zh_CN/rbot-wow.po +0 -0
- data/po/zh_CN/rbot-wserver.po +0 -0
- data/po/zh_CN/rbot-youtube.po +58 -0
- data/po/zh_CN/rbot.po +1006 -0
- data/po/zh_TW/rbot-alias.po +82 -0
- data/po/zh_TW/rbot-autoop.po +0 -0
- data/po/zh_TW/rbot-autorejoin.po +0 -0
- data/po/zh_TW/rbot-azgame.po +186 -0
- data/po/zh_TW/rbot-bans.po +0 -0
- data/po/zh_TW/rbot-bash.po +0 -0
- data/po/zh_TW/rbot-botsnack.po +0 -0
- data/po/zh_TW/rbot-cal.po +3 -0
- data/po/zh_TW/rbot-chanserv.po +0 -0
- data/po/zh_TW/rbot-chucknorris.po +0 -0
- data/po/zh_TW/rbot-debugger.po +0 -0
- data/po/zh_TW/rbot-deepthoughts.po +0 -0
- data/po/zh_TW/rbot-delicious.po +0 -0
- data/po/zh_TW/rbot-dice.po +0 -0
- data/po/zh_TW/rbot-dict.po +0 -0
- data/po/zh_TW/rbot-dictclient.po +111 -0
- data/po/zh_TW/rbot-digg.po +0 -0
- data/po/zh_TW/rbot-eightball.po +0 -0
- data/po/zh_TW/rbot-excuse.po +0 -0
- data/po/zh_TW/rbot-factoids.po +107 -0
- data/po/zh_TW/rbot-figlet.po +36 -0
- data/po/zh_TW/rbot-fish.po +0 -0
- data/po/zh_TW/rbot-forecast.po +0 -0
- data/po/zh_TW/rbot-fortune.po +0 -0
- data/po/zh_TW/rbot-freshmeat.po +0 -0
- data/po/zh_TW/rbot-grouphug.po +18 -0
- data/po/zh_TW/rbot-hl2.po +0 -0
- data/po/zh_TW/rbot-host.po +3 -0
- data/po/zh_TW/rbot-imdb.po +0 -0
- data/po/zh_TW/rbot-insult.po +0 -0
- data/po/zh_TW/rbot-iplookup.po +0 -0
- data/po/zh_TW/rbot-karma.po +0 -0
- data/po/zh_TW/rbot-keywords.po +24 -0
- data/po/zh_TW/rbot-lart.po +0 -0
- data/po/zh_TW/rbot-lastfm.po +172 -0
- data/po/zh_TW/rbot-linkbot.po +0 -0
- data/po/zh_TW/rbot-markov.po +20 -0
- data/po/zh_TW/rbot-math.po +0 -0
- data/po/zh_TW/rbot-modes.po +0 -0
- data/po/zh_TW/rbot-nickrecover.po +36 -0
- data/po/zh_TW/rbot-nickserv.po +104 -0
- data/po/zh_TW/rbot-nslookup.po +0 -0
- data/po/zh_TW/rbot-quakeauth.po +0 -0
- data/po/zh_TW/rbot-quiz.po +0 -0
- data/po/zh_TW/rbot-quotes.po +108 -0
- data/po/zh_TW/rbot-reaction.po +0 -0
- data/po/zh_TW/rbot-remind.po +0 -0
- data/po/zh_TW/rbot-remotectl.po +0 -0
- data/po/zh_TW/rbot-ri.po +0 -0
- data/po/zh_TW/rbot-roshambo.po +0 -0
- data/po/zh_TW/rbot-rot13.po +0 -0
- data/po/zh_TW/rbot-roulette.po +0 -0
- data/po/zh_TW/rbot-rss.po +20 -0
- data/po/zh_TW/rbot-salut.po +0 -0
- data/po/zh_TW/rbot-script.po +0 -0
- data/po/zh_TW/rbot-search.po +0 -0
- data/po/zh_TW/rbot-seen.po +0 -0
- data/po/zh_TW/rbot-shiritori.po +102 -0
- data/po/zh_TW/rbot-shortenurls.po +0 -0
- data/po/zh_TW/rbot-slashdot.po +0 -0
- data/po/zh_TW/rbot-spell.po +54 -0
- data/po/zh_TW/rbot-theyfightcrime.po +0 -0
- data/po/zh_TW/rbot-threat.po +0 -0
- data/po/zh_TW/rbot-time.po +0 -0
- data/po/zh_TW/rbot-topic.po +0 -0
- data/po/zh_TW/rbot-translator.po +77 -0
- data/po/zh_TW/rbot-tube.po +0 -0
- data/po/zh_TW/rbot-twitter.po +24 -0
- data/po/zh_TW/rbot-uno.po +512 -0
- data/po/zh_TW/rbot-urban.po +0 -0
- data/po/zh_TW/rbot-url.po +0 -0
- data/po/zh_TW/rbot-usermodes.po +0 -0
- data/po/zh_TW/rbot-wall.po +33 -0
- data/po/zh_TW/rbot-weather.po +0 -0
- data/po/zh_TW/rbot-wheelfortune.po +205 -0
- data/po/zh_TW/rbot-wow.po +0 -0
- data/po/zh_TW/rbot-wserver.po +0 -0
- data/po/zh_TW/rbot-youtube.po +58 -0
- data/po/zh_TW/rbot.po +1034 -0
- data/setup.rb +800 -574
- metadata +723 -108
- data/data/rbot/contrib/plugins/figlet.rb +0 -20
- data/data/rbot/contrib/plugins/ri.rb +0 -83
- data/data/rbot/plugins/demauro.rb +0 -95
- data/data/rbot/plugins/google.rb +0 -53
- data/data/rbot/plugins/opme.rb +0 -19
- data/data/rbot/plugins/roshambo.rb +0 -54
- data/data/rbot/plugins/rubyurl.rb +0 -39
- data/data/rbot/plugins/tinyurl.rb +0 -39
- data/data/rbot/plugins/xmlrpc.rb.disabled +0 -52
- data/data/rbot/templates/levels.rbot +0 -32
- data/data/rbot/templates/users.rbot +0 -1
- data/lib/rbot/auth.rb +0 -314
- data/lib/rbot/channel.rb +0 -54
- data/lib/rbot/httputil.rb +0 -309
- data/lib/rbot/utils.rb +0 -83
@@ -0,0 +1,717 @@
|
|
1
|
+
#-- vim:sw=2:et
|
2
|
+
#++
|
3
|
+
#
|
4
|
+
# :title: rbot utilities provider
|
5
|
+
#
|
6
|
+
# Author:: Tom Gilbert <tom@linuxbrit.co.uk>
|
7
|
+
# Author:: Giuseppe "Oblomov" Bilotta <giuseppe.bilotta@gmail.com>
|
8
|
+
#
|
9
|
+
# TODO some of these Utils should be rewritten as extensions to the approriate
|
10
|
+
# standard Ruby classes and accordingly be moved to extends.rb
|
11
|
+
|
12
|
+
require 'tempfile'
|
13
|
+
require 'set'
|
14
|
+
|
15
|
+
# Try to load htmlentities, fall back to an HTML escape table.
|
16
|
+
begin
|
17
|
+
require 'htmlentities'
|
18
|
+
rescue LoadError
|
19
|
+
gems = nil
|
20
|
+
begin
|
21
|
+
gems = require 'rubygems'
|
22
|
+
rescue LoadError
|
23
|
+
gems = false
|
24
|
+
end
|
25
|
+
if gems
|
26
|
+
retry
|
27
|
+
else
|
28
|
+
module ::Irc
|
29
|
+
module Utils
|
30
|
+
UNESCAPE_TABLE = {
|
31
|
+
'laquo' => '«',
|
32
|
+
'raquo' => '»',
|
33
|
+
'quot' => '"',
|
34
|
+
'apos' => '\'',
|
35
|
+
'micro' => 'µ',
|
36
|
+
'copy' => '©',
|
37
|
+
'trade' => '™',
|
38
|
+
'reg' => '®',
|
39
|
+
'amp' => '&',
|
40
|
+
'lt' => '<',
|
41
|
+
'gt' => '>',
|
42
|
+
'hellip' => '…',
|
43
|
+
'nbsp' => ' ',
|
44
|
+
'Agrave' => 'À',
|
45
|
+
'Aacute' => 'Á',
|
46
|
+
'Acirc' => 'Â',
|
47
|
+
'Atilde' => 'Ã',
|
48
|
+
'Auml' => 'Ä',
|
49
|
+
'Aring' => 'Å',
|
50
|
+
'AElig' => 'Æ',
|
51
|
+
'OElig' => 'Œ',
|
52
|
+
'Ccedil' => 'Ç',
|
53
|
+
'Egrave' => 'È',
|
54
|
+
'Eacute' => 'É',
|
55
|
+
'Ecirc' => 'Ê',
|
56
|
+
'Euml' => 'Ë',
|
57
|
+
'Igrave' => 'Ì',
|
58
|
+
'Iacute' => 'Í',
|
59
|
+
'Icirc' => 'Î',
|
60
|
+
'Iuml' => 'Ï',
|
61
|
+
'ETH' => 'Ð',
|
62
|
+
'Ntilde' => 'Ñ',
|
63
|
+
'Ograve' => 'Ò',
|
64
|
+
'Oacute' => 'Ó',
|
65
|
+
'Ocirc' => 'Ô',
|
66
|
+
'Otilde' => 'Õ',
|
67
|
+
'Ouml' => 'Ö',
|
68
|
+
'Oslash' => 'Ø',
|
69
|
+
'Ugrave' => 'Ù',
|
70
|
+
'Uacute' => 'Ú',
|
71
|
+
'Ucirc' => 'Û',
|
72
|
+
'Uuml' => 'Ü',
|
73
|
+
'Yacute' => 'Ý',
|
74
|
+
'THORN' => 'Þ',
|
75
|
+
'szlig' => 'ß',
|
76
|
+
'agrave' => 'à',
|
77
|
+
'aacute' => 'á',
|
78
|
+
'acirc' => 'â',
|
79
|
+
'atilde' => 'ã',
|
80
|
+
'auml' => 'ä',
|
81
|
+
'aring' => 'å',
|
82
|
+
'aelig' => 'æ',
|
83
|
+
'oelig' => 'œ',
|
84
|
+
'ccedil' => 'ç',
|
85
|
+
'egrave' => 'è',
|
86
|
+
'eacute' => 'é',
|
87
|
+
'ecirc' => 'ê',
|
88
|
+
'euml' => 'ë',
|
89
|
+
'igrave' => 'ì',
|
90
|
+
'iacute' => 'í',
|
91
|
+
'icirc' => 'î',
|
92
|
+
'iuml' => 'ï',
|
93
|
+
'eth' => 'ð',
|
94
|
+
'ntilde' => 'ñ',
|
95
|
+
'ograve' => 'ò',
|
96
|
+
'oacute' => 'ó',
|
97
|
+
'ocirc' => 'ô',
|
98
|
+
'otilde' => 'õ',
|
99
|
+
'ouml' => 'ö',
|
100
|
+
'oslash' => 'ø',
|
101
|
+
'ugrave' => 'ù',
|
102
|
+
'uacute' => 'ú',
|
103
|
+
'ucirc' => 'û',
|
104
|
+
'uuml' => 'ü',
|
105
|
+
'yacute' => 'ý',
|
106
|
+
'thorn' => 'þ',
|
107
|
+
'yuml' => 'ÿ'
|
108
|
+
}
|
109
|
+
end
|
110
|
+
end
|
111
|
+
end
|
112
|
+
end
|
113
|
+
|
114
|
+
begin
|
115
|
+
require 'hpricot'
|
116
|
+
module ::Irc
|
117
|
+
module Utils
|
118
|
+
AFTER_PAR_PATH = /^(?:div|span)$/
|
119
|
+
AFTER_PAR_EX = /^(?:td|tr|tbody|table)$/
|
120
|
+
AFTER_PAR_CLASS = /body|message|text/i
|
121
|
+
end
|
122
|
+
end
|
123
|
+
rescue LoadError
|
124
|
+
gems = nil
|
125
|
+
begin
|
126
|
+
gems = require 'rubygems'
|
127
|
+
rescue LoadError
|
128
|
+
gems = false
|
129
|
+
end
|
130
|
+
if gems
|
131
|
+
retry
|
132
|
+
else
|
133
|
+
module ::Irc
|
134
|
+
module Utils
|
135
|
+
# Some regular expressions to manage HTML data
|
136
|
+
|
137
|
+
# Title
|
138
|
+
TITLE_REGEX = /<\s*?title\s*?>(.+?)<\s*?\/title\s*?>/im
|
139
|
+
|
140
|
+
# H1, H2, etc
|
141
|
+
HX_REGEX = /<h(\d)(?:\s+[^>]*)?>(.*?)<\/h\1>/im
|
142
|
+
# A paragraph
|
143
|
+
PAR_REGEX = /<p(?:\s+[^>]*)?>.*?<\/?(?:p|div|html|body|table|td|tr)(?:\s+[^>]*)?>/im
|
144
|
+
|
145
|
+
# Some blogging and forum platforms use spans or divs with a 'body' or 'message' or 'text' in their class
|
146
|
+
# to mark actual text
|
147
|
+
AFTER_PAR1_REGEX = /<\w+\s+[^>]*(?:body|message|text)[^>]*>.*?<\/?(?:p|div|html|body|table|td|tr)(?:\s+[^>]*)?>/im
|
148
|
+
|
149
|
+
# At worst, we can try stuff which is comprised between two <br>
|
150
|
+
AFTER_PAR2_REGEX = /<br(?:\s+[^>]*)?\/?>.*?<\/?(?:br|p|div|html|body|table|td|tr)(?:\s+[^>]*)?\/?>/im
|
151
|
+
end
|
152
|
+
end
|
153
|
+
end
|
154
|
+
end
|
155
|
+
|
156
|
+
module ::Irc
|
157
|
+
|
158
|
+
# Miscellaneous useful functions
|
159
|
+
module Utils
|
160
|
+
@@bot = nil unless defined? @@bot
|
161
|
+
@@safe_save_dir = nil unless defined?(@@safe_save_dir)
|
162
|
+
|
163
|
+
# The bot instance
|
164
|
+
def Utils.bot
|
165
|
+
@@bot
|
166
|
+
end
|
167
|
+
|
168
|
+
# Set up some Utils routines which depend on the associated bot.
|
169
|
+
def Utils.bot=(b)
|
170
|
+
debug "initializing utils"
|
171
|
+
@@bot = b
|
172
|
+
@@safe_save_dir = "#{@@bot.botclass}/safe_save"
|
173
|
+
end
|
174
|
+
|
175
|
+
|
176
|
+
# Seconds per minute
|
177
|
+
SEC_PER_MIN = 60
|
178
|
+
# Seconds per hour
|
179
|
+
SEC_PER_HR = SEC_PER_MIN * 60
|
180
|
+
# Seconds per day
|
181
|
+
SEC_PER_DAY = SEC_PER_HR * 24
|
182
|
+
# Seconds per (30-day) month
|
183
|
+
SEC_PER_MNTH = SEC_PER_DAY * 30
|
184
|
+
# Second per (30*12 = 360 day) year
|
185
|
+
SEC_PER_YR = SEC_PER_MNTH * 12
|
186
|
+
|
187
|
+
# Auxiliary method needed by Utils.secs_to_string
|
188
|
+
def Utils.secs_to_string_case(array, var, string, plural)
|
189
|
+
case var
|
190
|
+
when 1
|
191
|
+
array << "1 #{string}"
|
192
|
+
else
|
193
|
+
array << "#{var} #{plural}"
|
194
|
+
end
|
195
|
+
end
|
196
|
+
|
197
|
+
# Turn a number of seconds into a human readable string, e.g
|
198
|
+
# 2 days, 3 hours, 18 minutes and 10 seconds
|
199
|
+
def Utils.secs_to_string(secs)
|
200
|
+
ret = []
|
201
|
+
years, secs = secs.divmod SEC_PER_YR
|
202
|
+
secs_to_string_case(ret, years, _("year"), _("years")) if years > 0
|
203
|
+
months, secs = secs.divmod SEC_PER_MNTH
|
204
|
+
secs_to_string_case(ret, months, _("month"), _("months")) if months > 0
|
205
|
+
days, secs = secs.divmod SEC_PER_DAY
|
206
|
+
secs_to_string_case(ret, days, _("day"), _("days")) if days > 0
|
207
|
+
hours, secs = secs.divmod SEC_PER_HR
|
208
|
+
secs_to_string_case(ret, hours, _("hour"), _("hours")) if hours > 0
|
209
|
+
mins, secs = secs.divmod SEC_PER_MIN
|
210
|
+
secs_to_string_case(ret, mins, _("minute"), _("minutes")) if mins > 0
|
211
|
+
secs = secs.to_i
|
212
|
+
secs_to_string_case(ret, secs, _("second"), _("seconds")) if secs > 0 or ret.empty?
|
213
|
+
case ret.length
|
214
|
+
when 0
|
215
|
+
raise "Empty ret array!"
|
216
|
+
when 1
|
217
|
+
return ret.to_s
|
218
|
+
else
|
219
|
+
return [ret[0, ret.length-1].join(", ") , ret[-1]].join(_(" and "))
|
220
|
+
end
|
221
|
+
end
|
222
|
+
|
223
|
+
# Turn a number of seconds into a hours:minutes:seconds e.g.
|
224
|
+
# 3:18:10 or 5'12" or 7s
|
225
|
+
#
|
226
|
+
def Utils.secs_to_short(seconds)
|
227
|
+
secs = seconds.to_i # make sure it's an integer
|
228
|
+
mins, secs = secs.divmod 60
|
229
|
+
hours, mins = mins.divmod 60
|
230
|
+
if hours > 0
|
231
|
+
return ("%s:%s:%s" % [hours, mins, secs])
|
232
|
+
elsif mins > 0
|
233
|
+
return ("%s'%s\"" % [mins, secs])
|
234
|
+
else
|
235
|
+
return ("%ss" % [secs])
|
236
|
+
end
|
237
|
+
end
|
238
|
+
|
239
|
+
# Returns human readable time.
|
240
|
+
# Like: 5 days ago
|
241
|
+
# about one hour ago
|
242
|
+
# options
|
243
|
+
# :start_date, sets the time to measure against, defaults to now
|
244
|
+
# :date_format, used with <tt>to_formatted_s<tt>, default to :default
|
245
|
+
def Utils.timeago(time, options = {})
|
246
|
+
start_date = options.delete(:start_date) || Time.new
|
247
|
+
date_format = options.delete(:date_format) || :default
|
248
|
+
delta_minutes = (start_date.to_i - time.to_i).floor / 60
|
249
|
+
if delta_minutes.abs <= (8724*60) # eight weeks? I'm lazy to count days for longer than that
|
250
|
+
distance = Utils.distance_of_time_in_words(delta_minutes);
|
251
|
+
if delta_minutes < 0
|
252
|
+
_("%{d} from now") % {:d => distance}
|
253
|
+
else
|
254
|
+
_("%{d} ago") % {:d => distance}
|
255
|
+
end
|
256
|
+
else
|
257
|
+
return _("on %{date}") % {:date => system_date.to_formatted_s(date_format)}
|
258
|
+
end
|
259
|
+
end
|
260
|
+
# Translates a number of minutes into verbal distances.
|
261
|
+
# e.g. 0.5 => less than a minute
|
262
|
+
# 70 => about one hour
|
263
|
+
def Utils.distance_of_time_in_words(minutes)
|
264
|
+
case
|
265
|
+
when minutes < 1
|
266
|
+
_("less than a minute")
|
267
|
+
when minutes < 50
|
268
|
+
_("%{m} minutes") % {:m => minutes}
|
269
|
+
when minutes < 90
|
270
|
+
_("about one hour")
|
271
|
+
when minutes < 1080
|
272
|
+
_("%{m} hours") % {:m => (minutes / 60).round}
|
273
|
+
when minutes < 1440
|
274
|
+
_("one day")
|
275
|
+
when minutes < 2880
|
276
|
+
_("about one day")
|
277
|
+
else
|
278
|
+
_("%{m} days") % {:m => (minutes / 1440).round}
|
279
|
+
end
|
280
|
+
end
|
281
|
+
|
282
|
+
|
283
|
+
# Execute an external program, returning a String obtained by redirecting
|
284
|
+
# the program's standards errors and output
|
285
|
+
#
|
286
|
+
def Utils.safe_exec(command, *args)
|
287
|
+
IO.popen("-") { |p|
|
288
|
+
if p
|
289
|
+
return p.readlines.join("\n")
|
290
|
+
else
|
291
|
+
begin
|
292
|
+
$stderr.reopen($stdout)
|
293
|
+
exec(command, *args)
|
294
|
+
rescue Exception => e
|
295
|
+
puts "exec of #{command} led to exception: #{e.pretty_inspect}"
|
296
|
+
Kernel::exit! 0
|
297
|
+
end
|
298
|
+
puts "exec of #{command} failed"
|
299
|
+
Kernel::exit! 0
|
300
|
+
end
|
301
|
+
}
|
302
|
+
end
|
303
|
+
|
304
|
+
|
305
|
+
# Safely (atomically) save to _file_, by passing a tempfile to the block
|
306
|
+
# and then moving the tempfile to its final location when done.
|
307
|
+
#
|
308
|
+
# call-seq: Utils.safe_save(file, &block)
|
309
|
+
#
|
310
|
+
def Utils.safe_save(file)
|
311
|
+
raise 'No safe save directory defined!' if @@safe_save_dir.nil?
|
312
|
+
basename = File.basename(file)
|
313
|
+
temp = Tempfile.new(basename,@@safe_save_dir)
|
314
|
+
temp.binmode
|
315
|
+
yield temp if block_given?
|
316
|
+
temp.close
|
317
|
+
File.rename(temp.path, file)
|
318
|
+
end
|
319
|
+
|
320
|
+
|
321
|
+
# Decode HTML entities in the String _str_, using HTMLEntities if the
|
322
|
+
# package was found, or UNESCAPE_TABLE otherwise.
|
323
|
+
#
|
324
|
+
def Utils.decode_html_entities(str)
|
325
|
+
if defined? ::HTMLEntities
|
326
|
+
return HTMLEntities.decode_entities(str)
|
327
|
+
else
|
328
|
+
str.gsub(/(&(.+?);)/) {
|
329
|
+
symbol = $2
|
330
|
+
# remove the 0-paddng from unicode integers
|
331
|
+
if symbol =~ /^#(\d+)$/
|
332
|
+
symbol = $1.to_i.to_s
|
333
|
+
end
|
334
|
+
|
335
|
+
# output the symbol's irc-translated character, or a * if it's unknown
|
336
|
+
UNESCAPE_TABLE[symbol] || (symbol.match(/^\d+$/) ? [symbol.to_i].pack("U") : '*')
|
337
|
+
}
|
338
|
+
end
|
339
|
+
end
|
340
|
+
|
341
|
+
# Try to grab and IRCify the first HTML par (<p> tag) in the given string.
|
342
|
+
# If possible, grab the one after the first heading
|
343
|
+
#
|
344
|
+
# It is possible to pass some options to determine how the stripping
|
345
|
+
# occurs. Currently supported options are
|
346
|
+
# strip:: Regex or String to strip at the beginning of the obtained
|
347
|
+
# text
|
348
|
+
# min_spaces:: minimum number of spaces a paragraph should have
|
349
|
+
#
|
350
|
+
def Utils.ircify_first_html_par(xml_org, opts={})
|
351
|
+
if defined? ::Hpricot
|
352
|
+
Utils.ircify_first_html_par_wh(xml_org, opts)
|
353
|
+
else
|
354
|
+
Utils.ircify_first_html_par_woh(xml_org, opts)
|
355
|
+
end
|
356
|
+
end
|
357
|
+
|
358
|
+
# HTML first par grabber using hpricot
|
359
|
+
def Utils.ircify_first_html_par_wh(xml_org, opts={})
|
360
|
+
doc = Hpricot(xml_org)
|
361
|
+
|
362
|
+
# Strip styles and scripts
|
363
|
+
(doc/"style|script").remove
|
364
|
+
|
365
|
+
debug doc
|
366
|
+
|
367
|
+
strip = opts[:strip]
|
368
|
+
strip = Regexp.new(/^#{Regexp.escape(strip)}/) if strip.kind_of?(String)
|
369
|
+
|
370
|
+
min_spaces = opts[:min_spaces] || 8
|
371
|
+
min_spaces = 0 if min_spaces < 0
|
372
|
+
|
373
|
+
txt = String.new
|
374
|
+
|
375
|
+
pre_h = pars = by_span = nil
|
376
|
+
|
377
|
+
while true
|
378
|
+
debug "Minimum number of spaces: #{min_spaces}"
|
379
|
+
|
380
|
+
# Initial attempt: <p> that follows <h\d>
|
381
|
+
if pre_h.nil?
|
382
|
+
pre_h = Hpricot::Elements[]
|
383
|
+
found_h = false
|
384
|
+
doc.search("*") { |e|
|
385
|
+
next if e.bogusetag?
|
386
|
+
case e.pathname
|
387
|
+
when /^h\d/
|
388
|
+
found_h = true
|
389
|
+
when 'p'
|
390
|
+
pre_h << e if found_h
|
391
|
+
end
|
392
|
+
}
|
393
|
+
debug "Hx: found: #{pre_h.pretty_inspect}"
|
394
|
+
end
|
395
|
+
|
396
|
+
pre_h.each { |p|
|
397
|
+
debug p
|
398
|
+
txt = p.to_html.ircify_html
|
399
|
+
txt.sub!(strip, '') if strip
|
400
|
+
debug "(Hx attempt) #{txt.inspect} has #{txt.count(" ")} spaces"
|
401
|
+
break unless txt.empty? or txt.count(" ") < min_spaces
|
402
|
+
}
|
403
|
+
|
404
|
+
return txt unless txt.empty? or txt.count(" ") < min_spaces
|
405
|
+
|
406
|
+
# Second natural attempt: just get any <p>
|
407
|
+
pars = doc/"p" if pars.nil?
|
408
|
+
debug "par: found: #{pars.pretty_inspect}"
|
409
|
+
pars.each { |p|
|
410
|
+
debug p
|
411
|
+
txt = p.to_html.ircify_html
|
412
|
+
txt.sub!(strip, '') if strip
|
413
|
+
debug "(par attempt) #{txt.inspect} has #{txt.count(" ")} spaces"
|
414
|
+
break unless txt.empty? or txt.count(" ") < min_spaces
|
415
|
+
}
|
416
|
+
|
417
|
+
return txt unless txt.empty? or txt.count(" ") < min_spaces
|
418
|
+
|
419
|
+
# Nothing yet ... let's get drastic: we look for non-par elements too,
|
420
|
+
# but only for those that match something that we know is likely to
|
421
|
+
# contain text
|
422
|
+
|
423
|
+
# Some blogging and forum platforms use spans or divs with a 'body' or
|
424
|
+
# 'message' or 'text' in their class to mark actual text. Since we want
|
425
|
+
# the class match to be partial and case insensitive, we collect
|
426
|
+
# the common elements that may have this class and then filter out those
|
427
|
+
# we don't need. If no divs or spans are found, we'll accept additional
|
428
|
+
# elements too (td, tr, tbody, table).
|
429
|
+
if by_span.nil?
|
430
|
+
by_span = Hpricot::Elements[]
|
431
|
+
extra = Hpricot::Elements[]
|
432
|
+
doc.search("*") { |el|
|
433
|
+
next if el.bogusetag?
|
434
|
+
case el.pathname
|
435
|
+
when AFTER_PAR_PATH
|
436
|
+
by_span.push el if el[:class] =~ AFTER_PAR_CLASS or el[:id] =~ AFTER_PAR_CLASS
|
437
|
+
when AFTER_PAR_EX
|
438
|
+
extra.push el if el[:class] =~ AFTER_PAR_CLASS or el[:id] =~ AFTER_PAR_CLASS
|
439
|
+
end
|
440
|
+
}
|
441
|
+
if by_span.empty? and not extra.empty?
|
442
|
+
by_span.concat extra
|
443
|
+
end
|
444
|
+
debug "other \#1: found: #{by_span.pretty_inspect}"
|
445
|
+
end
|
446
|
+
|
447
|
+
by_span.each { |p|
|
448
|
+
debug p
|
449
|
+
txt = p.to_html.ircify_html
|
450
|
+
txt.sub!(strip, '') if strip
|
451
|
+
debug "(other attempt \#1) #{txt.inspect} has #{txt.count(" ")} spaces"
|
452
|
+
break unless txt.empty? or txt.count(" ") < min_spaces
|
453
|
+
}
|
454
|
+
|
455
|
+
return txt unless txt.empty? or txt.count(" ") < min_spaces
|
456
|
+
|
457
|
+
# At worst, we can try stuff which is comprised between two <br>
|
458
|
+
# TODO
|
459
|
+
|
460
|
+
debug "Last candidate #{txt.inspect} has #{txt.count(" ")} spaces"
|
461
|
+
return txt unless txt.count(" ") < min_spaces
|
462
|
+
break if min_spaces == 0
|
463
|
+
min_spaces /= 2
|
464
|
+
end
|
465
|
+
end
|
466
|
+
|
467
|
+
# HTML first par grabber without hpricot
|
468
|
+
def Utils.ircify_first_html_par_woh(xml_org, opts={})
|
469
|
+
xml = xml_org.gsub(/<!--.*?-->/m, '').gsub(/<script(?:\s+[^>]*)?>.*?<\/script>/im, "").gsub(/<style(?:\s+[^>]*)?>.*?<\/style>/im, "")
|
470
|
+
|
471
|
+
strip = opts[:strip]
|
472
|
+
strip = Regexp.new(/^#{Regexp.escape(strip)}/) if strip.kind_of?(String)
|
473
|
+
|
474
|
+
min_spaces = opts[:min_spaces] || 8
|
475
|
+
min_spaces = 0 if min_spaces < 0
|
476
|
+
|
477
|
+
txt = String.new
|
478
|
+
|
479
|
+
while true
|
480
|
+
debug "Minimum number of spaces: #{min_spaces}"
|
481
|
+
header_found = xml.match(HX_REGEX)
|
482
|
+
if header_found
|
483
|
+
header_found = $'
|
484
|
+
while txt.empty? or txt.count(" ") < min_spaces
|
485
|
+
candidate = header_found[PAR_REGEX]
|
486
|
+
break unless candidate
|
487
|
+
txt = candidate.ircify_html
|
488
|
+
header_found = $'
|
489
|
+
txt.sub!(strip, '') if strip
|
490
|
+
debug "(Hx attempt) #{txt.inspect} has #{txt.count(" ")} spaces"
|
491
|
+
end
|
492
|
+
end
|
493
|
+
|
494
|
+
return txt unless txt.empty? or txt.count(" ") < min_spaces
|
495
|
+
|
496
|
+
# If we haven't found a first par yet, try to get it from the whole
|
497
|
+
# document
|
498
|
+
header_found = xml
|
499
|
+
while txt.empty? or txt.count(" ") < min_spaces
|
500
|
+
candidate = header_found[PAR_REGEX]
|
501
|
+
break unless candidate
|
502
|
+
txt = candidate.ircify_html
|
503
|
+
header_found = $'
|
504
|
+
txt.sub!(strip, '') if strip
|
505
|
+
debug "(par attempt) #{txt.inspect} has #{txt.count(" ")} spaces"
|
506
|
+
end
|
507
|
+
|
508
|
+
return txt unless txt.empty? or txt.count(" ") < min_spaces
|
509
|
+
|
510
|
+
# Nothing yet ... let's get drastic: we look for non-par elements too,
|
511
|
+
# but only for those that match something that we know is likely to
|
512
|
+
# contain text
|
513
|
+
|
514
|
+
# Attempt #1
|
515
|
+
header_found = xml
|
516
|
+
while txt.empty? or txt.count(" ") < min_spaces
|
517
|
+
candidate = header_found[AFTER_PAR1_REGEX]
|
518
|
+
break unless candidate
|
519
|
+
txt = candidate.ircify_html
|
520
|
+
header_found = $'
|
521
|
+
txt.sub!(strip, '') if strip
|
522
|
+
debug "(other attempt \#1) #{txt.inspect} has #{txt.count(" ")} spaces"
|
523
|
+
end
|
524
|
+
|
525
|
+
return txt unless txt.empty? or txt.count(" ") < min_spaces
|
526
|
+
|
527
|
+
# Attempt #2
|
528
|
+
header_found = xml
|
529
|
+
while txt.empty? or txt.count(" ") < min_spaces
|
530
|
+
candidate = header_found[AFTER_PAR2_REGEX]
|
531
|
+
break unless candidate
|
532
|
+
txt = candidate.ircify_html
|
533
|
+
header_found = $'
|
534
|
+
txt.sub!(strip, '') if strip
|
535
|
+
debug "(other attempt \#2) #{txt.inspect} has #{txt.count(" ")} spaces"
|
536
|
+
end
|
537
|
+
|
538
|
+
debug "Last candidate #{txt.inspect} has #{txt.count(" ")} spaces"
|
539
|
+
return txt unless txt.count(" ") < min_spaces
|
540
|
+
break if min_spaces == 0
|
541
|
+
min_spaces /= 2
|
542
|
+
end
|
543
|
+
end
|
544
|
+
|
545
|
+
# This method extracts title, content (first par) and extra
|
546
|
+
# information from the given document _doc_.
|
547
|
+
#
|
548
|
+
# _doc_ can be an URI, a Net::HTTPResponse or a String.
|
549
|
+
#
|
550
|
+
# If _doc_ is a String, only title and content information
|
551
|
+
# are retrieved (if possible), using standard methods.
|
552
|
+
#
|
553
|
+
# If _doc_ is an URI or a Net::HTTPResponse, additional
|
554
|
+
# information is retrieved, and special title/summary
|
555
|
+
# extraction routines are used if possible.
|
556
|
+
#
|
557
|
+
def Utils.get_html_info(doc, opts={})
|
558
|
+
case doc
|
559
|
+
when String
|
560
|
+
Utils.get_string_html_info(doc, opts)
|
561
|
+
when Net::HTTPResponse
|
562
|
+
Utils.get_resp_html_info(doc, opts)
|
563
|
+
when URI
|
564
|
+
ret = DataStream.new
|
565
|
+
@@bot.httputil.get_response(doc) { |resp|
|
566
|
+
ret.replace Utils.get_resp_html_info(resp, opts)
|
567
|
+
}
|
568
|
+
return ret
|
569
|
+
else
|
570
|
+
raise
|
571
|
+
end
|
572
|
+
end
|
573
|
+
|
574
|
+
class ::UrlLinkError < RuntimeError
|
575
|
+
end
|
576
|
+
|
577
|
+
# This method extracts title, content (first par) and extra
|
578
|
+
# information from the given Net::HTTPResponse _resp_.
|
579
|
+
#
|
580
|
+
# Currently, the only accepted options (in _opts_) are
|
581
|
+
# uri_fragment:: the URI fragment of the original request
|
582
|
+
# full_body:: get the whole body instead of
|
583
|
+
# @@bot.config['http.info_bytes'] bytes only
|
584
|
+
#
|
585
|
+
# Returns a DataStream with the following keys:
|
586
|
+
# text:: the (partial) body
|
587
|
+
# title:: the title of the document (if any)
|
588
|
+
# content:: the first paragraph of the document (if any)
|
589
|
+
# headers::
|
590
|
+
# the headers of the Net::HTTPResponse. The value is
|
591
|
+
# a Hash whose keys are lowercase forms of the HTTP
|
592
|
+
# header fields, and whose values are Arrays.
|
593
|
+
#
|
594
|
+
def Utils.get_resp_html_info(resp, opts={})
|
595
|
+
case resp
|
596
|
+
when Net::HTTPSuccess
|
597
|
+
loc = URI.parse(resp['x-rbot-location'] || resp['location']) rescue nil
|
598
|
+
if loc and loc.fragment and not loc.fragment.empty?
|
599
|
+
opts[:uri_fragment] ||= loc.fragment
|
600
|
+
end
|
601
|
+
ret = DataStream.new(opts.dup)
|
602
|
+
ret[:headers] = resp.to_hash
|
603
|
+
ret[:text] = partial = opts[:full_body] ? resp.body : resp.partial_body(@@bot.config['http.info_bytes'])
|
604
|
+
|
605
|
+
filtered = Utils.try_htmlinfo_filters(ret)
|
606
|
+
|
607
|
+
if filtered
|
608
|
+
return filtered
|
609
|
+
elsif resp['content-type'] =~ /^text\/|(?:x|ht)ml/
|
610
|
+
ret.merge!(Utils.get_string_html_info(partial, opts))
|
611
|
+
end
|
612
|
+
return ret
|
613
|
+
else
|
614
|
+
raise UrlLinkError, "getting link (#{resp.code} - #{resp.message})"
|
615
|
+
end
|
616
|
+
end
|
617
|
+
|
618
|
+
# This method runs an appropriately-crafted DataStream _ds_ through the
|
619
|
+
# filters in the :htmlinfo filter group, in order. If one of the filters
|
620
|
+
# returns non-nil, its results are merged in _ds_ and returned. Otherwise
|
621
|
+
# nil is returned.
|
622
|
+
#
|
623
|
+
# The input DataStream shuold have the downloaded HTML as primary key
|
624
|
+
# (:text) and possibly a :headers key holding the resonse headers.
|
625
|
+
#
|
626
|
+
def Utils.try_htmlinfo_filters(ds)
|
627
|
+
filters = @@bot.filter_names(:htmlinfo)
|
628
|
+
return nil if filters.empty?
|
629
|
+
cur = nil
|
630
|
+
# TODO filter priority
|
631
|
+
filters.each { |n|
|
632
|
+
debug "testing filter #{n}"
|
633
|
+
cur = @@bot.filter(@@bot.global_filter_name(n, :htmlinfo), ds)
|
634
|
+
debug "returned #{cur.pretty_inspect}"
|
635
|
+
break if cur
|
636
|
+
}
|
637
|
+
return ds.merge(cur) if cur
|
638
|
+
end
|
639
|
+
|
640
|
+
# HTML info filters often need to check if the webpage location
|
641
|
+
# of a passed DataStream _ds_ matches a given Regexp.
|
642
|
+
def Utils.check_location(ds, rx)
|
643
|
+
debug ds[:headers]
|
644
|
+
if h = ds[:headers]
|
645
|
+
loc = [h['x-rbot-location'],h['location']].flatten.grep(rx)
|
646
|
+
end
|
647
|
+
loc ||= []
|
648
|
+
debug loc
|
649
|
+
return loc.empty? ? nil : loc
|
650
|
+
end
|
651
|
+
|
652
|
+
# This method extracts title and content (first par)
|
653
|
+
# from the given HTML or XML document _text_, using
|
654
|
+
# standard methods (String#ircify_html_title,
|
655
|
+
# Utils.ircify_first_html_par)
|
656
|
+
#
|
657
|
+
# Currently, the only accepted option (in _opts_) is
|
658
|
+
# uri_fragment:: the URI fragment of the original request
|
659
|
+
#
|
660
|
+
def Utils.get_string_html_info(text, opts={})
|
661
|
+
debug "getting string html info"
|
662
|
+
txt = text.dup
|
663
|
+
title = txt.ircify_html_title
|
664
|
+
debug opts
|
665
|
+
if frag = opts[:uri_fragment] and not frag.empty?
|
666
|
+
fragreg = /<a\s+(?:[^>]+\s+)?(?:name|id)=["']?#{frag}["']?[^>]*>/im
|
667
|
+
debug fragreg
|
668
|
+
debug txt
|
669
|
+
if txt.match(fragreg)
|
670
|
+
# grab the post-match
|
671
|
+
txt = $'
|
672
|
+
end
|
673
|
+
debug txt
|
674
|
+
end
|
675
|
+
c_opts = opts.dup
|
676
|
+
c_opts[:strip] ||= title
|
677
|
+
content = Utils.ircify_first_html_par(txt, c_opts)
|
678
|
+
content = nil if content.empty?
|
679
|
+
return {:title => title, :content => content}
|
680
|
+
end
|
681
|
+
|
682
|
+
# Get the first pars of the first _count_ _urls_.
|
683
|
+
# The pages are downloaded using the bot httputil service.
|
684
|
+
# Returns an array of the first paragraphs fetched.
|
685
|
+
# If (optional) _opts_ :message is specified, those paragraphs are
|
686
|
+
# echoed as replies to the IRC message passed as _opts_ :message
|
687
|
+
#
|
688
|
+
def Utils.get_first_pars(urls, count, opts={})
|
689
|
+
idx = 0
|
690
|
+
msg = opts[:message]
|
691
|
+
retval = Array.new
|
692
|
+
while count > 0 and urls.length > 0
|
693
|
+
url = urls.shift
|
694
|
+
idx += 1
|
695
|
+
|
696
|
+
begin
|
697
|
+
info = Utils.get_html_info(URI.parse(url), opts)
|
698
|
+
|
699
|
+
par = info[:content]
|
700
|
+
retval.push(par)
|
701
|
+
|
702
|
+
if par
|
703
|
+
msg.reply "[#{idx}] #{par}", :overlong => :truncate if msg
|
704
|
+
count -=1
|
705
|
+
end
|
706
|
+
rescue
|
707
|
+
debug "Unable to retrieve #{url}: #{$!}"
|
708
|
+
next
|
709
|
+
end
|
710
|
+
end
|
711
|
+
return retval
|
712
|
+
end
|
713
|
+
|
714
|
+
end
|
715
|
+
end
|
716
|
+
|
717
|
+
Irc::Utils.bot = Irc::Bot::Plugins.manager.bot
|