#include <QDateTime>
Server::Server(QString net) : network(net) {
-
-
+ QString MQUOTE = QString('\020');
+ ctcpMDequoteHash[MQUOTE + '0'] = QString('\000');
+ ctcpMDequoteHash[MQUOTE + 'n'] = QString('\n');
+ ctcpMDequoteHash[MQUOTE + 'r'] = QString('\r');
+ ctcpMDequoteHash[MQUOTE + MQUOTE] = MQUOTE;
}
Server::~Server() {
emit displayMsg(Message::Server, "", params.join(" "), prefix);
break;
// Server error messages without param, just display them
- case 409: case 411: case 412: case 422: case 424: case 431: case 445: case 446: case 451: case 462:
+ case 409: case 411: case 412: case 422: case 424: case 445: case 446: case 451: case 462:
case 463: case 464: case 465: case 466: case 472: case 481: case 483: case 485: case 491: case 501: case 502:
+ case 431: // ERR_NONICKNAMEGIVEN
emit displayMsg(Message::Error, "", params.join(" "), prefix);
break;
// Server error messages, display them in red. First param will be appended.
- case 401: case 402: case 403: case 404: case 406: case 408: case 415: case 421: case 432: case 442:
+ case 401: case 402: case 403: case 404: case 406: case 408: case 415: case 421: case 442:
{ QString p = params.takeFirst();
emit displayMsg(Message::Error, "", params.join(" ") + " " + p, prefix);
break;
}
// Server error messages which will be displayed with a colon between the first param and the rest
- case 413: case 414: case 423: case 436: case 441: case 444: case 461:
+ case 413: case 414: case 423: case 441: case 444: case 461:
case 467: case 471: case 473: case 474: case 475: case 476: case 477: case 478: case 482:
+ case 436: // ERR_NICKCOLLISION
{ QString p = params.takeFirst();
emit displayMsg(Message::Error, "", p + ": " + params.join(" "));
break;
return userHandlers;
}
+QString Server::ctcpDequote(QString message) {
+ QString dequotedMessage;
+ QString messagepart;
+ int offset;
+
+ // copy dequote Message
+ for(int i = 0; i < message.size(); i++) {
+ messagepart = message[i];
+ offset = 0;
+ if(i+1 < message.size()) {
+ for(QHash<QString, QString>::iterator ctcpquote = ctcpMDequoteHash.begin(); ctcpquote != ctcpMDequoteHash.end(); ++ctcpquote) {
+ if(message.mid(i,2) == ctcpquote.key()) {
+ dequotedMessage += ctcpquote.value();
+ offset = 1;
+ }
+ }
+ }
+ dequotedMessage += messagepart;
+ i += offset;
+ }
+ return dequotedMessage;
+}
+
+QString Server::ctcpXdelimDequote(QString message) {
+ QString dequotedMessage;
+ QString messagepart;
+ QString XQUOTE = QString('\134');
+ QString XQUOTEQUOTED = XQUOTE + QString('\134');
+ QString XDELIM = QString('\001');
+ QString XDELIMQUOTED = XQUOTE + QString('a');
+
+ for(int i = 0; i < message.size(); i++) {
+ messagepart = message[i];
+ if(i+1 < message.size()) {
+ if(message.mid(i,2) == XQUOTEQUOTED) {
+ messagepart = XQUOTE;
+ i++;
+ } else if(message.mid(i,2) == XDELIMQUOTED) {
+ messagepart = XDELIM;
+ i++;
+ }
+ }
+ dequotedMessage += messagepart;
+ }
+ return dequotedMessage;
+}
+
+QStringList Server::parseCtcp(CtcpType ctcptype, QString prefix, QString target, QString message) {
+ QStringList messages;
+ QString ctcp;
+
+ //lowlevel message dequote
+ QString dequotedMessage = ctcpDequote(message);
+
+ // extract tagged / extended data
+ QString XDELIM = QString('\001');
+ while(dequotedMessage.contains(XDELIM)) {
+ messages << dequotedMessage.section(XDELIM,0,0);
+ ctcp = dequotedMessage.section(XDELIM,1,1);
+ dequotedMessage = dequotedMessage.section(XDELIM,2,2);
+
+ //dispatch the ctcp command
+ QString ctcpcmd = ctcp.section(' ', 0, 0);
+ QString ctcpparam = ctcp.section(' ', 1);
+
+ // Now we try to find a handler for this message. BTW, I do love the Trolltech guys ;-)
+ QString hname = ctcpcmd.toLower();
+ hname[0] = hname[0].toUpper();
+ hname = "handleCtcp" + hname;
+ if(!QMetaObject::invokeMethod(this, hname.toAscii(), Q_ARG(CtcpType, ctcptype), Q_ARG(QString, prefix), Q_ARG(QString, target), Q_ARG(QString, ctcpparam))) {
+ // Ok. Default handler it is.
+ defaultCtcpHandler(ctcptype, prefix, ctcpcmd, target, ctcpparam);
+ }
+
+
+ }
+ if(!dequotedMessage.isEmpty()) {
+ messages << dequotedMessage;
+ }
+ return messages;
+
+}
/**********************************************************************************/
Q_ASSERT(params.count() >= 2);
if(params.count()<2) emit displayMsg(Message::Plain, params[0], "", prefix);
else {
- if(params[0] == ownNick) {
- emit displayMsg(Message::Plain, "", params[1], prefix, Message::PrivMsg);
+ // it's possible to pack multiple privmsgs into one param using ctcp
+ QStringList messages = parseCtcp(Server::CtcpQuery, prefix, params[0], params[1]);
+ if(params[0].toLower() == ownNick.toLower()) { // Freenode sends nickname in lower case!
+ foreach(QString message, messages) {
+ if(!message.isEmpty()) {
+ emit displayMsg(Message::Plain, "", message, prefix, Message::PrivMsg);
+ }
+ }
+
} else {
+ //qDebug() << prefix << params;
Q_ASSERT(isChannelName(params[0])); // should be channel!
- emit displayMsg(Message::Plain, params[0], params[1], prefix);
+ foreach(QString message, messages) {
+ if(!message.isEmpty()) {
+ emit displayMsg(Message::Plain, params[0], message, prefix);
+ }
+ }
}
}
}
}
}
-/* RPL_NICKNAMEINUSER */
+/* ERR_ERRONEUSNICKNAME */
+void Server::handleServer432(QString prefix, QStringList params) {
+ if(params.size() < 2) {
+ // handle unreal-ircd bug, where unreal ircd doesnt supply a TARGET in ERR_ERRONEUSNICKNAME during registration phase:
+ // nick @@@
+ // :irc.scortum.moep.net 432 @@@ :Erroneous Nickname: Illegal characters
+ // correct server reply:
+ // :irc.scortum.moep.net 432 * @@@ :Erroneous Nickname: Illegal characters
+ emit displayMsg(Message::Error, "", tr("There is a nickname in your identity's nicklist which contains illegal characters"));
+ emit displayMsg(Message::Error, "", tr("Due to a bug in Unreal IRCd (and maybe other irc-servers too) we're unable to determine the erroneous nick"));
+ emit displayMsg(Message::Error, "", tr("Please use: /nick <othernick> to continue or clean up your nicklist"));
+ } else {
+ QString errnick = params[0];
+ emit displayMsg(Message::Error, "", tr("Nick %1 contains illegal characters").arg(errnick));
+ // if there is a problem while connecting to the server -> we handle it
+ // TODO rely on another source...
+ if(currentServer.isEmpty()) {
+ QStringList desiredNicks = identity["NickList"].toStringList();
+ int nextNick = desiredNicks.indexOf(errnick) + 1;
+ if (desiredNicks.size() > nextNick) {
+ putCmd("NICK", QStringList(desiredNicks[nextNick]));
+ } else {
+ emit displayMsg(Message::Error, "", tr("No free and valid nicks in nicklist found. use: /nick <othernick> to continue"));
+ }
+ }
+ }
+}
+
+/* ERR_NICKNAMEINUSE */
void Server::handleServer433(QString prefix, QStringList params) {
QString errnick = params[0];
emit displayMsg(Message::Error, "", tr("Nick %1 is already taken").arg(errnick));
if (desiredNicks.size() > nextNick) {
putCmd("NICK", QStringList(desiredNicks[nextNick]));
} else {
- emit displayMsg(Message::Error, "", "All nicks in nicklist taken... use: /nick <othernick> to continue");
+ emit displayMsg(Message::Error, "", tr("No free and valid nicks in nicklist found. use: /nick <othernick> to continue"));
}
}
}
+/***********************************************************************************/
+// CTCP HANDLER
+
+void Server::handleCtcpAction(CtcpType ctcptype, QString prefix, QString target, QString param) {
+ emit displayMsg(Message::Action, target, param, prefix);
+}
+
+void Server::defaultCtcpHandler(CtcpType ctcptype, QString prefix, QString cmd, QString target, QString param) {
+ emit displayMsg(Message::Error, "", QString("Unknown CTCP: ") + cmd + (" ") + param, prefix);
+}
+
+
/***********************************************************************************/
/* Exception classes for message handling */