diff --git a/irc/reply.go b/irc/reply.go deleted file mode 100644 index 87c58613..00000000 --- a/irc/reply.go +++ /dev/null @@ -1,584 +0,0 @@ -// Copyright (c) 2012-2014 Jeremy Latt -// Copyright (c) 2014-2015 Edmund Huber -// Copyright (c) 2016- Daniel Oaks -// released under the MIT license - -package irc - -/* - -type ReplyCode interface { - String() string -} - -type StringCode string - -func (code StringCode) String() string { - return string(code) -} - -type NumericCode uint - -func (code NumericCode) String() string { - return fmt.Sprintf("%03d", code) -} - -func NewStringReply(source Identifiable, code StringCode, - format string, args ...interface{}) string { - var header string - if source == nil { - header = code.String() + " " - } else { - header = fmt.Sprintf(":%s %s ", source, code) - } - var message string - if len(args) > 0 { - message = fmt.Sprintf(format, args...) - } else { - message = format - } - return header + message -} - -func NewNumericReply(target *Client, code NumericCode, - format string, args ...interface{}) string { - header := fmt.Sprintf(":%s %s %s ", target.server.Id(), code, target.Nick()) - var message string - if len(args) > 0 { - message = fmt.Sprintf(format, args...) - } else { - message = format - } - return header + message -} - -func (target *Client) NumericReply(code NumericCode, - format string, args ...interface{}) { - target.Reply(NewNumericReply(target, code, format, args...)) -} - -// -// multiline replies -// - -func joinedLen(names []string) int { - var l = len(names) - 1 // " " between names - for _, name := range names { - l += len(name) - } - return l -} - -func (target *Client) MultilineReply(names []string, code NumericCode, format string, - args ...interface{}) { - baseLen := len(NewNumericReply(target, code, format)) - tooLong := func(names []string) bool { - return (baseLen + joinedLen(names)) > MAX_REPLY_LEN - } - argsAndNames := func(names []string) []interface{} { - return append(args, strings.Join(names, " ")) - } - from, to := 0, 1 - for to < len(names) { - if (from < (to - 1)) && tooLong(names[from:to]) { - target.NumericReply(code, format, argsAndNames(names[from:to-1])...) - from = to - 1 - } else { - to += 1 - } - } - if from < len(names) { - target.NumericReply(code, format, argsAndNames(names[from:])...) - } -} - -// -// messaging replies -// - -func RplPrivMsg(source Identifiable, target Identifiable, message Text) string { - return NewStringReply(source, PRIVMSG, "%s :%s", target.Nick(), message) -} - -func RplCTCPAction(source Identifiable, target Identifiable, action CTCPText) string { - return RplPrivMsg(source, target, NewText(fmt.Sprintf("\x01ACTION %s\x01", action))) -} - -func RplNotice(source Identifiable, target Identifiable, message Text) string { - return NewStringReply(source, NOTICE, "%s :%s", target.Nick(), message) -} - -func RplNick(source Identifiable, newNick Name) string { - return NewStringReply(source, NICK, newNick.String()) -} - -func RplJoin(client *Client, channel *Channel) string { - return NewStringReply(client, JOIN, channel.name.String()) -} - -func RplPart(client *Client, channel *Channel, message Text) string { - return NewStringReply(client, PART, "%s :%s", channel, message) -} - -func RplModeChanges(client *Client, target *Client, changes ModeChanges) string { - return NewStringReply(client, MODE, "%s :%s", target.Nick(), changes) -} - -func RplChannelMode(client *Client, channel *Channel, - changes ChannelModeChanges) string { - return NewStringReply(client, MODE, "%s %s", channel, changes) -} - -func RplTopicMsg(source Identifiable, channel *Channel) string { - return NewStringReply(source, TOPIC, "%s :%s", channel, channel.topic) -} - -func RplPing(target Identifiable) string { - return NewStringReply(nil, PING, ":%s", target.Nick()) -} - -func RplPong(client *Client, msg Text) string { - // #5: IRC for Android will time out if it doesn't get the prefix back. - return NewStringReply(client, PONG, "%s :%s", client.server, msg.String()) -} - -func RplQuit(client *Client, message Text) string { - return NewStringReply(client, QUIT, ":%s", message) -} - -func RplError(message string) string { - return NewStringReply(nil, ERROR, ":%s", message) -} - -func RplInviteMsg(inviter *Client, invitee *Client, channel Name) string { - return NewStringReply(inviter, INVITE, "%s :%s", invitee.Nick(), channel) -} - -func RplKick(channel *Channel, client *Client, target *Client, comment Text) string { - return NewStringReply(client, KICK, "%s %s :%s", - channel, target.Nick(), comment) -} - -func RplKill(client *Client, target *Client, comment Text) string { - return NewStringReply(client, KICK, - "%s :%s", target.Nick(), comment) -} - -func RplCap(client *Client, subCommand CapSubCommand, arg interface{}) string { - // client.server needs to be here to workaround a parsing bug in weechat 1.4 - // and let it connect to the server (otherwise it doesn't respond to the CAP - // message with anything and just hangs on connection) - return NewStringReply(client.server, CAP, "%s %s :%s", client.Nick(), subCommand, arg) -} - -// numeric replies - -func (target *Client) RplWelcome() { - target.NumericReply(RPL_WELCOME, - ":Welcome to the Internet Relay Network %s", target.Id()) -} - -func (target *Client) RplYourHost() { - target.NumericReply(RPL_YOURHOST, - ":Your host is %s, running version %s", target.server.name, SEM_VER) -} - -func (target *Client) RplCreated() { - target.NumericReply(RPL_CREATED, - ":This server was created %s", target.server.ctime.Format(time.RFC1123)) -} - -func (target *Client) RplMyInfo() { - target.NumericReply(RPL_MYINFO, - "%s %s %s %s", - target.server.name, SEM_VER, SupportedUserModes, SupportedChannelModes) -} - -func (target *Client) RplISupport() { - for _, tokenline := range target.server.isupport.CachedReply { - target.NumericReply(RPL_ISUPPORT, - "%s :are supported by this server", - tokenline) - } -} - -func (target *Client) RplUModeIs(client *Client) { - target.NumericReply(RPL_UMODEIS, client.ModeString()) -} - -func (target *Client) RplNoTopic(channel *Channel) { - target.NumericReply(RPL_NOTOPIC, - "%s :No topic is set", channel.name) -} - -func (target *Client) RplTopic(channel *Channel) { - target.NumericReply(RPL_TOPIC, - "%s :%s", channel.name, channel.topic) -} - -// -// NB: correction in errata -func (target *Client) RplInvitingMsg(invitee *Client, channel Name) { - target.NumericReply(RPL_INVITING, - "%s %s", invitee.Nick(), channel) -} - -func (target *Client) RplEndOfNames(channel *Channel) { - target.NumericReply(RPL_ENDOFNAMES, - "%s :End of NAMES list", channel.name) -} - -// :You are now an IRC operator -func (target *Client) RplYoureOper() { - target.NumericReply(RPL_YOUREOPER, - ":You are now an IRC operator") -} - -func (target *Client) RplWhois(client *Client) { - target.RplWhoisUser(client) - if client.flags[Operator] { - target.RplWhoisOperator(client) - } - target.RplWhoisIdle(client) - target.RplWhoisChannels(client) - target.RplEndOfWhois(client) -} - -func (target *Client) RplWhoisUser(client *Client) { - target.NumericReply(RPL_WHOISUSER, - "%s %s %s * :%s", client.Nick(), client.username, client.hostname, - client.realname) -} - -func (target *Client) RplWhoisOperator(client *Client) { - target.NumericReply(RPL_WHOISOPERATOR, - "%s :is an IRC operator", client.Nick()) -} - -func (target *Client) RplWhoisIdle(client *Client) { - target.NumericReply(RPL_WHOISIDLE, - "%s %d %d :seconds idle, signon time", - client.Nick(), client.IdleSeconds(), client.SignonTime()) -} - -func (target *Client) RplEndOfWhois(client *Client) { - target.NumericReply(RPL_ENDOFWHOIS, - "%s :End of WHOIS list", client.Nick()) -} - -func (target *Client) RplChannelModeIs(channel *Channel) { - target.NumericReply(RPL_CHANNELMODEIS, - "%s %s", channel, channel.ModeString(target)) -} - -// ( "H" / "G" ) ["*"] [ ( "@" / "+" ) ] -// : -func (target *Client) RplWhoReply(channel *Channel, client *Client) { - channelName := "*" - flags := "" - - if client.flags[Away] { - flags = "G" - } else { - flags = "H" - } - if client.flags[Operator] { - flags += "*" - } - - if channel != nil { - flags += channel.members[client].Prefixes(target.capabilities[MultiPrefix]) - channelName = channel.name.String() - } - target.NumericReply(RPL_WHOREPLY, - "%s %s %s %s %s %s :%d %s", channelName, client.username, client.hostname, - client.server.name, client.Nick(), flags, client.hops, client.realname) -} - -// :End of WHO list -func (target *Client) RplEndOfWho(name Name) { - target.NumericReply(RPL_ENDOFWHO, - "%s :End of WHO list", name) -} - -func (target *Client) RplMaskList(mode ChannelMode, channel *Channel, mask Name) { - switch mode { - case BanMask: - target.RplBanList(channel, mask) - - case ExceptMask: - target.RplExceptList(channel, mask) - - case InviteMask: - target.RplInviteList(channel, mask) - } -} - -func (target *Client) RplEndOfMaskList(mode ChannelMode, channel *Channel) { - switch mode { - case BanMask: - target.RplEndOfBanList(channel) - - case ExceptMask: - target.RplEndOfExceptList(channel) - - case InviteMask: - target.RplEndOfInviteList(channel) - } -} - -func (target *Client) RplBanList(channel *Channel, mask Name) { - target.NumericReply(RPL_BANLIST, - "%s %s", channel, mask) -} - -func (target *Client) RplEndOfBanList(channel *Channel) { - target.NumericReply(RPL_ENDOFBANLIST, - "%s :End of channel ban list", channel) -} - -func (target *Client) RplExceptList(channel *Channel, mask Name) { - target.NumericReply(RPL_EXCEPTLIST, - "%s %s", channel, mask) -} - -func (target *Client) RplEndOfExceptList(channel *Channel) { - target.NumericReply(RPL_ENDOFEXCEPTLIST, - "%s :End of channel exception list", channel) -} - -func (target *Client) RplInviteList(channel *Channel, mask Name) { - target.NumericReply(RPL_INVITELIST, - "%s %s", channel, mask) -} - -func (target *Client) RplEndOfInviteList(channel *Channel) { - target.NumericReply(RPL_ENDOFINVITELIST, - "%s :End of channel invite list", channel) -} - -func (target *Client) RplNowAway() { - target.NumericReply(RPL_NOWAWAY, - ":You have been marked as being away") -} - -func (target *Client) RplUnAway() { - target.NumericReply(RPL_UNAWAY, - ":You are no longer marked as being away") -} - -func (target *Client) RplAway(client *Client) { - target.NumericReply(RPL_AWAY, - "%s :%s", client.Nick(), client.awayMessage) -} - -func (target *Client) RplIsOn(nicks []string) { - target.NumericReply(RPL_ISON, - ":%s", strings.Join(nicks, " ")) -} - -func (target *Client) RplMOTDStart() { - target.NumericReply(RPL_MOTDSTART, - ":- %s Message of the day - ", target.server.name) -} - -func (target *Client) RplMOTD(line string) { - target.NumericReply(RPL_MOTD, - ":- %s", line) -} - -func (target *Client) RplMOTDEnd() { - target.NumericReply(RPL_ENDOFMOTD, - ":End of MOTD command") -} - -func (target *Client) RplList(channel *Channel) { - target.NumericReply(RPL_LIST, - "%s %d :%s", channel, len(channel.members), channel.topic) -} - -func (target *Client) RplListEnd(server *Server) { - target.NumericReply(RPL_LISTEND, - ":End of LIST") -} - -func (target *Client) RplNamReply(channel *Channel) { - target.MultilineReply(channel.Nicks(target), RPL_NAMREPLY, - "= %s :%s", channel) -} - -func (target *Client) RplWhoisChannels(client *Client) { - target.MultilineReply(client.WhoisChannelsNames(target), RPL_WHOISCHANNELS, - "%s :%s", client.Nick()) -} - -func (target *Client) RplVersion() { - target.NumericReply(RPL_VERSION, - "%s %s", SEM_VER, target.server.name) -} - -func (target *Client) RplInviting(invitee *Client, channel Name) { - target.NumericReply(RPL_INVITING, - "%s %s", invitee.Nick(), channel) -} - -func (target *Client) RplTime() { - target.NumericReply(RPL_TIME, - "%s :%s", target.server.name, time.Now().Format(time.RFC1123)) -} - -func (target *Client) RplWhoWasUser(whoWas *WhoWas) { - target.NumericReply(RPL_WHOWASUSER, - "%s %s %s * :%s", - whoWas.nickname, whoWas.username, whoWas.hostname, whoWas.realname) -} - -func (target *Client) RplEndOfWhoWas(nickname Name) { - target.NumericReply(RPL_ENDOFWHOWAS, - "%s :End of WHOWAS", nickname) -} - -// -// errors (also numeric) -// - -func (target *Client) ErrAlreadyRegistered() { - target.NumericReply(ERR_ALREADYREGISTRED, - ":You may not reregister") -} - -func (target *Client) ErrNickNameInUse(nick Name) { - target.NumericReply(ERR_NICKNAMEINUSE, - "%s :Nickname is already in use", nick) -} - -func (target *Client) ErrUnknownCommand(code StringCode) { - target.NumericReply(ERR_UNKNOWNCOMMAND, - "%s :Unknown command", code) -} - -func (target *Client) ErrUsersDontMatch() { - target.NumericReply(ERR_USERSDONTMATCH, - ":Cannot change mode for other users") -} - -func (target *Client) ErrNeedMoreParams(command StringCode) { - target.NumericReply(ERR_NEEDMOREPARAMS, - "%s :Not enough parameters", command) -} - -func (target *Client) ErrNoSuchChannel(channel Name) { - target.NumericReply(ERR_NOSUCHCHANNEL, - "%s :No such channel", channel) -} - -func (target *Client) ErrUserOnChannel(channel *Channel, member *Client) { - target.NumericReply(ERR_USERONCHANNEL, - "%s %s :is already on channel", member.Nick(), channel.name) -} - -func (target *Client) ErrNotOnChannel(channel *Channel) { - target.NumericReply(ERR_NOTONCHANNEL, - "%s :You're not on that channel", channel.name) -} - -func (target *Client) ErrInviteOnlyChannel(channel *Channel) { - target.NumericReply(ERR_INVITEONLYCHAN, - "%s :Cannot join channel (+i)", channel.name) -} - -func (target *Client) ErrBadChannelKey(channel *Channel) { - target.NumericReply(ERR_BADCHANNELKEY, - "%s :Cannot join channel (+k)", channel.name) -} - -func (target *Client) ErrNoSuchNick(nick Name) { - target.NumericReply(ERR_NOSUCHNICK, - "%s :No such nick", nick) -} - -func (target *Client) ErrPasswdMismatch() { - target.NumericReply(ERR_PASSWDMISMATCH, ":Password incorrect") -} - -func (target *Client) ErrNoChanModes(channel *Channel) { - target.NumericReply(ERR_NOCHANMODES, - "%s :Channel doesn't support modes", channel) -} - -func (target *Client) ErrNoPrivileges() { - target.NumericReply(ERR_NOPRIVILEGES, ":Permission Denied") -} - -func (target *Client) ErrRestricted() { - target.NumericReply(ERR_RESTRICTED, ":Your connection is restricted!") -} - -func (target *Client) ErrNoSuchServer(server Name) { - target.NumericReply(ERR_NOSUCHSERVER, "%s :No such server", server) -} - -func (target *Client) ErrUserNotInChannel(channel *Channel, client *Client) { - target.NumericReply(ERR_USERNOTINCHANNEL, - "%s %s :They aren't on that channel", client.Nick(), channel) -} - -func (target *Client) ErrCannotSendToChan(channel *Channel) { - target.NumericReply(ERR_CANNOTSENDTOCHAN, - "%s :Cannot send to channel", channel) -} - -// :You're not channel operator -func (target *Client) ErrChanOPrivIsNeeded(channel *Channel) { - target.NumericReply(ERR_CHANOPRIVSNEEDED, - "%s :You're not channel operator", channel) -} - -func (target *Client) ErrNoMOTD() { - target.NumericReply(ERR_NOMOTD, ":MOTD File is missing") -} - -func (target *Client) ErrNoNicknameGiven() { - target.NumericReply(ERR_NONICKNAMEGIVEN, ":No nickname given") -} - -func (target *Client) ErrErroneusNickname(nick Name) { - target.NumericReply(ERR_ERRONEUSNICKNAME, - "%s :Erroneous nickname", nick) -} - -func (target *Client) ErrUnknownMode(mode ChannelMode, channel *Channel) { - target.NumericReply(ERR_UNKNOWNMODE, - "%s :is unknown mode char to me for %s", mode, channel) -} - -func (target *Client) ErrConfiguredMode(mode ChannelMode) { - target.NumericReply(ERR_UNKNOWNMODE, - "%s :can only change this mode in daemon configuration", mode) -} - -func (target *Client) ErrChannelIsFull(channel *Channel) { - target.NumericReply(ERR_CHANNELISFULL, - "%s :Cannot join channel (+l)", channel) -} - -func (target *Client) ErrWasNoSuchNick(nickname Name) { - target.NumericReply(ERR_WASNOSUCHNICK, - "%s :There was no such nickname", nickname) -} - -func (target *Client) ErrInvalidCapCmd(subCommand CapSubCommand) { - target.NumericReply(ERR_INVALIDCAPCMD, - "%s :Invalid CAP subcommand", subCommand) -} - -func (target *Client) ErrBannedFromChan(channel *Channel) { - target.NumericReply(ERR_BANNEDFROMCHAN, - "%s :Cannot join channel (+b)", channel) -} - -func (target *Client) ErrInviteOnlyChan(channel *Channel) { - target.NumericReply(ERR_INVITEONLYCHAN, - "%s :Cannot join channel (+i)", channel) -} -*/