X-Git-Url: https://git.netwichtig.de/gitweb/?a=blobdiff_plain;f=lib%2Frbot%2Fmessage.rb;h=155f9038b34640c9a9e6a704cb6a9bb2c49ea2c1;hb=f287bf1e73829434d92b46c333c3185373198518;hp=ba0d8cc9762f0e851ae67abe9086b364c2c3d832;hpb=a41182b02b0b4f70e0a4e4ffff9c831207a87ba5;p=user%2Fhenk%2Fcode%2Fruby%2Frbot.git diff --git a/lib/rbot/message.rb b/lib/rbot/message.rb index ba0d8cc9..155f9038 100644 --- a/lib/rbot/message.rb +++ b/lib/rbot/message.rb @@ -22,17 +22,22 @@ module Irc :default => ':', :wizard => true, :desc => "when replying with nick put this character after the nick of the user the bot is replying to" ) + Config.register BooleanValue.new('core.private_replies', + :default => false, + :desc => 'Should the bot reply to private instead of the channel?' + ) end end - # Define standard IRC attriubtes (not so standard actually, + # Define standard IRC attributes (not so standard actually, # but the closest thing we have ...) Bold = "\002" Underline = "\037" Reverse = "\026" Italic = "\011" NormalText = "\017" + AttributeRx = /#{Bold}|#{Underline}|#{Reverse}|#{Italic}|#{NormalText}/ # Color is prefixed by \003 and followed by optional # foreground and background specifications, two-digits-max @@ -41,6 +46,8 @@ module Irc Color = "\003" ColorRx = /#{Color}\d?\d?(?:,\d\d?)?/ + FormattingRx = /#{AttributeRx}|#{ColorRx}/ + # Standard color codes ColorCode = { :black => 1, @@ -65,7 +72,7 @@ module Irc :dark_gray => 14, :lightgray => 15, :light_gray => 15, - :white => 16 + :white => 0 } # Convert a String or Symbol into a color number @@ -79,7 +86,7 @@ module Irc data end if ColorCode.key?(f) - ColorCode[f] + ColorCode[f] else 0 end @@ -119,24 +126,49 @@ module Irc # User/Channel message was sent to attr_reader :target - # contents of the message + # contents of the message (stripped of initial/final format codes) attr_accessor :message # contents of the message (for logging purposes) attr_accessor :logmessage + # contents of the message (stripped of all formatting) + attr_accessor :plainmessage + # has the message been replied to/handled by a plugin? attr_accessor :replied + alias :replied? :replied # should the message be ignored? attr_accessor :ignored alias :ignored? :ignored - # should the message handler be excuted in new thread? - # if set to true or false, this overrides :thread option in map. if it's nil, - # the map option takes effect - attr_accessor :thread - alias :thread? :thread + # set this to true if the method that delegates the message is run in a thread + attr_accessor :in_thread + alias :in_thread? :in_thread + + def inspect(fields=nil) + ret = self.__to_s__[0..-2] + ret << ' bot=' << @bot.__to_s__ + ret << ' server=' << server.to_s + ret << ' time=' << time.to_s + ret << ' source=' << source.to_s + ret << ' target=' << target.to_s + ret << ' message=' << message.inspect + ret << ' logmessage=' << logmessage.inspect + ret << ' plainmessage=' << plainmessage.inspect + ret << fields if fields + ret << ' (identified)' if identified? + if address? + ret << ' (addressed to me' + ret << ', with prefix' if prefixed? + ret << ')' + end + ret << ' (replied)' if replied? + ret << ' (ignored)' if ignored? + ret << ' (in thread)' if in_thread? + ret << '>' + end # instantiate a new Message # bot:: associated bot class @@ -151,12 +183,13 @@ module Irc @bot = bot @source = source @address = false + @prefixed = false @target = target - @message = BasicUserMessage.stripcolour message + @message = message || "" @replied = false @server = server @ignored = false - @thread = nil + @in_thread = false @identified = false if @msg_wants_id && @server.capabilities[:"identify-msg"] @@ -168,6 +201,8 @@ module Irc end end @logmessage = @message.dup + @plainmessage = BasicUserMessage.strip_formatting(@message) + @message = BasicUserMessage.strip_initial_formatting(@message) if target && target == @bot.myself @address = true @@ -207,9 +242,10 @@ module Irc return @address end - # has this message been replied to by a plugin? - def replied? - return @replied + # returns true if the messaged was addressed to the bot via the address + # prefix. This can be used to tell appart "!do this" from "botname, do this" + def prefixed? + return @prefixed end # strip mIRC colour escapes from a string @@ -220,6 +256,15 @@ module Irc ret end + def BasicUserMessage.strip_initial_formatting(string) + return "" unless string + ret = string.gsub(/^#{FormattingRx}|#{FormattingRx}$/,"") + end + + def BasicUserMessage.strip_formatting(string) + string.gsub(FormattingRx,"") + end + end # class for handling welcome messages from the server @@ -237,6 +282,24 @@ module Irc # (address? will return true in this case) class UserMessage < BasicUserMessage + def inspect + fields = ' plugin=' << plugin.inspect + fields << ' params=' << params.inspect + fields << ' channel=' << channel.to_s if channel + fields << ' (reply to ' << replyto.to_s << ')' + if self.private? + fields << ' (private)' + else + fields << ' (public)' + end + if self.action? + fields << ' (action)' + elsif ctcp + fields << ' (CTCP ' << ctcp << ')' + end + super(fields) + end + # for plugin messages, the name of the plugin invoked by the message attr_reader :plugin @@ -290,6 +353,7 @@ module Irc bot.config['core.address_prefix'].each {|mprefix| if @message.gsub!(/^#{Regexp.escape(mprefix)}\s*/, "") @address = true + @prefixed = true break end } @@ -309,11 +373,14 @@ module Irc @action = @ctcp == 'ACTION' debug "Received CTCP command #{@ctcp} with options #{@message} (action? #{@action})" @logmessage = @message.dup + @plainmessage = BasicUserMessage.strip_formatting(@message) + @message = BasicUserMessage.strip_initial_formatting(@message) end # free splitting for plugins @params = @message.dup - if @params.gsub!(/^\s*(\S+)[\s$]*/, "") + # Created messges (such as by fake_message) can contain multiple lines + if @params.gsub!(/\A\s*(\S+)[\s$]*/m, "") @plugin = $1.downcase @params = nil unless @params.length > 0 end @@ -339,26 +406,56 @@ module Irc # So if the message is private, it will reply to the user. If it was # in a channel, it will reply in the channel. def plainreply(string, options={}) - @bot.say @replyto, string, options - @replied = true + reply string, {:nick => false}.merge(options) end # Same as reply, but when replying in public it adds the nick of the user # the bot is replying to def nickreply(string, options={}) - extra = self.public? ? "#{@source}#{@bot.config['core.nick_postfix']} " : "" - @bot.say @replyto, extra + string, options - @replied = true + reply string, {:nick => true}.merge(options) end - # the default reply style is to nickreply unless the reply already contains - # the nick or core.reply_with_nick is set to false + # Same as nickreply, but always prepend the target's nick. + def nickreply!(string, options={}) + reply string, {:nick => true, :forcenick => true}.merge(options) + end + + # The general way to reply to a command. The following options are available: + # :nick [false, true, :auto] + # state if the nick of the user calling the command should be prepended + # :auto uses core.reply_with_nick + # + # :forcenick [false, true] + # if :nick is true, always prepend the target's nick, even if the nick + # already appears in the reply. Defaults to false. # + # :to [:private, :public, :auto] + # where should the bot reply? + # :private always reply to the nick + # :public reply to the channel (if available) + # :auto uses core.private_replies def reply(string, options={}) - if @bot.config['core.reply_with_nick'] and not string =~ /(?:^|\W)#{Regexp.escape(@source.to_s)}(?:$|\W)/ - return nickreply(string, options) + opts = {:nick => :auto, :forcenick => false, :to => :auto}.merge options + + if opts[:nick] == :auto + opts[:nick] = @bot.config['core.reply_with_nick'] + end + + if !self.public? + opts[:to] = :private + elsif opts[:to] == :auto + opts[:to] = @bot.config['core.private_replies'] ? :private : :public + end + + if (opts[:nick] && + opts[:to] != :private && + (string !~ /(?:^|\W)#{Regexp.escape(@source.to_s)}(?:$|\W)/ || + opts[:forcenick])) + string = "#{@source}#{@bot.config['core.nick_postfix']} #{string}" end - plainreply(string, options) + to = (opts[:to] == :private) ? source : @channel + @bot.say to, string, options + @replied = true end # convenience method to reply to a message with an action. It's the @@ -377,30 +474,30 @@ module Irc @bot.ctcp_notice @source, @ctcp, string, options end - # convenience method to reply "okay" in the current language to the - # message - def plainokay - self.plainreply @bot.lang.get("okay") + # convenience method to reply a literal message in the current language to the message + def plain_literal(ident) + self.reply @bot.lang.get(ident), :nick => false end # Like the above, but append the username - def nickokay - str = @bot.lang.get("okay").dup + def nick_literal(ident) + str = @bot.lang.get(ident).dup if self.public? # remove final punctuation str.gsub!(/[!,.]$/,"") str += ", #{@source}" end - self.plainreply str + self.reply str, :nick => false end # the default okay style is the same as the default reply style - # def okay - if @bot.config['core.reply_with_nick'] - return nickokay - end - plainokay + @bot.config['core.reply_with_nick'] ? nick_literal('okay') : plain_literal('okay') + end + + # thanks the user in reply + def thanks + @bot.config['core.reply_with_nick'] ? nick_literal('thanks') : plain_literal('thanks') end # send a NOTICE to the message source @@ -413,17 +510,17 @@ module Irc # class to manage IRC PRIVMSGs class PrivMessage < UserMessage - def initialize(bot, server, source, target, message) - @msg_wants_id = true - super + def initialize(bot, server, source, target, message, opts={}) + @msg_wants_id = opts[:handle_id] + super(bot, server, source, target, message) end end # class to manage IRC NOTICEs class NoticeMessage < UserMessage - def initialize(bot, server, source, target, message) - @msg_wants_id = true - super + def initialize(bot, server, source, target, message, opts={}) + @msg_wants_id = opts[:handle_id] + super(bot, server, source, target, message) end end @@ -434,6 +531,11 @@ module Irc # channel user was kicked from attr_reader :channel + def inspect + fields = ' channel=' << channel.to_s + super(fields) + end + def initialize(bot, server, source, target, channel, message="") super(bot, server, source, target, message) @channel = channel @@ -447,6 +549,11 @@ module Irc # channel user was invited to attr_reader :channel + def inspect + fields = ' channel=' << channel.to_s + super(fields) + end + def initialize(bot, server, source, target, channel, message="") super(bot, server, source, target, message) @channel = channel @@ -459,6 +566,7 @@ module Irc attr_accessor :is_on def initialize(bot, server, source, oldnick, newnick) super(bot, server, source, oldnick, newnick) + @address = (source == @bot.myself) @is_on = [] end @@ -469,6 +577,12 @@ module Irc def newnick return @message end + + def inspect + fields = ' old=' << oldnick + fields << ' new=' << newnick + super(fields) + end end # class to manage mode changes @@ -479,8 +593,43 @@ module Irc @address = (source == @bot.myself) @modes = [] end + + def inspect + fields = ' modes=' << modes.inspect + super(fields) + end + end + + # class to manage WHOIS replies + class WhoisMessage < BasicUserMessage + attr_reader :whois + def initialize(bot, server, source, target, whois) + super(bot, server, source, target, "") + @address = (target == @bot.myself) + @whois = whois + end + + def inspect + fields = ' whois=' << whois.inspect + super(fields) + end end + # class to manage LIST replies + class ListMessage < BasicUserMessage + attr_accessor :list + def initialize(bot, server, source, target, list=Hash.new) + super(bot, server, source, target, "") + @list = [] + end + + def inspect + fields = ' list=' << list.inspect + super(fields) + end + end + + # class to manage NAME replies class NamesMessage < BasicUserMessage attr_accessor :users @@ -488,6 +637,27 @@ module Irc super(bot, server, source, target, message) @users = [] end + + def inspect + fields = ' users=' << users.inspect + super(fields) + end + end + + # class to manager Ban list replies + class BanlistMessage < BasicUserMessage + # the bans + attr_accessor :bans + + def initialize(bot, server, source, target, message="") + super(bot, server, source, target, message) + @bans = [] + end + + def inspect + fields = ' bans=' << bans.inspect + super(fields) + end end class QuitMessage < BasicUserMessage @@ -515,12 +685,24 @@ module Irc @channel = channel @info_or_set = nil end + + def inspect + fields = ' topic=' << topic + fields << ' (set on ' << timestamp << ')' + super(fields) + end end # class to manage channel joins class JoinMessage < BasicUserMessage # channel joined attr_reader :channel + + def inspect + fields = ' channel=' << channel.to_s + super(fields) + end + def initialize(bot, server, source, channel, message="") super(bot, server, source, channel, message) @channel = channel @@ -534,6 +716,18 @@ module Irc class PartMessage < JoinMessage end + # class to handle ERR_NOSUCHNICK and ERR_NOSUCHCHANNEL + class NoSuchTargetMessage < BasicUserMessage + # the channel or nick that was not found + attr_reader :target + + def initialize(bot, server, source, target, message='') + super(bot, server, source, target, message) + + @target = target + end + end + class UnknownMessage < BasicUserMessage end end