summaryrefslogtreecommitdiff
path: root/icqj_s7_sss_mod/icq_clients.c
diff options
context:
space:
mode:
Diffstat (limited to 'icqj_s7_sss_mod/icq_clients.c')
-rw-r--r--icqj_s7_sss_mod/icq_clients.c1030
1 files changed, 1030 insertions, 0 deletions
diff --git a/icqj_s7_sss_mod/icq_clients.c b/icqj_s7_sss_mod/icq_clients.c
new file mode 100644
index 0000000..0be0abf
--- /dev/null
+++ b/icqj_s7_sss_mod/icq_clients.c
@@ -0,0 +1,1030 @@
+// ---------------------------------------------------------------------------80
+// ICQ plugin for Miranda Instant Messenger
+// ________________________________________
+//
+// Copyright 2000,2001 Richard Hughes, Roland Rabien, Tristan Van de Vreede
+// Copyright 2001,2002 Jon Keating, Richard Hughes
+// Copyright 2002,2003,2004 Martin berg, Sam Kothari, Robert Rainwater
+// Copyright 2004,2005,2006,2007 Joe Kucera
+// Copyright 2006,2007,2008 [sss], chaos.persei, [sin], Faith Healer, Theif, nullbie
+// Copyright 2008 [sss], chaos.persei, nullbie, baloo, jarvis
+//
+// This program is free software; you can redistribute it and/or
+// modify it under the terms of the GNU General Public License
+// as published by the Free Software Foundation; either version 2
+// of the License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program; if not, write to the Free Software
+// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+//
+// -----------------------------------------------------------------------------
+//
+// File name : $Source$
+// Revision : $Revision: 36 $
+// Last change on : $Date: 2007-08-05 03:45:10 +0300 (Вс, 05 авг 2007) $
+// Last change by : $Author: sss123next $
+//
+// DESCRIPTION:
+//
+// Provides capability & signature based client detection
+//
+// -----------------------------------------------------------------------------
+
+#include "icqoscar.h"
+
+
+capstr* MatchCap(char* buf, int bufsize, const capstr* cap, int capsize)
+{
+ while (bufsize>0) // search the buffer for a capability
+ {
+ if (!memcmp(buf, cap, capsize))
+ {
+ return (capstr*)buf; // give found capability for version info
+ }
+ else
+ {
+ buf += 0x10;
+ bufsize -= 0x10;
+ }
+ }
+ return 0;
+}
+
+
+
+static void makeClientVersion(char *szBuf, const char* szClient, unsigned v1, unsigned v2, unsigned v3, unsigned v4)
+{
+ if (v4)
+ null_snprintf(szBuf, 64, "%s%u.%u.%u.%u", szClient, v1, v2, v3, v4);
+ else if (v3)
+ null_snprintf(szBuf, 64, "%s%u.%u.%u", szClient, v1, v2, v3);
+ else
+ null_snprintf(szBuf, 64, "%s%u.%u", szClient, v1, v2);
+}
+
+
+
+
+char* cliLibicq2k = "libicq2000";
+char* cliLicqVer = "Licq ";
+char* cliCentericq = "Centericq";
+char* cliLibicqUTF = "libicq2000 (Unicode)";
+char* cliTrillian = "Trillian";
+char* cliQip = "QIP %s";
+char* cliIM2 = "IM2";
+char* cliSpamBot = "Spam Bot";
+
+
+char* detectUserClient(HANDLE hContact, DWORD dwUin, WORD wVersion, DWORD dwFT1, DWORD dwFT2, DWORD dwFT3, DWORD dwOnlineSince, BYTE bDirectFlag, DWORD dwDirectCookie, DWORD dwWebPort, BYTE* caps, WORD wLen, BYTE* bClientId, char* szClientBuf)
+{
+ capstr* capId;
+ LPSTR szClient = {0};
+
+ *bClientId = CLID_ALTERNATIVE; // Most clients does not tick as MsgIDs
+
+ // Is this a Miranda IM client?
+ if (dwFT1 == 0xffffffff)
+ {
+ if (dwFT2 == 0xffffffff)
+ { // This is Gaim not Miranda
+ szClient = "Gaim";
+ }
+ else if (!dwFT2 && wVersion == 7)
+ { // This is WebICQ not Miranda
+ szClient = "WebICQ";
+ }
+ else if (!dwFT2 && dwFT3 == 0x3B7248ED)
+ { // And this is most probably Spam Bot
+ szClient = cliSpamBot;
+ }
+ else if(!MatchCap(caps,wLen, &capIcqJs7, 4)&&!MatchCap(caps,wLen, &capIcqJSin, 4)&&!MatchCap(caps,wLen, &capIcqJp, 4)&&!MatchCap(caps,wLen, &capIcqJen, 4))
+ { // Yes this is most probably Miranda, get the version info (need better realization)
+ unsigned ver1 = (dwFT2>>24)&0xFF;
+ unsigned ver2 = (dwFT2>>16)&0xFF;
+ unsigned ver3 = (dwFT2>>8)&0xFF;
+ unsigned ver4 = (dwFT2)&0xFF;
+ if(ver1 & 0x80)
+ {
+ makeClientVersion(szClientBuf, "Miranda IM (ICQ 0.", ver2, ver3, ver4, 0);
+ lstrcatA(szClientBuf," alpha)");
+ }
+ else
+ {
+ makeClientVersion(szClientBuf, "Miranda IM (ICQ ", ver1, ver2, ver3, ver4);
+ lstrcatA(szClientBuf,")");
+ }
+ szClient = szClientBuf;
+ *bClientId = CLID_MIRANDA;
+ }
+ }
+ else if ((dwFT1 & 0xFF7F0000) == 0x7D000000)
+ { // This is probably an Licq client
+ DWORD ver = dwFT1 & 0xFFFF;
+ makeClientVersion(szClientBuf, cliLicqVer, ver / 1000, (ver / 10) % 100, ver % 10, 0);
+ if (dwFT1 & 0x00800000)
+ strcat(szClientBuf, "/SSL");
+ szClient = szClientBuf;
+ }
+ else if (dwFT1 == 0xffffff8f)
+ {
+ szClient = "StrICQ";
+ }
+ else if (dwFT1 == 0xffffff42)
+ {
+ szClient = "mICQ";
+ }
+ else if (dwFT1 == 0xffffffbe)
+ {
+ unsigned ver1 = (dwFT2>>24)&0xFF;
+ unsigned ver2 = (dwFT2>>16)&0xFF;
+ unsigned ver3 = (dwFT2>>8)&0xFF;
+
+ makeClientVersion(szClientBuf, "Alicq ", ver1, ver2, ver3, 0);
+
+ szClient = szClientBuf;
+ }
+ else if (dwFT1 == 0xFFFFFF7F)
+ {
+ szClient = "&RQ";
+ }
+ else if (dwFT1 == 0xFFFFFFAB)
+ {
+ szClient = "YSM";
+ }
+ else if (dwFT1 == 0x04031980)
+ {
+ szClient = "vICQ";
+ }
+ else if ((dwFT1 == 0x3AA773EE) && (dwFT2 == 0x3AA66380))
+ {
+ szClient = cliLibicq2k;
+ }
+ else if (dwFT1 == 0x3B75AC09)
+ {
+ szClient = cliTrillian;
+ }
+ else if (dwFT1 == 0x3BA8DBAF) // FT2: 0x3BEB5373; FT3: 0x3BEB5262;
+ {
+ if (wVersion == 2)
+ szClient = "stICQ";
+ }
+ else if (dwFT1 == 0xFFFFFFFE && dwFT3 == 0xFFFFFFFE)
+ {
+ szClient = "Jimm";
+ }
+ else if (dwFT1 == 0x3FF19BEB && dwFT3 == 0x3FF19BEB)
+ {
+ szClient = cliIM2;
+ }
+ else if (dwFT1 == 0xDDDDEEFF && !dwFT2 && !dwFT3)
+ {
+ szClient = "SmartICQ";
+ }
+ else if ((dwFT1 & 0xFFFFFFF0) == 0x494D2B00 && !dwFT2 && !dwFT3)
+ { // last byte of FT1: (5 = Win32, 3 = SmartPhone, Pocket PC)
+ szClient = "IM+";
+ }
+ else if (dwFT1 == 0x3B4C4C0C && !dwFT2 && dwFT3 == 0x3B7248ed)
+ {
+ szClient = "KXicq2";
+ }
+ else if (dwFT1 == 0xFFFFF666 && !dwFT3)
+ { // this is R&Q (Rapid Edition)
+ null_snprintf(szClientBuf, 64, "R&Q %u", (unsigned)dwFT2);
+ szClient = szClientBuf;
+ }
+ else if (dwFT1 == 0x66666666 && dwFT3 == 0x66666666)
+ { // http://darkjimm.ucoz.ru/
+ strcpy(szClientBuf, "D[i]Chat");
+ if (dwFT2 == 0x10000)
+ strcat(szClientBuf, " v.0.1a");
+ else if (dwFT2 == 0x22)
+ strcat(szClientBuf, " v.0.2b");
+
+ szClient = szClientBuf;
+ }
+ else if (dwFT1 == dwFT2 && dwFT2 == dwFT3 && wVersion == 8)
+ {
+ if ((dwFT1 < dwOnlineSince + 3600) && (dwFT1 > (dwOnlineSince - 3600)))
+ {
+ szClient = cliSpamBot;
+ }
+ }
+ else if (!dwFT1 && !dwFT2 && !dwFT3 && !wVersion && !wLen && dwWebPort == 0x75BB)
+ {
+ szClient = cliSpamBot;
+ }
+ else if (dwFT1 == 0x44F523B0 && dwFT2 == 0x44F523A6 && dwFT3 == 0x44F523A6 && wVersion == 8)
+ {
+ szClient = "Virus";
+ }
+
+ { // capabilities based detection
+
+ if (dwUin && caps)
+ {
+ // check capabilities for client identification
+ if ((capId = MatchCap(caps, wLen, &capMirandaIm, 8))&&(!MatchCap(caps, wLen, &capIcqJSin, 4)&&!MatchCap(caps,wLen,&capIcqJs7, 4)&&!MatchCap(caps,wLen,&capIcqJp, 4)&&!MatchCap(caps,wLen, &capIcqJen, 4)))
+ { // new Miranda Signature
+ char sz[64];
+ static char szTmp[256];
+ BYTE mver0 = (*capId)[0x8];
+ BYTE mver1 = (*capId)[0x9];
+ BYTE mver2 = (*capId)[0xA];
+ BYTE mver3 = (*capId)[0xB];
+ BYTE iver0 = (*capId)[0xC];
+ BYTE iver1 = (*capId)[0xD];
+ BYTE iver2 = (*capId)[0xE];
+ BYTE iver3 = (*capId)[0xF];
+ strcpy(szTmp, "Miranda IM ");
+ if (MatchCap(caps,wLen,&capMirandaMobile,13))
+ lstrcatA(szTmp, "Mobile ");
+ if (mver0 == 0x80) // alpha builds directly stamped
+ {
+ if (mver2 == 0x00)
+ mir_snprintf(sz, sizeof(sz), "0.%u alpha build #%u", mver1, mver3); // 0.x alpha #x
+ else
+ mir_snprintf(sz, sizeof(sz), "0.%u.%u preview #%u", mver1, mver2,mver3); // 0.x.x preview #x
+ lstrcatA(szTmp, sz);
+ }
+ else // others, but also for those where bug caused that alpha flag was not send
+ {
+ if (mver2 == 0x00)
+ mir_snprintf(sz, sizeof(sz), "%u.%u", mver0, mver1); // 0.x final
+ else
+ mir_snprintf(sz, sizeof(sz), "%u.%u.%u", mver0, mver1, mver2); // 0.x.x final
+ lstrcatA(szTmp, sz);
+ if (mver3 != 0x00 && mver3 != 0x64) // if some build version is set, then it's probably alpha
+ {
+ mir_snprintf(sz, sizeof(sz), " alpha build #%u", mver3); // ".. alpha #x"
+ lstrcatA(szTmp, sz);
+ }
+ }
+ if (dwFT1 == 0x7fffffff ||(BYTE)((dwFT3>>24)&0xFF) == 0x80)
+ lstrcatA(szTmp, " Unicode");
+ lstrcatA(szTmp," (ICQ ");
+ if(MatchCap(caps,wLen,&capIcqJs7old, 0x10)||MatchCap(caps,wLen,&capIcqJs7s, 0x10)) //detecting sss & s7 old mod versions
+ lstrcatA(szTmp, " S7 & SSS (old)");
+ else switch (iver0) {
+ case 0x81 : lstrcatA(szTmp, " BM"); break; // detecting BM mod
+ break;
+ case 0x88 : lstrcatA(szTmp, " eternity (old)"); break; // eternity mod
+ default : break;
+ }
+ switch (iver2) {
+ case 0x58 : lstrcatA(szTmp, " eternity/PlusPlus++ Mod"); break; // detecting naturalized eternity/PlusPlus++ mod
+ default : break;
+ }
+ lstrcatA(szTmp, " ");
+ switch (iver0) {
+ case 0x88 : // eternity (old)
+ case 0x81 : // BM
+ case 0x80 : mir_snprintf(sz, sizeof(sz), "0.%u.%u.%u alpha)", iver1, iver2, iver3); // mir and bm 0.x
+ break;
+ default : mir_snprintf(sz, sizeof(sz), "%u.%u.%u.%u)",iver0, iver1, iver2, iver3); // other x.x (only for 6.6.6.0?)
+ }
+ lstrcatA(szTmp,sz);
+ if (dwFT3 == 0x5AFEC0DE||MatchCap(caps,wLen,&capIcqJs7s, 0x10))
+ lstrcatA(szTmp, " (SecureIM)");
+ szClient = szTmp;
+ *bClientId = CLID_MIRANDA;
+ }
+ else if (MatchCap(caps, wLen, &capTrillian, 0x10) || MatchCap(caps, wLen, &capTrilCrypt, 0x10))
+ { // this is Trillian, check for new versions
+ if (CheckContactCapabilities(hContact, CAPF_RTF))
+ {
+ if (CheckContactCapabilities(hContact, CAPF_AIM_FILE))
+ szClient = "Trillian Astra";
+ else
+ { // workaroud for a bug in Trillian - make it receive msgs, other features will not work!
+ ClearContactCapabilities(hContact, CAPF_SRV_RELAY);
+ szClient = "Trillian v3";
+ }
+ }
+ else
+ szClient = cliTrillian;
+ }
+ else if ((capId = MatchCap(caps, wLen, &capSimOld, 0xF)) && ((*capId)[0xF] != 0x92 && (*capId)[0xF] >= 0x20 || (*capId)[0xF] == 0))
+ {
+ int hiVer = (((*capId)[0xF]) >> 6) - 1;
+ unsigned loVer = (*capId)[0xF] & 0x1F;
+ if ((hiVer < 0) || ((hiVer == 0) && (loVer == 0)))
+ szClient = "Kopete";
+ else
+ {
+ null_snprintf(szClientBuf, 64, "SIM %u.%u", (unsigned)hiVer, loVer);
+ szClient = szClientBuf;
+ }
+ }
+ else if (capId = MatchCap(caps, wLen, &capClimm, 0xC))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ unsigned ver4 = (*capId)[0xF];
+
+ makeClientVersion(szClientBuf, "climm ", ver1, ver2, ver3, ver4);
+ if ((ver1 & 0x80) == 0x80)
+ strcat(szClientBuf, " alpha");
+ if (dwFT3 == 0x02000020)
+ strcat(szClientBuf, "/Win32");
+ else if (dwFT3 == 0x03000800)
+ strcat(szClientBuf, "/MacOS X");
+
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capSim, 0xC))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ unsigned ver4 = (*capId)[0xF];
+
+ makeClientVersion(szClientBuf, "SIM ", ver1, ver2, ver3, ver4 & 0x0F);
+ if (ver4 & 0x80)
+ strcat(szClientBuf,"/Win32");
+ else if (ver4 & 0x40)
+ strcat(szClientBuf,"/MacOS X");
+
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capLicq, 0xC))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD] % 100;
+ unsigned ver3 = (*capId)[0xE];
+
+ makeClientVersion(szClientBuf, cliLicqVer, ver1, ver2, ver3, 0);
+ if ((*capId)[0xF])
+ strcat(szClientBuf,"/SSL");
+
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capKopete, 0xC))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ unsigned ver4 = (*capId)[0xF];
+
+ makeClientVersion(szClientBuf, "Kopete ", ver1, ver2, ver3, ver4);
+
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capmIcq, 0xC))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ unsigned ver4 = (*capId)[0xF];
+
+ makeClientVersion(szClientBuf, "mICQ ", ver1, ver2, ver3, ver4);
+ if ((ver1 & 0x80) == 0x80) strcat(szClientBuf, " alpha");
+
+ szClient = szClientBuf;
+ }
+ else if (MatchCap(caps, wLen, &capIm2, 0x10))
+ { // IM2 v2 provides also Aim Icon cap
+ szClient = cliIM2;
+ }
+ else if (capId = MatchCap(caps, wLen, &capAndRQ, 9))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xB];
+ unsigned ver3 = (*capId)[0xA];
+ unsigned ver4 = (*capId)[9];
+
+ makeClientVersion(szClientBuf, "&RQ ", ver1, ver2, ver3, ver4);
+
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capRAndQ, 9))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xB];
+ unsigned ver3 = (*capId)[0xA];
+ unsigned ver4 = (*capId)[9];
+
+ makeClientVersion(szClientBuf, "R&Q ", ver1, ver2, ver3, ver4);
+
+ szClient = szClientBuf;
+ }
+ else if (MatchCap(caps, wLen, &capIMadering, 0x10))
+ { // http://imadering.com
+ szClient = "IMadering";
+ }
+ else if (MatchCap(caps, wLen, &capQipPDA, 0x10))
+ {
+ szClient = "QIP PDA (Windows)";
+ }
+ else if (MatchCap(caps, wLen, &capQipMobile, 0x10))
+ {
+ szClient = "QIP Mobile (Java)";
+ }
+ else if (MatchCap(caps,wLen,&capQipSymbian,0x10))
+ {
+ szClient = "QIP Mobile (Symbian)";
+ }
+ else if (MatchCap(caps, wLen, &capQipInfium, 0x10))
+ {
+ char ver[10];
+
+ strcpy(szClientBuf, "QIP Infium");
+ if (dwFT1)
+ { // add build
+ null_snprintf(ver, 10, " (%d)", dwFT1);
+ strcat(szClientBuf, ver);
+ }
+ if (dwFT2 == 0x0000000B)
+ strcat(szClientBuf, " Beta");
+
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capQip, 0xE))
+ {
+ char ver[10];
+
+ if (dwFT3 == 0x0F)
+ strcpy(ver, "2005");
+ else
+ {
+ strncpy(ver, (*capId)+11, 5);
+ ver[5] = '\0'; // fill in missing zero
+ }
+
+ null_snprintf(szClientBuf, 64, cliQip, ver);
+ if (dwFT1 && dwFT2 == 0x0E)
+ { // add QIP build
+ null_snprintf(ver, 10, " (%d%d%d%d)", dwFT1 >> 0x18, (dwFT1 >> 0x10) & 0xFF, (dwFT1 >> 0x08) & 0xFF, dwFT1 & 0xFF);
+ strcat(szClientBuf, ver);
+ }
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capmChat, 0xA))
+ {
+ strcpy(szClientBuf, "mChat ");
+ strncat(szClientBuf, (*capId) + 0xA, 6);
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capJimm, 5))
+ {
+ strcpy(szClientBuf, "Jimm ");
+ strncat(szClientBuf, (*capId) + 5, 11);
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capCorePager, 0xA))
+ { // http://corepager.net.ru/index/0-2
+ strcpy(szClientBuf, "CORE Pager");
+ if (dwFT2 == 0x0FFFF0011 && dwFT3 == 0x1100FFFF && (dwFT1 >> 0x18))
+ {
+ char ver[16];
+
+ null_snprintf(ver, 10, " %d.%d", dwFT1 >> 0x18, (dwFT1 >> 0x10) & 0xFF);
+ if ((dwFT1 & 0xFF) == 0x0B)
+ strcat(ver, " Beta");
+ strcat(szClientBuf, ver);
+ }
+ szClient = szClientBuf;
+ }
+ else if (MatchCap(caps, wLen, &capMacIcq, 0x10))
+ {
+ szClient = "ICQ for Mac";
+ }
+ else if (MatchCap(caps, wLen, &capUim, 0x10))
+ {
+ szClient = "uIM";
+ }
+ else if (MatchCap(caps, wLen, &capAnastasia, 0x10))
+ { // http://chis.nnov.ru/anastasia
+ szClient = "Anastasia";
+ }
+ else if (capId = MatchCap(caps, wLen, &capPalmJicq, 0xC))
+ { // http://www.jsoft.ru
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ unsigned ver4 = (*capId)[0xF];
+
+ makeClientVersion(szClientBuf, "JICQ ", ver1, ver2, ver3, ver4);
+
+ szClient = szClientBuf;
+ }
+ else if (MatchCap(caps, wLen, &capInluxMsgr, 0x10))
+ { // http://www.inlusoft.com
+ szClient = "Inlux Messenger";
+ }
+ else if (capId = MatchCap(caps, wLen, &capVmICQ, 5))
+ {
+ strcpy(szClientBuf, "VmICQ ");
+ strncat(szClientBuf, (*capId) + 5, 11);
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capSmapeR, 0x07))
+ { // http://www.smape.com/smaper
+ strcpy(szClientBuf, "SmapeR");
+ strncat(szClientBuf, (*capId) + 6, 10);
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capMipClient, 0x04))
+ { //http://mip.rufon.net - new signature
+ strcpy(szClientBuf, "MIP ");
+ strncat(szClientBuf, (*capId) + 4, 12);
+ szClient = szClientBuf;
+ }
+ else if(capId = MatchCap(caps,wLen,&capYapp, 4))
+ {
+ strcpy(szClientBuf, "Yapp! v");
+ strncat(szClientBuf, (*capId) + 8, 5);
+ szClient = szClientBuf;
+ }
+ else if(!wVersion && CheckContactCapabilities(hContact, CAPF_DIRECT|CAPF_RTF|CAPF_TYPING|CAPF_UTF))
+ szClient = "GlICQ";
+ else if (szClient == cliLibicq2k)
+ { // try to determine which client is behind libicq2000
+ if (CheckContactCapabilities(hContact, CAPF_RTF))
+ szClient = cliCentericq; // centericq added rtf capability to libicq2000
+ else if (CheckContactCapabilities(hContact, CAPF_UTF))
+ szClient = cliLibicqUTF; // IcyJuice added unicode capability to libicq2000
+ // others - like jabber transport uses unmodified library, thus cannot be detected
+ }
+ else if (szClient == NULL && !((capId = MatchCap(caps, wLen, &capIcqJs7, 4))||(capId = MatchCap(caps, wLen, &capIcqJSin, 4))||(capId = MatchCap(caps, wLen, &capIcqJp, 4))||(capId = MatchCap(caps, wLen, &capIcqJen, 4)))) // HERE ENDS THE SIGNATURE DETECTION, after this only feature default will be detected
+ {
+ if (wVersion == 8 && CheckContactCapabilities(hContact, CAPF_XTRAZ) && (MatchCap(caps, wLen, &capIMSecKey1, 6) || MatchCap(caps, wLen, &capIMSecKey2, 6)))
+ { // ZA mangled the version, OMG!
+ wVersion = 9;
+ }
+ if (wVersion == 8 && (MatchCap(caps, wLen, &capComm20012, 0x10) || CheckContactCapabilities(hContact, CAPF_SRV_RELAY)))
+ { // try to determine 2001-2003 versions
+ if (MatchCap(caps, wLen, &capIs2001, 0x10))
+ {
+ if (!dwFT1 && !dwFT2 && !dwFT3)
+ {
+ if (CheckContactCapabilities(hContact, CAPF_RTF))
+ szClient = "TICQClient"; // possibly also older GnomeICU
+ else
+ szClient = "ICQ for Pocket PC";
+ }
+ else
+ {
+ *bClientId = CLID_GENERIC;
+ szClient = "ICQ 2001";
+ }
+ }
+ else if (MatchCap(caps, wLen, &capIs2002, 0x10))
+ {
+ *bClientId = CLID_GENERIC;
+ szClient = "ICQ 2002";
+ }
+ else if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_RTF) && !MatchCap(caps, wLen, &capIcqJs7, 0x4) && !MatchCap(caps,wLen,&capIcqJSin, 0x4))
+ {
+ if (!dwFT1 && !dwFT2 && !dwFT3)
+ {
+ if (!dwWebPort)
+ szClient = "GnomeICU 0.99.5+"; // no other way
+ else
+ szClient = "IC@";
+ }
+ else
+ {
+ *bClientId = 0;
+ szClient = "ICQ 2002/2003a";
+ }
+ }
+ else if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_TYPING))
+ {
+ if (!dwFT1 && !dwFT2 && !dwFT3)
+ {
+ szClient = "PreludeICQ";
+ }
+ }
+ }
+ else if (wVersion == 9)
+ { // try to determine lite versions
+ if (CheckContactCapabilities(hContact, CAPF_XTRAZ))
+ {
+ *bClientId = CLID_GENERIC;
+ if (CheckContactCapabilities(hContact, CAPF_AIM_FILE))
+ {
+ if (MatchCap(caps, wLen, &captZers, 0x10))
+ { // capable of tZers ?
+ if (MatchCap(caps, wLen, &capHtmlMsgs, 0x10))
+ {
+ strcpy(szClientBuf, "ICQ 6");
+ *bClientId = CLID_ICQ6;
+ }
+ else
+ {
+ strcpy(szClientBuf, "icq5.1");
+ }
+ SetContactCapabilities(hContact, CAPF_STATUSMSGEXT);
+ }
+ else
+ {
+ strcpy(szClientBuf, "icq5");
+ }
+ if (MatchCap(caps, wLen, &capRambler, 0x10))
+ {
+ strcat(szClientBuf, " (Rambler)");
+ }
+ else if (MatchCap(caps, wLen, &capAbv, 0x10))
+ {
+ strcat(szClientBuf, " (Abv)");
+ }
+ else if (MatchCap(caps, wLen, &capNetvigator, 0x10))
+ {
+ strcat(szClientBuf, " (Netvigator)");
+ }
+ szClient = szClientBuf;
+ }
+ else if (!CheckContactCapabilities(hContact, CAPF_DIRECT))
+ if (CheckContactCapabilities(hContact, CAPF_RTF))
+ { // most probably Qnext - try to make that shit at least receiving our msgs
+ ClearContactCapabilities(hContact, CAPF_SRV_RELAY);
+ NetLog_Server("Forcing simple messages (QNext client).");
+ szClient = "QNext";
+ }
+ else
+ szClient = "pyICQ";
+ else
+ szClient = "ICQ Lite v4";
+ }
+ else if (!CheckContactCapabilities(hContact, CAPF_DIRECT))
+ {
+ if (CheckContactCapabilities(hContact, CAPF_UTF) && !CheckContactCapabilities(hContact, CAPF_RTF))
+ szClient = "pyICQ";
+ }
+ }
+ else if (wVersion == 7)
+ {
+ if (CheckContactCapabilities(hContact, CAPF_RTF))
+ szClient = "GnomeICU"; // this is an exception
+ else if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY))
+ {
+ if (!dwFT1 && !dwFT2 && !dwFT3)
+ szClient = "&RQ";
+ else
+ {
+ *bClientId = CLID_GENERIC;
+ szClient = "ICQ 2000";
+ }
+ }
+ else if (CheckContactCapabilities(hContact, CAPF_UTF))
+ {
+ if (CheckContactCapabilities(hContact, CAPF_TYPING))
+ szClient = "Icq2Go! (Java)";
+ else if (wClass & CLASS_WIRELESS)
+ szClient = "Pocket Web 1&1";
+ else
+ szClient = "Icq2Go! (Flash)"; // fixing overlay in Fingerprint
+ }
+ /* else if (CheckContactCapabilities(hContact, CAPF_UTF))
+ {
+ if (CheckContactCapabilities(hContact, CAPF_TYPING))
+ szClient = "Icq2Go! (Java)";
+ else
+ szClient = "Icq2Go! (Flash)";
+ }*/
+ }
+ else if (wVersion == 0xA)
+ {
+ if (!CheckContactCapabilities(hContact, CAPF_RTF) && !CheckContactCapabilities(hContact, CAPF_UTF))
+ { // this is bad, but we must do it - try to detect QNext
+ ClearContactCapabilities(hContact, CAPF_SRV_RELAY);
+ NetLog_Server("Forcing simple messages (QNext client).");
+ szClient = "QNext";
+ }
+ else if (!CheckContactCapabilities(hContact, CAPF_RTF) && CheckContactCapabilities(hContact, CAPF_UTF) && !dwFT1 && !dwFT2 && !dwFT3)
+ { // not really good, but no other option
+ szClient = "NanoICQ";
+ }
+ }
+ else if (wVersion == 0)
+ { // capability footprint based detection - not really reliable
+ if (!dwFT1 && !dwFT2 && !dwFT3 && !dwWebPort && !dwDirectCookie)
+ { // DC info is empty
+ if (CheckContactCapabilities(hContact, CAPF_TYPING) && MatchCap(caps, wLen, &capIs2001, 0x10) &&
+ MatchCap(caps, wLen, &capIs2002, 0x10) && MatchCap(caps, wLen, &capComm20012, 0x10))
+ szClient = cliSpamBot;
+ else if (MatchCap(caps, wLen, &capAimIcon, 0x10) && MatchCap(caps, wLen, &capAimDirect, 0x10) &&
+ CheckContactCapabilities(hContact, CAPF_AIM_FILE | CAPF_UTF))
+ { // detect libgaim
+ if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY))
+ szClient = "Adium X"; // yeah, AFAIK only Adium has this fixed
+ else
+ szClient = "libgaim";
+ }
+ else if (MatchCap(caps, wLen, &capAimIcon, 0x10) && MatchCap(caps, wLen, &capAimDirect, 0x10) &&
+ MatchCap(caps, wLen, &capAimChat, 0x10) && CheckContactCapabilities(hContact, CAPF_AIM_FILE) && wLen == 0x40)
+ szClient = "libgaim"; // Gaim 1.5.1 most probably
+ else if (MatchCap(caps, wLen, &capAimChat, 0x10) && CheckContactCapabilities(hContact, CAPF_AIM_FILE) && wLen == 0x20)
+ szClient = "Easy Message";
+ else if (MatchCap(caps, wLen, &capAimIcon, 0x10) && MatchCap(caps, wLen, &capAimChat, 0x10) && CheckContactCapabilities(hContact, CAPF_UTF) && wLen == 0x30)
+ szClient = "Meebo";
+ else if (MatchCap(caps, wLen, &capAimIcon, 0x10) && CheckContactCapabilities(hContact, CAPF_UTF) && wLen == 0x20)
+ szClient = "PyICQ-t Jabber Transport";
+ else if (MatchCap(caps, wLen, &capAimIcon, 0x10) && MatchCap(caps, wLen, &capIcqLite, 0x10) && CheckContactCapabilities(hContact, CAPF_UTF | CAPF_XTRAZ))
+ szClient = "PyICQ-t Jabber Transport";
+ else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_SRV_RELAY | CAPF_DIRECT | CAPF_TYPING) && wLen == 0x40)
+ szClient = "Agile Messenger"; // Smartphone 2002
+ else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_SRV_RELAY | CAPF_DIRECT | CAPF_AIM_FILE) && MatchCap(caps, wLen, &capAimFileShare, 0x10))
+ szClient = "Slick"; // http://lonelycatgames.com/?app=slick
+ }
+ }
+ }
+ }
+ else if (!dwUin)
+ { // detect AIM clients
+ if (caps)
+ {
+ if (capId = MatchCap(caps, wLen, &capAimOscar, 8))
+ { // AimOscar Signature
+ char sz[64];
+ static char szTmp[256];
+ BYTE mver0 = (*capId)[0x8];
+ BYTE mver1 = (*capId)[0x9];
+ BYTE mver2 = (*capId)[0xA];
+ BYTE mver3 = (*capId)[0xB];
+ BYTE iver0 = (*capId)[0xC];
+ BYTE iver1 = (*capId)[0xD];
+ BYTE iver2 = (*capId)[0xE];
+ BYTE iver3 = (*capId)[0xF];
+ strcpy(szTmp, "Miranda IM ");
+ if (mver0 & 0x80)
+ {
+ mir_snprintf(sz, sizeof(sz), "0.%u.%u alpha build #%u", mver1, mver2,mver3);
+ lstrcatA(szTmp, sz);
+ }
+ else
+ {
+ mir_snprintf(sz, sizeof(sz), "%u.%u.%u.%u", mver0, mver1, mver2,mver3);
+ lstrcatA(szTmp, sz);
+ }
+ lstrcatA(szTmp," (AIM v");
+ if (iver0 & 0x80)
+ {
+ mir_snprintf(sz, sizeof(sz), "0.%u.%u.%u alpha)", iver1, iver2, iver3);
+ lstrcatA(szTmp,sz);
+ }
+ else
+ {
+ mir_snprintf(sz, sizeof(sz), "%u.%u.%u.%u)",iver0, iver1, iver2, iver3);
+ lstrcatA(szTmp,sz);
+ }
+ /* if (dwFT3 == 0x5AFEC0DE) //I don't know if integratiion with SecureIM is done
+ lstrcatA(szTmp, " (with SecureIM)"); */
+ szClient = szTmp;
+ }
+ else if (capId = MatchCap(caps, wLen, &capSim, 0xC))
+ { // Sim is universal
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ makeClientVersion(szClientBuf, "SIM ", ver1, ver2, ver3, 0);
+ if ((*capId)[0xF] & 0x80)
+ strcat(szClientBuf,"/Win32");
+ else if ((*capId)[0xF] & 0x40)
+ strcat(szClientBuf,"/MacOS X");
+ szClient = szClientBuf;
+ }
+ else if (capId = MatchCap(caps, wLen, &capKopete, 0xC))
+ {
+ unsigned ver1 = (*capId)[0xC];
+ unsigned ver2 = (*capId)[0xD];
+ unsigned ver3 = (*capId)[0xE];
+ unsigned ver4 = (*capId)[0xF];
+ makeClientVersion(szClientBuf, "Kopete ", ver1, ver2, ver3, ver4);
+ szClient = szClientBuf;
+ }
+ else if (MatchCap(caps, wLen, &capIm2, 0x10))
+ { // IM2 extensions
+ szClient = cliIM2;
+ }
+ else if (MatchCap(caps, wLen, &capNaim, 0x8))
+ {
+ szClient = "naim";
+ }
+ else if(MatchCap(caps, wLen, &capAimIcon, 0x10))
+ {
+ if(MatchCap(caps, wLen, &capAimAddins, 0x10) && MatchCap(caps, wLen, &capAimUnk2, 0x10))
+ {
+ szClient = "AIM 5";
+ }
+ else if(MatchCap(caps, wLen, &capAimAudio, 0x10) && MatchCap(caps, wLen, &capAimContactSnd, 0x10))
+ {
+ szClient = "AIM Pro";
+ }
+ else if(MatchCap(caps, wLen, &capAimAudio, 0x10))
+ {
+ szClient = "AIM 6";
+ }
+ else
+ {
+ szClient = "AIM";
+ }
+ }
+ }
+ }
+ }
+ if (!szClient&&!MatchCap(caps, wLen, &capIcqJs7, 4)&&!MatchCap(caps, wLen, &capIcqJSin, 4)&&!MatchCap(caps, wLen, &capIcqJp, 4)&&!MatchCap(caps, wLen, &capIcqJen, 4))
+ {
+ NetLog_Server("No client identification, put default ICQ client for protocol.");
+ *bClientId = CLID_GENERIC;
+ switch (wVersion)
+ { // client detection failed, provide default clients
+ case 6:
+ szClient = "ICQ99";
+ break;
+ case 7:
+ szClient = "ICQ 2000/Icq2Go";
+ break;
+ case 8:
+ szClient = "ICQ 2001-2003a";
+ break;
+ case 9:
+ szClient = "ICQ Lite";
+ break;
+ case 0xA:
+ szClient = "ICQ 2003b";
+ break;
+ case 0xB:
+ szClient = "QIP 2005a";
+ break;
+ }
+ }
+ if (!szClient)
+ {
+ if ((capId = MatchCap(caps, wLen, &capIcqJs7, 4))||(capId = MatchCap(caps, wLen, &capIcqJSin, 4))||(capId = MatchCap(caps, wLen, &capIcqJp, 4))||(capId = MatchCap(caps, wLen, &capIcqJen, 4))) //detecting mod and core version, please fix me
+ {
+ char sz[64];
+ static char szTmp[256];
+ BYTE mver0 = (*capId)[0x4];
+ BYTE mver1 = (*capId)[0x5];
+ BYTE mver2 = (*capId)[0x6];
+ BYTE mver3 = (*capId)[0x7];
+ BYTE iver0 = (*capId)[0x8];
+ BYTE iver1 = (*capId)[0x9];
+ BYTE iver2 = (*capId)[0xA];
+ BYTE iver3 = (*capId)[0xB];
+ BYTE secure = (*capId)[0xC];
+ BYTE ModVer = (*capId)[0x3];
+ BYTE Unicode = (BYTE)(dwFT3>>24)&0xFF;
+ if (mver1 < 20 && iver1 < 20)
+ {
+ strcpy(szTmp, "Miranda IM ");
+ if (mver0 == 0x80)
+ {
+ if (mver2 == 0x00)
+ mir_snprintf(sz, sizeof(sz), "0.%u alpha build #%u", mver1, mver3);
+ else
+ mir_snprintf(sz, sizeof(sz), "0.%u.%u alpha build #%u", mver1, mver2,mver3);
+ lstrcatA(szTmp, sz);
+ }
+ else
+ {
+ if (mver2 == 0x00)
+ mir_snprintf(sz, sizeof(sz), "%u.%u", mver0, mver1);
+ else
+ mir_snprintf(sz, sizeof(sz), "%u.%u.%u", mver0, mver1, mver2);
+ lstrcatA(szTmp, sz);
+ if (mver3 != 0x00 && mver3 != 0x64)
+ {
+ mir_snprintf(sz, sizeof(sz), " alpha build #%u", mver3);
+ lstrcatA(szTmp, sz);
+ }
+ }
+ if (Unicode == 0x80||dwFT1 == 0x7fffffff)
+ lstrcatA(szTmp," Unicode");
+ if (ModVer == 'p')
+ lstrcatA(szTmp," (ICQ "ICQ_MODNAME"");
+ else if (((*capId)[0x0]) == 's')
+ lstrcatA(szTmp," (ICQ S!N");
+ else if (((*capId)[0x0]) == 'e')
+ lstrcatA(szTmp," (ICQ eternity/PlusPlus++"); //mod of mod of mod of mod of mod
+ else if(ModVer == 'j')
+ lstrcatA(szTmp," (ICQ S7 & SSS");
+ if (iver0 == 0x80)
+ {
+ mir_snprintf(sz, sizeof(sz), " 0.%u.%u.%u alpha)", iver1, iver2, iver3);
+ lstrcatA(szTmp,sz);
+ }
+ else
+ {
+ mir_snprintf(sz, sizeof(sz), " %u.%u.%u.%u)",iver0, iver1, iver2, iver3);
+ lstrcatA(szTmp,sz);
+ }
+ if (secure != 0 && secure != 20||dwFT3 == 0x5AFEC0DE)
+ lstrcatA(szTmp, " (SecureIM)");
+ }
+ else if(MatchCap(caps,wLen,&capIcqJs7s,0x10))
+ strcpy(szTmp,"Miranda IM (ICQ SSS & S7)(SecureIM)");
+ else if(MatchCap(caps,wLen,&capIcqJs7old,0x10))
+ strcpy(szTmp,"Miranda IM (ICQ SSS & S7)");
+ szClient = szTmp;
+ }
+ else if(CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_DIRECT | CAPF_TYPING) && wVersion > 1000)
+ {
+ szClient = "QIP 2005a (as Unknown)";
+ }
+ else if(CheckContactCapabilities(hContact, CAPF_UTF)&&!CheckContactCapabilities(hContact, CAPF_SRV_RELAY)&&!CheckContactCapabilities(hContact, CAPF_TYPING))//i hate pocket web ;)
+ szClient = "Pocket Web 1&1";
+ }
+ if (szClient)
+ {
+ char* szExtra = NULL;
+ if (MatchCap(caps, wLen, &capSimpLite, 0x10))
+ szExtra = " + SimpLite";
+ if (MatchCap(caps, wLen, &capSimpPro, 0x10))
+ szExtra = " + SimpPro";
+ if (MatchCap(caps, wLen, &capIMsecure, 0x10) || MatchCap(caps, wLen, &capIMSecKey1, 6) || MatchCap(caps, wLen, &capIMSecKey2, 6))
+ szExtra = " + IMsecure";
+ if (!strstr(szClient, "Miranda") && ((capId = MatchCap(caps, wLen, &capIcqJs7, 4))||(capId = MatchCap(caps, wLen, &capIcqJp, 4))||(capId = MatchCap(caps, wLen, &capIcqJSin, 4))||(capId = MatchCap(caps, wLen, &capIcqJen, 4)))) //detecting mod and core version, please fix me
+ {
+ char sz[64];
+ static char szTmp[256];
+ BYTE mver0 = (*capId)[0x4];
+ BYTE mver1 = (*capId)[0x5];
+ BYTE mver2 = (*capId)[0x6];
+ BYTE mver3 = (*capId)[0x7];
+ BYTE iver0 = (*capId)[0x8];
+ BYTE iver1 = (*capId)[0x9];
+ BYTE iver2 = (*capId)[0xA];
+ BYTE iver3 = (*capId)[0xB];
+ BYTE secure = (*capId)[0xC];
+ BYTE ModVer = (*capId)[0x3];
+ BYTE Unicode = (BYTE)(dwFT3>>24)&0xFF;
+ if (mver1 < 20 && iver1 < 20)
+ {
+ strcpy(szTmp, " (Miranda IM ");
+ if (mver0 & 0x80)
+ {
+ mir_snprintf(sz, sizeof(sz), "0.%u.%u alpha build #%u", mver1, mver2,mver3);
+ lstrcatA(szTmp, sz);
+ }
+ else
+ {
+ mir_snprintf(sz, sizeof(sz), "%u.%u.%u.%u", mver0, mver1, mver2,mver3);
+ lstrcatA(szTmp, sz);
+ }
+ if (Unicode == 0x80||dwFT1 == 0x7fffffff)
+ lstrcatA(szTmp," Unicode");
+ if (ModVer == 'p')
+ lstrcatA(szTmp," (ICQ "ICQ_MODNAME"");
+ else if (((*capId)[0x0]) == 's')
+ lstrcatA(szTmp," (ICQ S!N");
+ else if (((*capId)[0x0]) == 'e')
+ lstrcatA(szTmp," (ICQ eternity/PlusPlus++"); //mod of mod of mod of mod of mod
+ else if(ModVer == 'j')
+ lstrcatA(szTmp," (ICQ S7 & SSS");
+ if (iver0 == 0x80)
+ {
+ mir_snprintf(sz, sizeof(sz), " 0.%u.%u.%u alpha)", iver1, iver2, iver3);
+ lstrcatA(szTmp,sz);
+ }
+ else
+ {
+ mir_snprintf(sz, sizeof(sz), " %u.%u.%u.%u)",iver0, iver1, iver2, iver3);
+ lstrcatA(szTmp,sz);
+ }
+ lstrcatA(szTmp,")");
+ if (secure != 0 && secure != 20||dwFT3 == 0x5AFEC0DE)
+ lstrcatA(szTmp, " (SecureIM)");
+ }
+ if (iver1 >= 20 && mver1 >= 20||MatchCap(caps, wLen, &capIcqJs7old, 0x10))
+ {
+ strcpy(sz, " (Miranda IM (ICQ S7 & SSS))");
+ strcpy(szTmp,sz);
+ if (MatchCap(caps, wLen, &capIcqJs7s, 0x10))
+ lstrcatA(sz," (SecureIM))");
+ }
+ szExtra=szTmp;
+ }
+ if (szExtra)
+ {
+ if (szClient != szClientBuf)
+ {
+ strcpy(szClientBuf, szClient);
+ szClient = szClientBuf;
+ }
+ if (szClient != szExtra)
+ lstrcatA(szClient, szExtra);
+ }
+ }
+ { // custom miranda packs
+ char *capId;
+ if (capId = (char*)MatchCap(caps, wLen, &capMimPack, 4))
+ {
+ char szPack[16];
+ mir_snprintf(szPack, 16, " [%.12s]", capId+4);
+ lstrcatA(szClient, szPack);
+ }
+ }
+ NetLog_Server("Client identified as %s", szClient);
+// ICQWriteContactSettingUtf(hContact, "MirVer", szClient); //thi already fixed in other place
+ return szClient;
+}