| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- # bash.org quote fetcher
- # Must .chanset #channel +bash
- #
- # Usage: !bash [optional search terms]
- # If search terms are not provided, fetch random quotes.
- #
- # Keeps fetched quotes in memory until displayed, including all results per
- # search term
- package require http
- package require htmlparse
- namespace eval ::bash {
- variable trigger !bash
- variable line_length 399
- variable max_lines 10
- variable useragent "Mozilla/5.0 (compatible; Y!J; for robot study; keyoshid)"
- variable output_cmd putserv
- setudef flag bash
- bind pub -|- $::bash::trigger ::bash::handler
- variable url http://bash.org/?
- variable list_regexp {<p class="quote">.*?<p class="qt">.*?</p>}
- variable quote_regexp {<p class="quote">.*?<b>#(.*?)</b>.*?class="qa".*?</a>\((.*)\)<a.*?<p class="qt">(.*?)</p>}
- variable random_quotes []
- variable search_quotes []
- }
- proc ::bash::quote_output {chan quote} {
- if {$quote == ""} {
- $::bash::output_cmd "PRIVMSG $chan :No result!"
- return
- }
- set number [dict get $quote number]
- set rating [dict get $quote rating]
- set quote [::htmlparse::mapEscapes [dict get $quote quote]]
- set quote [regsub -all -- {<br />} $quote ""]
- $::bash::output_cmd "PRIVMSG $chan :#\002${number}\002 (Rating: ${rating})"
- foreach line [split $quote \n] {
- if {[incr count] > $::bash::max_lines} {
- $::bash::output_cmd "PRIVMSG $chan :Output truncated. ${::bash::url}${number}"
- break
- }
- foreach subline [::bash::split_line $::bash::line_length $line] {
- $::bash::output_cmd "PRIVMSG $chan : $subline"
- }
- }
- }
- proc ::bash::handler {nick uhost hand chan argv} {
- if {![channel get $chan bash]} { return }
- if {$argv == ""} {
- if {[catch {::bash::random $chan} result]} {
- $::bash::output_cmd "PRIVMSG $chan :Error: $result"
- return
- }
- ::bash::quote_output $chan $result
- } else {
- if {[catch {::bash::search $argv $chan} result]} {
- $::bash::output_cmd "PRIVMSG $chan :Error: $result"
- return
- }
- ::bash::quote_output $chan $result
- }
- }
- proc ::bash::random {chan} {
- if {![llength $::bash::random_quotes]} {
- $::bash::output_cmd "PRIVMSG $chan :Fetching new random quotes..."
- set ::bash::random_quotes [::bash::fetch ${::bash::url}random1]
- }
- set quote [lindex $::bash::random_quotes 0]
- set ::bash::random_quotes [lreplace $::bash::random_quotes 0 0]
- return $quote
- }
- proc ::bash::search {query chan} {
- if {![dict exists $::bash::search_quotes $query]} {
- $::bash::output_cmd "PRIVMSG $chan :Fetching results..."
- set url ${::bash::url}[::http::formatQuery search $query sort 0 show 25]
- dict set ::bash::search_quotes $query [::bash::fetch $url]
- }
- set quotes [dict get $::bash::search_quotes $query]
- set quote [lindex $quotes 0]
- set quotes [lreplace $quotes 0 0]
- # Remove key if no more quotes after removal of one, else set quotes to remaining
- if {![llength $quotes]} {
- dict unset ::bash::search_quotes $query
- } else {
- dict set ::bash::search_quotes $query $quotes
- }
- return $quote
- }
- proc ::bash::fetch {url} {
- ::http::config -useragent $::bash::useragent
- set token [::http::geturl $url -timeout 10000]
- set data [::http::data $token]
- set ncode [::http::ncode $token]
- ::http::cleanup $token
- if {$ncode != 200} {
- error "HTTP fetch error $ncode: $data"
- }
- return [::bash::parse $data]
- }
- proc ::bash::parse {html} {
- set quotes []
- foreach raw_quote [regexp -all -inline -- $::bash::list_regexp $html] {
- if {![regexp $::bash::quote_regexp $raw_quote -> number rating quote]} {
- error "Parse error"
- }
- # Strip <font color=green|red> from rating
- regsub -all {<font.*?>} $rating {} rating
- regsub -all {</font>} $rating {} rating
- lappend quotes [list number $number rating $rating quote $quote]
- }
- return $quotes
- }
- # by fedex
- proc ::bash::split_line {max str} {
- set last [expr {[string length $str] -1}]
- set start 0
- set end [expr {$max -1}]
- set lines []
- while {$start <= $last} {
- if {$last >= $end} {
- set end [string last { } $str $end]
- }
- lappend lines [string trim [string range $str $start $end]]
- set start $end
- set end [expr {$start + $max}]
- }
- return $lines
- }
- putlog "bash.tcl loaded"
|