]> git.netwichtig.de Git - user/henk/code/ruby/rbot.git/blobdiff - data/rbot/plugins/keywords.rb
New unreplied() method for plugins that want to handle PRIVMSGs unreplied by any...
[user/henk/code/ruby/rbot.git] / data / rbot / plugins / keywords.rb
index 3ae760241e778a8d1db90a8b8735fc1bdb98cd76..855bfe6aa4e25bb4472ed839be7621a923fde5b1 100644 (file)
@@ -9,10 +9,10 @@ class Keyword
 
   # type of keyword (e.g. "is" or "are")
   attr_reader :type
-  
+
   # type::   type of keyword (e.g "is" or "are")
   # values:: array of values
-  # 
+  #
   # create a keyword of type +type+ with values +values+
   def initialize(type, values)
     @type = type.downcase
@@ -71,21 +71,24 @@ class Keyword
   end
 end
 
-# keyword plugin class. 
+# keywords class.
 #
 # Handles all that stuff like "bot: foo is bar", "bot: foo?"
 #
 # Fallback after core and auth have had a look at a message and refused to
 # handle it, checks for a keyword command or lookup, otherwise the message
 # is delegated to plugins
-class KeywordPlugin < Plugin
+class Keywords < Plugin
   BotConfig.register BotConfigBooleanValue.new('keyword.listen',
     :default => false,
     :desc => "Should the bot listen to all chat and attempt to automatically detect keywords? (e.g. by spotting someone say 'foo is bar')")
   BotConfig.register BotConfigBooleanValue.new('keyword.address',
     :default => true,
     :desc => "Should the bot require that keyword lookups are addressed to it? If not, the bot will attempt to lookup foo if someone says 'foo?' in channel")
-  
+  BotConfig.register BotConfigIntegerValue.new('keyword.search_results',
+    :default => 3,
+    :desc => "How many search results to display at a time")
+
   # create a new KeywordPlugin instance, associated to bot +bot+
   def initialize
     super
@@ -95,7 +98,7 @@ class KeywordPlugin < Plugin
     upgrade_data
 
     scan
-    
+
     # import old format keywords into DBHash
     if(File.exist?("#{@bot.botclass}/keywords.rbot"))
       log "auto importing old keywords.rbot"
@@ -113,7 +116,7 @@ class KeywordPlugin < Plugin
       File.rename("#{@bot.botclass}/keywords.rbot", "#{@bot.botclass}/keywords.rbot.old")
     end
   end
-  
+
   # drop static keywords and reload them from files, picking up any new
   # keyword files that have been added
   def rescan
@@ -129,9 +132,9 @@ class KeywordPlugin < Plugin
       next unless f =~ /\.db$/
       log "upgrading keyword db #{f} (rbot 0.9.5 or prior) database format"
       newname = f.gsub(/\.db$/, ".kdb")
-      old = BDB::Hash.open f, nil, 
+      old = BDB::Hash.open f, nil,
                            "r+", 0600
-      new = BDB::CIBtree.open(newname, nil, 
+      new = BDB::CIBtree.open(newname, nil,
                               BDB::CREATE | BDB::EXCL,
                               0600)
       old.each {|k,v|
@@ -141,7 +144,7 @@ class KeywordPlugin < Plugin
       new.close
       File.delete(f)
     }
-    
+
     # then scan for current DBTree files, and load them
     Dir["#{@bot.botclass}/keywords/*"].each {|f|
       next unless f =~ /\.kdb$/
@@ -150,7 +153,7 @@ class KeywordPlugin < Plugin
       debug "keywords module: loading DBTree file #{f}, key #{key}"
       @statickeywords[key] = hsh
     }
-    
+
     # then scan for non DB files, and convert/import them and delete
     Dir["#{@bot.botclass}/keywords/*"].each {|f|
       next if f =~ /\.kdb$/
@@ -182,7 +185,7 @@ class KeywordPlugin < Plugin
   def upgrade_data
     if File.exist?("#{@bot.botclass}/keywords.db")
       log "upgrading old keywords (rbot 0.9.5 or prior) database format"
-      old = BDB::Hash.open "#{@bot.botclass}/keywords.db", nil, 
+      old = BDB::Hash.open "#{@bot.botclass}/keywords.db", nil,
                            "r+", 0600
       old.each {|k,v|
         @keywords[k] = v
@@ -191,10 +194,10 @@ class KeywordPlugin < Plugin
       @keywords.flush
       File.rename("#{@bot.botclass}/keywords.db", "#{@bot.botclass}/keywords.db.old")
     end
-  
+
     if File.exist?("#{@bot.botclass}/keyword.db")
       log "upgrading old keywords (rbot 0.9.9 or prior) database format"
-      old = BDB::CIBtree.open "#{@bot.botclass}/keyword.db", nil, 
+      old = BDB::CIBtree.open "#{@bot.botclass}/keyword.db", nil,
                            "r+", 0600
       old.each {|k,v|
         @keywords[k] = v
@@ -209,6 +212,7 @@ class KeywordPlugin < Plugin
   def save
     @keywords.flush
   end
+
   def oldsave
     File.open("#{@bot.botclass}/keywords.rbot", "w") do |file|
       @keywords.each do |key, value|
@@ -216,7 +220,7 @@ class KeywordPlugin < Plugin
       end
     end
   end
-  
+
   # lookup keyword +key+, return it or nil
   def [](key)
     return nil if key.nil?
@@ -250,39 +254,45 @@ class KeywordPlugin < Plugin
 
   # m::     PrivMessage containing message info
   # key::   key being queried
-  # dunno:: optional, if true, reply "dunno" if +key+ not found
-  # 
+  # quiet:: optional, if false, complain if +key+ is not found
+  #
   # handle a message asking about a keyword
-  def keyword(m, key, dunno=true)
+  def keyword_lookup(m, key, quiet = false)
     return if key.nil?
-     unless(kw = self[key])
-       m.reply @bot.lang.get("dunno") if (dunno)
-       return
-     end
-     response = kw.to_s
-     response.gsub!(/<who>/, m.sourcenick)
-     if(response =~ /^<reply>\s*(.*)/)
-       m.reply "#$1"
-     elsif(response =~ /^<action>\s*(.*)/)
-       @bot.action m.replyto, "#$1"
-     elsif(m.public? && response =~ /^<topic>\s*(.*)/)
-       topic = $1
-       @bot.topic m.target, topic
-     else
-       m.reply "#{key} #{kw.type} #{response}"
-     end
+    unless(kw = self[key])
+      m.reply "sorry, I don't know about \"#{key}\"" unless quiet
+      return
+    end
+
+    response = kw.to_s
+    response.gsub!(/<who>/, m.sourcenick)
+
+    if(response =~ /^<reply>\s*(.*)/)
+      m.reply $1
+    elsif(response =~ /^<action>\s*(.*)/)
+      m.act $1
+    elsif(m.public? && response =~ /^<topic>\s*(.*)/)
+      @bot.topic m.target, $1
+    else
+      m.reply "#{key} #{kw.type} #{response}"
+    end
   end
 
-  
+
   # handle a message which alters a keyword
-  # like "foo is bar", or "no, foo is baz", or "foo is also qux"
-  def keyword_command(sourcenick, target, lhs, mhs, rhs, quiet=false)
+  # like "foo is bar" or "foo is also qux"
+  def keyword_command(m, lhs, mhs, rhs, quiet = false)
     debug "got keyword command #{lhs}, #{mhs}, #{rhs}"
+    return if lhs.strip.empty?
+
     overwrite = false
     overwrite = true if(lhs.gsub!(/^no,\s*/, ""))
+    also = false
     also = true if(rhs.gsub!(/^also\s+/, ""))
+
     values = rhs.split(/\s+\|\s+/)
     lhs = Keyword.unescape lhs
+
     if(overwrite || also || !has_key?(lhs))
       if(also && has_key?(lhs))
         kw = self[lhs]
@@ -291,67 +301,65 @@ class KeywordPlugin < Plugin
       else
         @keywords[lhs] = Keyword.new(mhs, values).dump
       end
-      @bot.okay target if !quiet
+      m.okay if !quiet
     elsif(has_key?(lhs))
       kw = self[lhs]
-      @bot.say target, "but #{lhs} #{kw.type} #{kw.desc}" if kw && !quiet
+      m.reply "but #{lhs} #{kw.type} #{kw.desc}" if kw && !quiet
     end
   end
 
   # return help string for Keywords with option topic +topic+
-  def help(plugin, topic="")
-    case topic
-      when "overview"
-        return "set: <keyword> is <definition>, overide: no, <keyword> is <definition>, add to definition: <keyword> is also <definition>, random responses: <keyword> is <definition> | <definition> [| ...], plurals: <keyword> are <definition>, escaping: \\is, \\are, \\|, specials: <reply>, <action>, <who>"
-      when "set"
-        return "set => <keyword> is <definition>"
-      when "plurals"
-        return "plurals => <keywords> are <definition>"
-      when "override"
-        return "overide => no, <keyword> is <definition>"
-      when "also"
-        return "also => <keyword> is also <definition>"
-      when "random"
-        return "random responses => <keyword> is <definition> | <definition> [| ...]"
-      when "get"
-        return "asking for keywords => (with addressing) \"<keyword>?\", (without addressing) \"'<keyword>\""
-      when "tell"
-        return "tell <nick> about <keyword> => if <keyword> is known, tell <nick>, via /msg, its definition"
-      when "forget"
-        return "forget <keyword> => forget fact <keyword>"
-      when "keywords"
-        return "keywords => show current keyword counts"
-      when "<reply>"
-        return "<reply> => normal response is \"<keyword> is <definition>\", but if <definition> begins with <reply>, the response will be \"<definition>\""
-      when "<action>"
-        return "<action> => makes keyword respnse \"/me <definition>\""
-      when "<who>"
-        return "<who> => replaced with questioner in reply"
-      when "<topic>"
-        return "<topic> => respond by setting the topic to the rest of the definition"
-      when "search"
-        return "keyword search [--all] [--full] <regexp> => search keywords for <regexp>. If --all is set, search static keywords too, if --full is set, search definitions too."
+  def help(plugin, topic = '')
+    case plugin
+    when /keyword/
+      case topic
+      when 'lookup'
+        'keyword [lookup] <keyword> => look up the definition for a keyword; writing "lookup" is optional'
+      when 'set'
+        'keyword set <keyword> is/are <definition> => define a keyword, definition can contain "|" to separate multiple randomly chosen replies'
+      when 'forget'
+        'keyword forget <keyword> => forget a keyword'
+      when 'tell'
+        'keyword tell <nick> about <keyword> => tell somebody about a keyword'
+      when 'search'
+        'keyword search [--all] [--full] <pattern> => search keywords for <pattern>, which can be a regular expression. If --all is set, search static keywords too, if --full is set, search definitions too.'
+      when 'listen'
+        'when the config option "keyword.listen" is set to false, rbot will try to extract keyword definitions from regular channel messages'
+      when 'address'
+        'when the config option "keyword.address" is set to true, rbot will try to answer channel questions of the form "<keyword>?"'
+      when '<reply>'
+        '<reply> => normal response is "<keyword> is <definition>", but if <definition> begins with <reply>, the response will be "<definition>"'
+      when '<action>'
+        '<action> => makes keyword respond with "/me <definition>"'
+      when '<who>'
+        '<who> => replaced with questioner in reply'
+      when '<topic>'
+        '<topic> => respond by setting the topic to the rest of the definition'
       else
-        return "Keyword module (Fact learning and regurgitation) topics: overview, set, plurals, override, also, random, get, tell, forget, keywords, keywords search, <reply>, <action>, <who>, <topic>"
+        'keyword module (fact learning and regurgitation) topics: lookup, set, forget, tell, search, listen, address, <reply>, <action>, <who>, <topic>'
+      end
+    when "forget"
+      'forget <keyword> => forget a keyword'
+    when "tell"
+      'tell <nick> about <keyword> => tell somebody about a keyword'
+    when "learn"
+      'learn that <keyword> is/are <definition> => define a keyword, definition can contain "|" to separate multiple randomly chosen replies'
+    else
+      'keyword module (fact learning and regurgitation) topics: lookup, set, forget, tell, search, listen, address, <reply>, <action>, <who>, <topic>'
     end
   end
 
   # handle a message asking the bot to tell someone about a keyword
-  def keyword_tell(m, param)
-    target = param[:target]
-    key = nil
-
-    # extract the keyword from the message, because unfortunately
-    # the message mapper doesn't preserve whtiespace
-    if m.message =~ /about\s+(.+)$/
-      key = $1
-    end
-
+  def keyword_tell(m, target, key)
     unless(kw = self[key])
       m.reply @bot.lang.get("dunno_about_X") % key
       return
     end
-    
+    if target == @bot.nick
+      m.reply "very funny, trying to make me tell something to myself"
+      return
+    end
+
     response = kw.to_s
     response.gsub!(/<who>/, m.sourcenick)
     if(response =~ /^<reply>\s*(.*)/)
@@ -367,7 +375,7 @@ class KeywordPlugin < Plugin
   end
 
   # return the number of known keywords
-  def keyword_stats(m, param)
+  def keyword_stats(m)
     length = 0
     @statickeywords.each {|k,v|
       length += v.length
@@ -376,56 +384,59 @@ class KeywordPlugin < Plugin
   end
 
   # search for keywords, optionally also the definition and the static keywords
-  def keyword_search(m, param)
-    str = param[:pattern]
-    all = (param[:all] == '--all')
-    full = (param[:full] == '--full')
-    
+  def keyword_search(m, key, full = false, all = false, from = 1)
     begin
-      re = Regexp.new(str, Regexp::IGNORECASE)
-      if(@bot.auth.allow?("keyword", m.source, m.replyto))
-        matches = Array.new
-        @keywords.each {|k,v|
-          kw = Keyword.restore(v)
-          if re.match(k) || (full && re.match(kw.desc))
-            matches << [k,kw]
-          end
-        }
-        if all
-          @statickeywords.each {|k,v|
-            v.each {|kk,vv|
-              kw = Keyword.restore(vv)
-              if re.match(kk) || (full && re.match(kw.desc))
-                matches << [kk,kw]
-              end
-            }
-          }
+      if key =~ /^\/(.+)\/$/
+        re = Regexp.new($1, Regexp::IGNORECASE)
+      else
+        re = Regexp.new(Regexp.escape(key), Regexp::IGNORECASE)
+      end
+
+      matches = Array.new
+      @keywords.each {|k,v|
+        kw = Keyword.restore(v)
+        if re.match(k) || (full && re.match(kw.desc))
+          matches << [k,kw]
         end
-        if matches.length == 1
-          rkw = matches[0]
-          m.reply "#{rkw[0]} #{rkw[1].type} #{rkw[1].desc}"
-        elsif matches.length > 0
-          i = 0
-          matches.each {|rkw|
-            m.reply "[#{i+1}/#{matches.length}] #{rkw[0]} #{rkw[1].type} #{rkw[1].desc}"
-            i += 1
-            break if i == 3
+      }
+      if all
+        @statickeywords.each {|k,v|
+          v.each {|kk,vv|
+            kw = Keyword.restore(vv)
+            if re.match(kk) || (full && re.match(kw.desc))
+              matches << [kk,kw]
+            end
           }
-        else
-          m.reply "no keywords match #{str}"
+        }
+      end
+
+      if matches.length == 1
+        rkw = matches[0]
+        m.reply "#{rkw[0]} #{rkw[1].type} #{rkw[1].desc}"
+      elsif matches.length > 0
+        if from > matches.length
+          m.reply "#{matches.length} found, can't tell you about #{from}"
+          return
         end
+        i = 1
+        matches.each {|rkw|
+          m.reply "[#{i}/#{matches.length}] #{rkw[0]} #{rkw[1].type} #{rkw[1].desc}" if i >= from
+          i += 1
+          break if i == from+@bot.config['keyword.search_results']
+        }
+      else
+        m.reply "no keywords match #{key}"
       end
     rescue RegexpError => e
-      m.reply "no keywords match #{str}: #{e}"
+      m.reply "no keywords match #{key}: #{e}"
     rescue
       debug e.inspect
-      m.reply "no keywords match #{str}: an error occurred"
+      m.reply "no keywords match #{key}: an error occurred"
     end
   end
 
   # forget one of the dynamic keywords
-  def keyword_forget(m, param)
-    key = param[:key]
+  def keyword_forget(m, key)
     if(@keywords.has_key?(key))
       @keywords.delete(key)
       @bot.okay m.replyto
@@ -433,38 +444,68 @@ class KeywordPlugin < Plugin
   end
 
   # privmsg handler
-  def listen(m)
-    return if m.replied?
-    if(m.address?)
-      if(!(m.message =~ /\\\?\s*$/) && m.message =~ /^(.*\S)\s*\?\s*$/)
-        keyword m, $1 if(@bot.auth.allow?("keyword", m.source, m.replyto))
-      elsif(m.message =~ /^(.*?)\s+(is|are)\s+(.*)$/)
-        keyword_command(m.sourcenick, m.replyto, $1, $2, $3) if(@bot.auth.allow?("keycmd", m.source, m.replyto))
+  def privmsg(m)
+    case m.plugin
+    when "keyword"
+      case m.params
+      when /^set\s+(.+?)\s+(is|are)\s+(.+)$/
+        keyword_command(m, $1, $2, $3) if @bot.auth.allow?('keycmd', m.source, m.replyto)
+      when /^forget\s+(.+)$/
+        keyword_forget(m, $1) if @bot.auth.allow?('keycmd', m.source, m.replyto)
+      when /^lookup\s+(.+)$/
+        keyword_lookup(m, $1) if @bot.auth.allow?('keyword', m.source, m.replyto)
+      when /^stats\s*$/
+        keyword_stats(m) if @bot.auth.allow?('keyword', m.source, m.replyto)
+      when /^search\s+(.+)$/
+        key = $1
+        full = key.sub!('--full ', '')
+        all = key.sub!('--all ', '')
+        if key.sub!(/--from (\d+) /, '')
+          from = $1.to_i
+        else
+          from = 1
+        end
+        from = 1 unless from > 0
+        keyword_search(m, key, full, all, from) if @bot.auth.allow?('keyword', m.source, m.replyto)
+      when /^tell\s+(\S+)\s+about\s+(.+)$/
+        keyword_tell(m, $1, $2) if @bot.auth.allow?('keyword', m.source, m.replyto)
+      else
+        keyword_lookup(m, m.params) if @bot.auth.allow?('keyword', m.source, m.replyto)
       end
-    else
-      # in channel message, not to me
-      # TODO option to do if(m.message =~ /^(.*)$/, ie try any line as a
-      # keyword lookup.
-      if(m.message =~ /^'(.*)$/ || (!@bot.config["keyword.address"] && m.message =~ /^(.*\S)\s*\?\s*$/))
-        keyword m, $1, false if(@bot.auth.allow?("keyword", m.source))
-      elsif(@bot.config["keyword.listen"] == true && (m.message =~ /^(.*?)\s+(is|are)\s+(.*)$/))
-        # TODO MUCH more selective on what's allowed here
-        keyword_command(m.sourcenick, m.replyto, $1, $2, $3, true) if(@bot.auth.allow?("keycmd", m.source))
+    when "forget"
+      keyword_forget(m, m.params) if @bot.auth.allow?('keycmd', m.source, m.replyto)
+    when "tell"
+      if m.params =~ /(\S+)\s+about\s+(.+)$/
+        keyword_tell(m, $1, $2) if @bot.auth.allow?('keyword', m.source, m.replyto)
+      else
+        m.reply "wrong 'tell' syntax"
+      end
+    when "learn"
+      if m.params =~ /^that\s+(.+?)\s+(is|are)\s+(.+)$/
+        keyword_command(m, $1, $2, $3) if @bot.auth.allow?('keycmd', m.source, m.replyto)
+      else
+        m.reply "wrong 'learn' syntax"
       end
     end
   end
-end
-
-plugin = KeywordPlugin.new
 
-plugin.map 'keyword stats', :action => 'keyword_stats'
+  def unreplied(m)
+    # return if m.address?
+    # in channel message, not to me
+    # TODO option to do if(m.message =~ /^(.*)$/, ie try any line as a
+    # keyword lookup.
+    if !@bot.config["keyword.address"] && m.message =~ /^(.*\S)\s*\?\s*$/
+      keyword_lookup m, $1, true if @bot.auth.allow?("keyword", m.source)
+    elsif @bot.config["keyword.listen"] && (m.message =~ /^(.*?)\s+(is|are)\s+(.*)$/)
+      # TODO MUCH more selective on what's allowed here
+      keyword_command m, $1, $2, $3, true if @bot.auth.allow?("keycmd", m.source)
+    end
+  end
+end
 
-plugin.map 'keyword search :all :full :pattern', :action => 'keyword_search',
-           :defaults => {:all => '', :full => ''},
-           :requirements => {:all => '--all', :full => '--full'}
-           
-plugin.map 'keyword forget :key', :action => 'keyword_forget'
-plugin.map 'forget :key', :action => 'keyword_forget', :auth => 'keycmd'
+plugin = Keywords.new
+plugin.register 'keyword'
+plugin.register 'forget'
+plugin.register 'tell'
+plugin.register 'learn'
 
-plugin.map 'keyword tell :target about *keyword', :action => 'keyword_tell'
-plugin.map 'tell :target about *keyword', :action => 'keyword_tell', :auth => 'keyword'