// ---------------------------------------------------------------------------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-2004 Martin Цberg, Sam Kothari, Robert Rainwater // Copyright © 2004-2010 Joe Kucera // Copyright © 2012-2017 Miranda NG Team // // 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. // ----------------------------------------------------------------------------- // DESCRIPTION: // // Provides capability & signature based client detection // ----------------------------------------------------------------------------- #include "stdafx.h" const capstr capShortCaps = {0x09, 0x46, 0x00, 0x00, 0x4c, 0x7f, 0x11, 0xd1, 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}; // CAP_AIM_BUDDYICON static const char* makeClientVersion(char *szBuf, const char *szClient, unsigned v1, unsigned v2, unsigned v3, unsigned v4) { if (v4) mir_snprintf(szBuf, 64, "%s%u.%u.%u.%u", szClient, v1, v2, v3, v4); else if (v3) mir_snprintf(szBuf, 64, "%s%u.%u.%u", szClient, v1, v2, v3); else mir_snprintf(szBuf, 64, "%s%u.%u", szClient, v1, v2); return szBuf; } static void verToStr(char *szStr, int v) { char szVer[64]; makeClientVersion(szVer, "", (v >> 24) & 0x7F, (v >> 16) & 0xFF, (v >> 8) & 0xFF, v & 0xFF); mir_strcat(szStr, szVer); if (v & 0x80000000) mir_strcat(szStr, " alpha"); } static char* MirandaVersionToStringEx(char* szStr, int bUnicode, const char* szPlug, int v, int m) { if (!v) // this is not Miranda return nullptr; mir_strcpy(szStr, "Miranda IM "); if (!m && v == 1) verToStr(szStr, 0x80010200); else if (!m && (v & 0x7FFFFFFF) <= 0x030301) verToStr(szStr, v); else { if (m) { verToStr(szStr, m); mir_strcat(szStr, " "); } if (bUnicode) mir_strcat(szStr, "Unicode "); mir_strcat(szStr, "("); mir_strcat(szStr, szPlug); mir_strcat(szStr, " v"); verToStr(szStr, v); mir_strcat(szStr, ")"); } return szStr; } char* MirandaModToString(char* szStr, capstr* capId, int bUnicode, const char* szModName) { // decode icqj mod version DWORD mver = (*capId)[0x4] << 0x18 | (*capId)[0x5] << 0x10 | (*capId)[0x6] << 8 | (*capId)[0x7]; DWORD iver = (*capId)[0x8] << 0x18 | (*capId)[0x9] << 0x10 | (*capId)[0xA] << 8 | (*capId)[0xB]; DWORD scode = (*capId)[0xC] << 0x18 | (*capId)[0xD] << 0x10 | (*capId)[0xE] << 8 | (*capId)[0xF]; char *szClient = MirandaVersionToStringEx(szStr, bUnicode, szModName, iver, mver); if (scode == 0x5AFEC0DE) mir_strcat(szClient, " + SecureIM"); return szClient; } const capstr capMirandaIm = {'M', 'i', 'r', 'a', 'n', 'd', 'a', 'M', 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capMirandaNg = {'M', 'i', 'r', 'a', 'n', 'd', 'a', 'N', 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capIcqJs7 = {'i', 'c', 'q', 'j', ' ', 'S', 'e', 'c', 'u', 'r', 'e', ' ', 'I', 'M', 0, 0}; const capstr capIcqJSin = {'s', 'i', 'n', 'j', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; // Miranda ICQJ S!N const capstr capIcqJp = {'i', 'c', 'q', 'p', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capAimOscar = {'M', 'i', 'r', 'a', 'n', 'd', 'a', 'A', 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capMimMobile = {'M', 'i', 'r', 'a', 'n', 'd', 'a', 'M', 'o', 'b', 'i', 'l', 'e', 0, 0, 0}; const capstr capMimPack = {'M', 'I', 'M', '/', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; // Custom Miranda Pack const capstr capTrillian = {0x97, 0xb1, 0x27, 0x51, 0x24, 0x3c, 0x43, 0x34, 0xad, 0x22, 0xd6, 0xab, 0xf7, 0x3f, 0x14, 0x09}; const capstr capTrilCrypt = {0xf2, 0xe7, 0xc7, 0xf4, 0xfe, 0xad, 0x4d, 0xfb, 0xb2, 0x35, 0x36, 0x79, 0x8b, 0xdf, 0x00, 0x00}; const capstr capSim = {'S', 'I', 'M', ' ', 'c', 'l', 'i', 'e', 'n', 't', ' ', ' ', 0, 0, 0, 0}; const capstr capSimOld = {0x97, 0xb1, 0x27, 0x51, 0x24, 0x3c, 0x43, 0x34, 0xad, 0x22, 0xd6, 0xab, 0xf7, 0x3f, 0x14, 0x00}; const capstr capLicq = {'L', 'i', 'c', 'q', ' ', 'c', 'l', 'i', 'e', 'n', 't', ' ', 0, 0, 0, 0}; const capstr capKopete = {'K', 'o', 'p', 'e', 't', 'e', ' ', 'I', 'C', 'Q', ' ', ' ', 0, 0, 0, 0}; const capstr capmIcq = {'m', 'I', 'C', 'Q', ' ', 0xA9, ' ', 'R', '.', 'K', '.', ' ', 0, 0, 0, 0}; const capstr capClimm = {'c', 'l', 'i', 'm', 'm', 0xA9, ' ', 'R', '.', 'K', '.', ' ', 0, 0, 0, 0}; const capstr capAndRQ = {'&', 'R', 'Q', 'i', 'n', 's', 'i', 'd', 'e', 0, 0, 0, 0, 0, 0, 0}; const capstr capRAndQ = {'R', '&', 'Q', 'i', 'n', 's', 'i', 'd', 'e', 0, 0, 0, 0, 0, 0, 0}; const capstr capIMadering = {'I', 'M', 'a', 'd', 'e', 'r', 'i', 'n', 'g', ' ', 'C', 'l', 'i', 'e', 'n', 't'}; const capstr capmChat = {'m', 'C', 'h', 'a', 't', ' ', 'i', 'c', 'q', ' ', 0, 0, 0, 0, 0, 0}; const capstr capJimm = {'J', 'i', 'm', 'm', ' ', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capCorePager = {'C', 'O', 'R', 'E', ' ', 'P', 'a', 'g', 'e', 'r', 0, 0, 0, 0, 0, 0}; const capstr capDiChat = {'D', '[', 'i', ']', 'C', 'h', 'a', 't', ' ', 0, 0, 0, 0, 0, 0, 0}; const capstr capVmIcq = {'V', 'm', 'I', 'C', 'Q', ' ', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capSmapeR = {'S', 'm', 'a', 'p', 'e', 'r', ' ', 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capAnastasia = {0x44, 0xE5, 0xBF, 0xCE, 0xB0, 0x96, 0xE5, 0x47, 0xBD, 0x65, 0xEF, 0xD6, 0xA3, 0x7E, 0x36, 0x02}; const capstr capPalmJicq = {'J', 'I', 'C', 'Q', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capInluxMsgr = {0xA7, 0xE4, 0x0A, 0x96, 0xB3, 0xA0, 0x47, 0x9A, 0xB8, 0x45, 0xC9, 0xE4, 0x67, 0xC5, 0x6B, 0x1F}; const capstr capYapp = {'Y', 'a', 'p', 'p', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capMipClient = {0x4d, 0x49, 0x50, 0x20, 0x43, 0x6c, 0x69, 0x65, 0x6e, 0x74, 0x20, 0x76, 0x00, 0x00, 0x00, 0x00}; const capstr capPigeon = {'P', 'I', 'G', 'E', 'O', 'N', '!', 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capDigsbyBeta = {0x09, 0x46, 0x01, 0x05, 0x4c, 0x7f, 0x11, 0xd1, 0x82, 0x22, 0x44, 0x45, 0x45, 0x53, 0x54, 0x00}; const capstr capDigsby = {'d', 'i', 'g', 's', 'b', 'y', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capJapp = {0x6a, 0x61, 0x70, 0x70, 0xa9, 0x20, 0x62, 0x79, 0x20, 0x53, 0x65, 0x72, 0x67, 0x6f, 0x00, 0x00}; const capstr capNaim = {0xFF, 0xFF, 0xFF, 0xFF, 'n', 'a', 'i', 'm', 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capCitron = {0x09, 0x19, 0x19, 0x82, 0xde, 0xad, 0xbe, 0xef, 0xca, 0xfe, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}; const capstr capQip = {0x56, 0x3F, 0xC8, 0x09, 0x0B, 0x6F, 0x41, 'Q', 'I', 'P', ' ', '2', '0', '0', '5', 'a'}; const capstr capQipPDA = {0x56, 0x3F, 0xC8, 0x09, 0x0B, 0x6F, 0x41, 'Q', 'I', 'P', ' ', ' ', ' ', ' ', ' ', '!'}; const capstr capQipSymbian = {0x51, 0xAD, 0xD1, 0x90, 0x72, 0x04, 0x47, 0x3D, 0xA1, 0xA1, 0x49, 0xF4, 0xA3, 0x97, 0xA4, 0x1F}; const capstr capQipIphone = {0x60, 0xDE, 0x5C, 0x8A, 0xDF, 0x8C, 0x4E, 0x1D, 0xA4, 0xC8, 0xBC, 0x3B, 0xD9, 0x79, 0x4D, 0xD8}; const capstr capQipMobile = {0xB0, 0x82, 0x62, 0xF6, 0x7F, 0x7C, 0x45, 0x61, 0xAD, 0xC1, 0x1C, 0x6D, 0x75, 0x70, 0x5E, 0xC5}; const capstr capQipInfium = {0x7C, 0x73, 0x75, 0x02, 0xC3, 0xBE, 0x4F, 0x3E, 0xA6, 0x9F, 0x01, 0x53, 0x13, 0x43, 0x1E, 0x1A}; const capstr capQip2010 = {0x7A, 0x7B, 0x7C, 0x7D, 0x7E, 0x7F, 0x0A, 0x03, 0x0B, 0x04, 0x01, 0x53, 0x00, 0x00, 0x00, 0x00}; const capstr capQip2012 = {0x7F, 0x7F, 0x7C, 0x7D, 0x7E, 0x7F, 0x0A, 0x03, 0x0B, 0x04, 0x01, 0x53, 0x13, 0x43, 0x1E, 0x1A}; const capstr capIm2 = {0x74, 0xED, 0xC3, 0x36, 0x44, 0xDF, 0x48, 0x5B, 0x8B, 0x1C, 0x67, 0x1A, 0x1F, 0x86, 0x09, 0x9F}; // IM2 Ext Msg const capstr capQutIm = {'q', 'u', 't', 'i', 'm', 0x30, 0x2e, 0x31, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; const capstr capBayan = {'b', 'a', 'y', 'a', 'n', 'I', 'C', 'Q', 0, 0, 0, 0, 0, 0, 0, 0}; const capstr capJabberJIT = {'J', 'I', 'T', ' ', 0x76, 0x2E, 0x31, 0x2E, 0x78, 0x2E, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00}; const capstr capIcqKid2 = {'I', 'c', 'q', 'K', 'i', 'd', '2', 0x00, 0x05, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; const capstr capWebIcqPro = {'W', 'e', 'b', 'I', 'c', 'q', 'P', 'r', 'o', ' ', 0, 0, 0, 0, 0, 0}; const capstr capJasmine = {0x4A, 0x61, 0x73, 0x6D, 0x69, 0x6E, 0x65, 0x20, 0x76, 0x65, 0x72, 0xFF, 0x00, 0x00, 0x00, 0x00}; const capstr capMraJava = {0x4a, 0x32, 0x4d, 0x45, 0x20, 0x6d, 0x40, 0x61, 0x67, 0x65, 0x6e, 0x74, 0x00, 0x00, 0x00, 0x00}; const capstr capMacIcq = {0xdd, 0x16, 0xf2, 0x02, 0x84, 0xe6, 0x11, 0xd4, 0x90, 0xdb, 0x00, 0x10, 0x4b, 0x9b, 0x4b, 0x7d}; const capstr capIs2001 = {0x2e, 0x7a, 0x64, 0x75, 0xfa, 0xdf, 0x4d, 0xc8, 0x88, 0x6f, 0xea, 0x35, 0x95, 0xfd, 0xb6, 0xdf}; const capstr capIs2002 = {0x10, 0xcf, 0x40, 0xd1, 0x4c, 0x7f, 0x11, 0xd1, 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}; const capstr capComm20012 = {0xa0, 0xe9, 0x3f, 0x37, 0x4c, 0x7f, 0x11, 0xd1, 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}; const capstr capStrIcq = {0xa0, 0xe9, 0x3f, 0x37, 0x4f, 0xe9, 0xd3, 0x11, 0xbc, 0xd2, 0x00, 0x04, 0xac, 0x96, 0xdd, 0x96}; const capstr capIcqLiteNew = {0xc8, 0x95, 0x3a, 0x9f, 0x21, 0xf1, 0x4f, 0xaa, 0xb0, 0xb2, 0x6d, 0xe6, 0x63, 0xab, 0xf5, 0xb7}; const capstr capXtrazVideo = {0x17, 0x8C, 0x2D, 0x9B, 0xDA, 0xA5, 0x45, 0xBB, 0x8D, 0xDB, 0xF3, 0xBD, 0xBD, 0x53, 0xA1, 0x0A}; const capstr capOscarChat = {0x74, 0x8F, 0x24, 0x20, 0x62, 0x87, 0x11, 0xD1, 0x82, 0x22, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}; const capstr capUim = {0xA7, 0xE4, 0x0A, 0x96, 0xB3, 0xA0, 0x47, 0x9A, 0xB8, 0x45, 0xC9, 0xE4, 0x67, 0xC5, 0x6B, 0x1F}; const capstr capRambler = {0x7E, 0x11, 0xB7, 0x78, 0xA3, 0x53, 0x49, 0x26, 0xA8, 0x02, 0x44, 0x73, 0x52, 0x08, 0xC4, 0x2A}; const capstr capAbv = {0x00, 0xE7, 0xE0, 0xDF, 0xA9, 0xD0, 0x4F, 0xe1, 0x91, 0x62, 0xC8, 0x90, 0x9A, 0x13, 0x2A, 0x1B}; const capstr capNetvigator = {0x4C, 0x6B, 0x90, 0xA3, 0x3D, 0x2D, 0x48, 0x0E, 0x89, 0xD6, 0x2E, 0x4B, 0x2C, 0x10, 0xD9, 0x9F}; const capstr captZers = {0xb2, 0xec, 0x8f, 0x16, 0x7c, 0x6f, 0x45, 0x1b, 0xbd, 0x79, 0xdc, 0x58, 0x49, 0x78, 0x88, 0xb9}; // CAP_TZERS const capstr capSimpLite = {0x53, 0x49, 0x4D, 0x50, 0x53, 0x49, 0x4D, 0x50, 0x53, 0x49, 0x4D, 0x50, 0x53, 0x49, 0x4D, 0x50}; const capstr capSimpPro = {0x53, 0x49, 0x4D, 0x50, 0x5F, 0x50, 0x52, 0x4F, 0x53, 0x49, 0x4D, 0x50, 0x5F, 0x50, 0x52, 0x4F}; const capstr capIMsecure = {'I', 'M', 's', 'e', 'c', 'u', 'r', 'e', 'C', 'p', 'h', 'r', 0x00, 0x00, 0x06, 0x01}; // ZoneLabs const capstr capIMSecKey1 = {1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; // ZoneLabs const capstr capIMSecKey2 = {2, 2, 2, 2, 2, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; // ZoneLabs const capstr capFakeHtml = {0x01, 0x38, 0xca, 0x7b, 0x76, 0x9a, 0x49, 0x15, 0x88, 0xf2, 0x13, 0xfc, 0x00, 0x97, 0x9e, 0xa8}; const capstr capIcqIos = {0x09, 0x46, 0x13, 0x52, 0x4c, 0x7f, 0x11, 0xd1, 0x82, 0x22, 'D', 'E', 'S', 'T', 0x00, 0x00}; const capstr capIcqAndroid = {0x09, 0x46, 0x13, 0x53, 0x4c, 0x7f, 0x11, 0xd1, 0x82, 0x22, 'D', 'E', 'S', 'T', 0x00, 0x00}; const shortcapstr capAimIcon = { 0x13, 0x46 }; // CAP_AIM_BUDDYICON const shortcapstr capAimDirect = { 0x13, 0x45 }; // CAP_AIM_DIRECTIM const shortcapstr capAimFileShare = { 0x13, 0x48 }; // CAP_AIM_FILE_SHARE const shortcapstr capIcqDevils = { 0x13, 0x4C }; // CAP_DEVILS const shortcapstr capAimSmartCaps = { 0x01, 0xFF }; const shortcapstr capAimLiveVideo = { 0x01, 0x01 }; // CAP_AIM_LIVE_VIDEO const shortcapstr capAimLiveAudio = { 0x01, 0x04 }; // CAP_AIM_LIVE_AUDIO const shortcapstr capStatusTextAware = { 0x01, 0x0A }; // CAP_HOST_STATUS_TEXT_AWARE const char* cliLibicq2k = "libicq2000"; const char* cliLicqVer = "Licq "; const char* cliCentericq = "Centericq"; const char* cliLibicqUTF = "libicq2000 (Unicode)"; const char* cliTrillian = "Trillian"; const char* cliTrillian4 = "Trillian Astra"; const char* cliQip = "QIP %s"; const char* cliIM2 = "IM2"; const char* cliSpamBot = "Spam Bot"; const char* CIcqProto::detectUserClient( MCONTACT hContact, int nIsICQ, WORD wUserClass, DWORD dwOnlineSince, const char *szCurrentClient, WORD wVersion, DWORD dwFT1, DWORD dwFT2, DWORD dwFT3, DWORD dwDirectCookie, DWORD dwWebPort, /* ICQ specific */ BYTE *caps, size_t wLen, /* Client capabilities */ BYTE *bClientId, /* Output: detected client-type */ char *szClientBuf) { LPCSTR szClient = nullptr; int bMirandaIM = FALSE; *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 { // Yes this is most probably Miranda, get the version info szClient = MirandaVersionToStringEx(szClientBuf, 0, "ICQ", dwFT2, 0); *bClientId = CLID_MIRANDA; bMirandaIM = TRUE; } } else if (dwFT1 == 0x7fffffff) { // This is Miranda with unicode core szClient = MirandaVersionToStringEx(szClientBuf, 1, "ICQ", dwFT2, 0); *bClientId = CLID_MIRANDA; bMirandaIM = TRUE; } else if ((dwFT1 & 0xFF7F0000) == 0x7D000000) { // This is probably an Licq client DWORD ver = dwFT1 & 0xFFFF; szClient = makeClientVersion(szClientBuf, cliLicqVer, ver / 1000, (ver / 10) % 100, ver % 10, 0); if (dwFT1 & 0x00800000) mir_strcat(szClientBuf, "/SSL"); } 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; szClient = makeClientVersion(szClientBuf, "Alicq ", ver1, ver2, ver3, 0); } else if (dwFT1 == 0xFFFFFF7F) szClient = "&RQ"; else if (dwFT1 == 0xFFFFF666) { // this is R&Q (Rapid Edition) mir_snprintf(szClientBuf, 64, "R&Q %u", (unsigned)dwFT2); szClient = szClientBuf; } 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 == 0x66666666 && dwFT3 == 0x66666666) { // http://darkjimm.ucoz.ru/ if (dwFT2 == 0x10000) { mir_strcpy(szClientBuf, "D[i]Chat v."); mir_strcat(szClientBuf, "0.1a"); } else { makeClientVersion(szClientBuf, "D[i]Chat v.", (dwFT2 >> 8) & 0x0F, (dwFT2 >> 4) & 0x0F, 0, 0); if ((dwFT2 & 0x0F) == 1) mir_strcat(szClientBuf, " alpha"); else if ((dwFT2 & 0x0F) == 2) mir_strcat(szClientBuf, " beta"); else if ((dwFT2 & 0x0F) == 3) mir_strcat(szClientBuf, " final"); } 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 capstr *capId; char ver[16]; if (nIsICQ && caps) { // check capabilities for client identification if (capId = MatchCapability(caps, wLen, &capMirandaIm, 8)) { // new Miranda Signature DWORD iver = (*capId)[0xC] << 0x18 | (*capId)[0xD] << 0x10 | (*capId)[0xE] << 8 | (*capId)[0xF]; DWORD mver = (*capId)[0x8] << 0x18 | (*capId)[0x9] << 0x10 | (*capId)[0xA] << 8 | (*capId)[0xB]; szClient = MirandaVersionToStringEx(szClientBuf, dwFT1 == 0x7fffffff, "ICQ", iver, mver); if (MatchCapability(caps, wLen, &capIcqJs7, 0x4)) { // detect mod mir_strcat(szClientBuf, " (s7 & sss)"); if (MatchCapability(caps, wLen, &capIcqJs7, 0xE)) mir_strcat(szClientBuf, " + SecureIM"); } else if ((dwFT1 & 0x7FFFFFFF) == 0x7FFFFFFF) { if (MatchCapability(caps, wLen, &capMimMobile)) mir_strcat(szClientBuf, " (Mobile)"); if (dwFT3 == 0x5AFEC0DE) mir_strcat(szClientBuf, " + SecureIM"); } *bClientId = CLID_MIRANDA; bMirandaIM = TRUE; } else if (capId = MatchCapability(caps, wLen, &capMirandaNg, 8)) { WORD v[4]; BYTE *buf = *capId + 8; unpackWord(&buf, &v[0]); unpackWord(&buf, &v[1]); unpackWord(&buf, &v[2]); unpackWord(&buf, &v[3]); mir_snprintf(szClientBuf, MAX_PATH, "Miranda NG ICQ %d.%d.%d.%d", v[0], v[1], v[2], v[3]); szClient = szClientBuf; if ((dwFT1 & 0x7FFFFFFF) == 0x7FFFFFFF && dwFT3 == 0x5AFEC0DE) mir_strcat(szClientBuf, " + SecureIM"); *bClientId = CLID_MIRANDA; bMirandaIM = TRUE; } else if (capId = MatchCapability(caps, wLen, &capIcqJs7, 4)) { // detect newer icqj mod szClient = MirandaModToString(szClientBuf, capId, dwFT3 == 0x80000000, "ICQ S7 & SSS"); bMirandaIM = TRUE; } else if (capId = MatchCapability(caps, wLen, &capIcqJSin, 4)) { // detect newer icqj mod szClient = MirandaModToString(szClientBuf, capId, dwFT3 == 0x80000000, "ICQ S!N"); bMirandaIM = TRUE; } else if (capId = MatchCapability(caps, wLen, &capIcqJp, 4)) { // detect icqj plus mod szClient = MirandaModToString(szClientBuf, capId, dwFT3 == 0x80000000, "ICQ Plus"); bMirandaIM = TRUE; } else if (capId = MatchCapability(caps, wLen, &capJasmine, 12)) { BYTE *p = (*capId) + 12; szClient = makeClientVersion(szClientBuf, "Jasmine IM v", p[0], p[1], p[2], p[3]); } else if (capId = MatchCapability(caps, wLen, &capMraJava, 12)) { unsigned ver1 = (*capId)[13]; unsigned ver2 = (*capId)[14]; szClient = makeClientVersion(szClientBuf, "Mail.ru Agent (Java) v", ver1, ver2, 0, 0); } else if (MatchCapability(caps, wLen, &capTrillian) || MatchCapability(caps, wLen, &capTrilCrypt)) { // this is Trillian, check for new versions if (CheckContactCapabilities(hContact, CAPF_RTF)) { if (CheckContactCapabilities(hContact, CAPF_OSCAR_FILE)) szClient = cliTrillian4; 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 if (MatchCapability(caps, wLen, &capFakeHtml) || CheckContactCapabilities(hContact, CAPF_OSCAR_FILE)) szClient = cliTrillian4; else szClient = cliTrillian; } else if ((capId = MatchCapability(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 szClient = makeClientVersion(szClientBuf, "SIM ", (unsigned)hiVer, loVer, 0, 0); } else if (capId = MatchCapability(caps, wLen, &capSim, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; szClient = makeClientVersion(szClientBuf, "SIM ", ver1, ver2, ver3, ver4 & 0x0F); if (ver4 & 0x80) mir_strcat(szClientBuf, "/Win32"); else if (ver4 & 0x40) mir_strcat(szClientBuf, "/MacOS X"); } else if (capId = MatchCapability(caps, wLen, &capLicq, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD] % 100; unsigned ver3 = (*capId)[0xE]; szClient = makeClientVersion(szClientBuf, cliLicqVer, ver1, ver2, ver3, 0); if ((*capId)[0xF]) mir_strcat(szClientBuf, "/SSL"); } else if (capId = MatchCapability(caps, wLen, &capKopete, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; szClient = makeClientVersion(szClientBuf, "Kopete ", ver1, ver2, ver3, ver4); } else if (capId = MatchCapability(caps, wLen, &capClimm, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; szClient = makeClientVersion(szClientBuf, "climm ", ver1, ver2, ver3, ver4); if ((ver1 & 0x80) == 0x80) mir_strcat(szClientBuf, " alpha"); if (dwFT3 == 0x02000020) mir_strcat(szClientBuf, "/Win32"); else if (dwFT3 == 0x03000800) mir_strcat(szClientBuf, "/MacOS X"); } else if (capId = MatchCapability(caps, wLen, &capmIcq, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; szClient = makeClientVersion(szClientBuf, "mICQ ", ver1, ver2, ver3, ver4); if ((ver1 & 0x80) == 0x80) mir_strcat(szClientBuf, " alpha"); } // IM2 v2 provides also Aim Icon cap else if (MatchCapability(caps, wLen, &capIm2)) szClient = cliIM2; else if (capId = MatchCapability(caps, wLen, &capAndRQ, 9)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xB]; unsigned ver3 = (*capId)[0xA]; unsigned ver4 = (*capId)[9]; szClient = makeClientVersion(szClientBuf, "&RQ ", ver1, ver2, ver3, ver4); } else if (capId = MatchCapability(caps, wLen, &capRAndQ, 9)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xB]; unsigned ver3 = (*capId)[0xA]; unsigned ver4 = (*capId)[9]; szClient = makeClientVersion(szClientBuf, "R&Q ", ver1, ver2, ver3, ver4); } // http://imadering.com else if (MatchCapability(caps, wLen, &capIMadering)) szClient = "IMadering"; else if (MatchCapability(caps, wLen, &capQipPDA)) szClient = "QIP PDA (Windows)"; else if (MatchCapability(caps, wLen, &capQipSymbian)) szClient = "QIP PDA (Symbian)"; else if (MatchCapability(caps, wLen, &capQipIphone)) szClient = "QIP Mobile (IPhone)"; else if (MatchCapability(caps, wLen, &capQipMobile)) szClient = "QIP Mobile (Java)"; else if (MatchCapability(caps, wLen, &capQipInfium)) { mir_strcpy(szClientBuf, "QIP Infium"); if (dwFT1) { mir_snprintf(ver, " (%d)", dwFT1); mir_strcat(szClientBuf, ver); } if (dwFT2 == 0x0B) mir_strcat(szClientBuf, " Beta"); szClient = szClientBuf; } else if (MatchCapability(caps, wLen, &capQip2010, 12)) { mir_strcpy(szClientBuf, "QIP 2010"); if (dwFT1) { mir_snprintf(ver, " (%d)", dwFT1); mir_strcat(szClientBuf, ver); } szClient = szClientBuf; } else if (MatchCapability(caps, wLen, &capQip2012, 12)) { mir_strcpy(szClientBuf, "QIP 2012"); if (dwFT1) { mir_snprintf(ver, " (%d)", dwFT1); mir_strcat(szClientBuf, ver); } szClient = szClientBuf; } else if (capId = MatchCapability(caps, wLen, &capQip, 0xE)) { if (dwFT3 == 0x0F) mir_strcpy(ver, "2005"); else null_strcpy(ver, (char*)(*capId) + 11, 5); mir_snprintf(szClientBuf, 64, cliQip, ver); if (dwFT1 && dwFT2 == 0x0E) { mir_snprintf(ver, " (%d%d%d%d)", dwFT1 >> 0x18, (dwFT1 >> 0x10) & 0xFF, (dwFT1 >> 0x08) & 0xFF, dwFT1 & 0xFF); mir_strcat(szClientBuf, ver); } szClient = szClientBuf; } else if (capId = MatchCapability(caps, wLen, &capmChat, 0xA)) { mir_strcpy(szClientBuf, "mChat "); mir_strncat(szClientBuf, (char*)(*capId) + 0xA, 6); szClient = szClientBuf; } else if (capId = MatchCapability(caps, wLen, &capJimm, 5)) { mir_strcpy(szClientBuf, "Jimm "); mir_strncat(szClientBuf, (char*)(*capId) + 5, 11); szClient = szClientBuf; } // http://corepager.net.ru/index/0-2 else if (capId = MatchCapability(caps, wLen, &capCorePager, 0xA)) { mir_strcpy(szClientBuf, "CORE Pager"); if (dwFT2 == 0x0FFFF0011 && dwFT3 == 0x1100FFFF && (dwFT1 >> 0x18)) { mir_snprintf(ver, " %d.%d", dwFT1 >> 0x18, (dwFT1 >> 0x10) & 0xFF); if ((dwFT1 & 0xFF) == 0x0B) mir_strcat(ver, " Beta"); mir_strcat(szClientBuf, ver); } szClient = szClientBuf; } // http://darkjimm.ucoz.ru/ else if (capId = MatchCapability(caps, wLen, &capDiChat, 9)) { mir_strcpy(szClientBuf, "D[i]Chat"); mir_strncat(szClientBuf, (char*)(*capId) + 8, 8); szClient = szClientBuf; } else if (MatchCapability(caps, wLen, &capMacIcq)) szClient = "ICQ for Mac"; else if (MatchCapability(caps, wLen, &capUim)) szClient = "uIM"; // http://chis.nnov.ru/anastasia else if (MatchCapability(caps, wLen, &capAnastasia)) szClient = "Anastasia"; // http://www.jsoft.ru else if (capId = MatchCapability(caps, wLen, &capPalmJicq, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; szClient = makeClientVersion(szClientBuf, "JICQ ", ver1, ver2, ver3, ver4); } // http://www.inlusoft.com else if (MatchCapability(caps, wLen, &capInluxMsgr)) szClient = "Inlux Messenger"; // http://mip.rufon.net else if (capId = MatchCapability(caps, wLen, &capMipClient, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; if (ver1 < 30) makeClientVersion(szClientBuf, "MIP ", ver1, ver2, ver3, ver4); else { mir_strcpy(szClientBuf, "MIP "); mir_strncat(szClientBuf, (char*)(*capId) + 11, 5); } szClient = szClientBuf; } //http://mip.rufon.net - new signature else if (capId = MatchCapability(caps, wLen, &capMipClient, 0x04)) { mir_strcpy(szClientBuf, "MIP "); mir_strncat(szClientBuf, (char*)(*capId) + 4, 12); szClient = szClientBuf; } else if (capId = MatchCapability(caps, wLen, &capVmIcq, 0x06)) { mir_strcpy(szClientBuf, "VmICQ"); mir_strncat(szClientBuf, (char*)(*capId) + 5, 11); szClient = szClientBuf; } // http://www.smape.com/smaper else if (capId = MatchCapability(caps, wLen, &capSmapeR, 0x07)) { mir_strcpy(szClientBuf, "SmapeR"); mir_strncat(szClientBuf, (char*)(*capId) + 6, 10); szClient = szClientBuf; } // http://yapp.ru else if (capId = MatchCapability(caps, wLen, &capYapp, 0x04)) { mir_strcpy(szClientBuf, "Yapp! v"); mir_strncat(szClientBuf, (char*)(*capId) + 8, 5); szClient = szClientBuf; } // http://www.dibsby.com (newer builds) else if (MatchCapability(caps, wLen, &capDigsby, 0x06)) szClient = "Digsby"; // http://www.digsby.com - probably by mistake (feature detection as well) else if (MatchCapability(caps, wLen, &capDigsbyBeta)) szClient = "Digsby"; // http://www.japp.org.ua else if (MatchCapability(caps, wLen, &capJapp)) szClient = "japp"; // http://pigeon.vpro.ru else if (MatchCapability(caps, wLen, &capPigeon, 0x07)) szClient = "PIGEON!"; // http://www.qutim.org else if (capId = MatchCapability(caps, wLen, &capQutIm, 0x05)) { if ((*capId)[0x6] == 0x2E) { // old qutim id unsigned ver1 = (*capId)[0x5] - 0x30; unsigned ver2 = (*capId)[0x7] - 0x30; makeClientVersion(szClientBuf, "qutIM ", ver1, ver2, 0, 0); } else { // new qutim id unsigned ver1 = (*capId)[0x6]; unsigned ver2 = (*capId)[0x7]; unsigned ver3 = (*capId)[0x8]; unsigned ver4 = ((*capId)[0x9] << 8) || (*capId)[0xA]; makeClientVersion(szClientBuf, "qutIM ", ver1, ver2, ver3, ver4); switch ((*capId)[0x5]) { case 'l': mir_strcat(szClientBuf, "/Linux"); break; case 'w': mir_strcat(szClientBuf, "/Win32"); break; case 'm': mir_strcat(szClientBuf, "/MacOS X"); break; } } szClient = szClientBuf; } // http://www.barobin.com/bayanICQ.html else if (capId = MatchCapability(caps, wLen, &capBayan, 8)) { mir_strcpy(szClientBuf, "bayanICQ "); mir_strncat(szClientBuf, (char*)(*capId) + 8, 5); szClient = szClientBuf; } else if (capId = MatchCapability(caps, wLen, &capJabberJIT, 0x04)) szClient = "Jabber ICQ Transport"; // http://sourceforge.net/projects/icqkid2 else if (capId = MatchCapability(caps, wLen, &capIcqKid2, 0x07)) { unsigned ver1 = (*capId)[0x7]; unsigned ver2 = (*capId)[0x8]; unsigned ver3 = (*capId)[0x9]; unsigned ver4 = (*capId)[0xA]; szClient = makeClientVersion(szClientBuf, "IcqKid2 v", ver1, ver2, ver3, ver4); } // http://intrigue.ru/workshop/webicqpro/webicqpro.html else if (capId = MatchCapability(caps, wLen, &capWebIcqPro, 0x0A)) szClient = "WebIcqPro"; // http://www.citron-im.com else if (capId = MatchCapability(caps, wLen, &capCitron)) szClient = "Citron IM"; // try to determine which client is behind libicq2000 else if (szClient == cliLibicq2k) { 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 } // THE SIGNATURE DETECTION ENDS HERE, after this only feature default will be detected else if (szClient == nullptr) { // ZA mangled the version, OMG! if (wVersion == 8 && CheckContactCapabilities(hContact, CAPF_XTRAZ) && (MatchCapability(caps, wLen, &capIMSecKey1, 6) || MatchCapability(caps, wLen, &capIMSecKey2, 6))) wVersion = ICQ_VERSION; // try to determine 2001-2003 versions if (wVersion == 8 && (MatchCapability(caps, wLen, &capComm20012) || CheckContactCapabilities(hContact, CAPF_SRV_RELAY))) { if (MatchCapability(caps, wLen, &capIs2001)) { 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 (MatchCapability(caps, wLen, &capIs2002)) { *bClientId = CLID_GENERIC; szClient = "ICQ 2002"; } else if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_RTF)) { if (!dwFT1 && !dwFT2 && !dwFT3) { if (!dwWebPort) szClient = "GnomeICU 0.99.5+"; // no other way else szClient = "IC@"; } else { *bClientId = CLID_GENERIC; szClient = "ICQ 2002/2003a"; } } // libpurple (e.g. Pidgin 2.7.x) else if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_TYPING | CAPF_XTRAZ) && MatchCapability(caps, wLen, &capOscarChat) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchCapability(caps, wLen, &capFakeHtml)) { if (MatchShortCapability(caps, wLen, &capAimDirect)) szClient = "libpurple"; else szClient = "Meebo"; } else if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_TYPING)) { if (!dwFT1 && !dwFT2 && !dwFT3) szClient = "PreludeICQ"; } } else if (wVersion == 8) { if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_TYPING) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchShortCapability(caps, wLen, &capAimDirect)) szClient = "imo.im"; //https://imo.im/ - Web IM } // try to determine lite versions else if (wVersion >= 9) { if (CheckContactCapabilities(hContact, CAPF_XTRAZ)) { *bClientId = CLID_GENERIC; if (CheckContactCapabilities(hContact, CAPF_OSCAR_FILE)) { if (MatchCapability(caps, wLen, &captZers)) { // capable of tZers ? if (MatchCapability(caps, wLen, &capIcqLiteNew) && MatchShortCapability(caps, wLen, &capStatusTextAware) && MatchShortCapability(caps, wLen, &capAimLiveVideo) && MatchShortCapability(caps, wLen, &capAimLiveAudio)) { mir_strcpy(szClientBuf, "ICQ 7"); } else if (MatchCapability(caps, wLen, &capFakeHtml)) { if (MatchShortCapability(caps, wLen, &capAimLiveVideo) && MatchShortCapability(caps, wLen, &capAimLiveAudio)) { mir_strcpy(szClientBuf, "ICQ 6"); *bClientId = CLID_ICQ6; } else if (CheckContactCapabilities(hContact, CAPF_RTF) && !CheckContactCapabilities(hContact, CAPF_CONTACTS) && MatchShortCapability(caps, wLen, &capIcqDevils)) { mir_strcpy(szClientBuf, "Qnext v4"); // finally handles SRV_RELAY correctly *bClientId = CLID_ALTERNATIVE; } } else mir_strcpy(szClientBuf, "icq5.1"); } else mir_strcpy(szClientBuf, "icq5"); if (MatchCapability(caps, wLen, &capRambler)) mir_strcat(szClientBuf, " (Rambler)"); else if (MatchCapability(caps, wLen, &capAbv)) mir_strcat(szClientBuf, " (Abv)"); else if (MatchCapability(caps, wLen, &capNetvigator)) mir_strcat(szClientBuf, " (Netvigator)"); szClient = szClientBuf; } else if (!CheckContactCapabilities(hContact, CAPF_ICQDIRECT)) { *bClientId = CLID_ALTERNATIVE; if (CheckContactCapabilities(hContact, CAPF_RTF)) { // most probably Qnext - try to make that shit at least receiving our msgs ClearContactCapabilities(hContact, CAPF_SRV_RELAY); debugLogA("Forcing simple messages (QNext client)."); szClient = "Qnext"; } else if (CheckContactCapabilities(hContact, CAPF_TYPING) && MatchCapability(caps, wLen, &captZers) && MatchCapability(caps, wLen, &capFakeHtml)) { if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF) && MatchShortCapability(caps, wLen, &capAimLiveAudio)) szClient = "Mail.ru Agent (PC)"; else szClient = "Fring"; } else szClient = "pyICQ"; } else szClient = "ICQ Lite v4"; } else if (MatchCapability(caps, wLen, &capIcqLiteNew)) szClient = "ICQ Lite"; // the new ICQ Lite based on ICQ6 else if (!CheckContactCapabilities(hContact, CAPF_ICQDIRECT)) { if (MatchCapability(caps, wLen, &capFakeHtml) && MatchCapability(caps, wLen, &capOscarChat) && MatchShortCapability(caps, wLen, &capAimSmartCaps)) szClient = cliTrillian4; else 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 (wUserClass & CLASS_WIRELESS) szClient = "Pocket Web 1&1"; else szClient = "Icq2Go!"; } } 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); debugLogA("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 == 0xB) { if (CheckContactCapabilities(hContact, CAPF_XTRAZ | CAPF_SRV_RELAY | CAPF_TYPING | CAPF_UTF) && MatchShortCapability(caps, wLen, &capIcqDevils)) szClient = "Mail.ru Agent (Symbian)"; } 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_UTF | CAPF_CONTACTS | CAPF_XTRAZ | CAPF_OSCAR_FILE)) szClient = "ICQ 8"; else if (CheckContactCapabilities(hContact, CAPF_TYPING) && MatchCapability(caps, wLen, &capIs2001) && MatchCapability(caps, wLen, &capIs2002) && MatchCapability(caps, wLen, &capComm20012)) szClient = cliSpamBot; else if (MatchShortCapability(caps, wLen, &capAimIcon) && MatchShortCapability(caps, wLen, &capAimDirect) && CheckContactCapabilities(hContact, CAPF_OSCAR_FILE | CAPF_UTF)) { // detect libgaim/libpurple versions if (CheckContactCapabilities(hContact, CAPF_SRV_RELAY)) szClient = "Adium X"; // yeah, AFAIK only Adium has this fixed else if (CheckContactCapabilities(hContact, CAPF_TYPING)) szClient = "libpurple"; else szClient = "libgaim"; } else if (MatchShortCapability(caps, wLen, &capAimIcon) && MatchShortCapability(caps, wLen, &capAimDirect) && MatchCapability(caps, wLen, &capOscarChat) && CheckContactCapabilities(hContact, CAPF_OSCAR_FILE) && wLen == 0x40) szClient = "libgaim"; // Gaim 1.5.1 most probably else if (CheckContactCapabilities(hContact, CAPF_OSCAR_FILE) && MatchCapability(caps, wLen, &capOscarChat) && wLen == 0x20) szClient = "Easy Message"; else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_TYPING) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchCapability(caps, wLen, &capOscarChat) && wLen == 0x40) szClient = "Meebo"; else if (CheckContactCapabilities(hContact, CAPF_UTF) && MatchShortCapability(caps, wLen, &capAimIcon) && wLen == 0x20) szClient = "PyICQ-t Jabber Transport"; else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_XTRAZ) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchCapability(caps, wLen, &capXtrazVideo)) szClient = "PyICQ-t Jabber Transport"; else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_SRV_RELAY | CAPF_ICQDIRECT | CAPF_TYPING) && wLen == 0x40) szClient = "Agile Messenger"; // Smartphone 2002 else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_SRV_RELAY | CAPF_ICQDIRECT | CAPF_OSCAR_FILE) && MatchShortCapability(caps, wLen, &capAimFileShare)) szClient = "Slick"; // http://lonelycatgames.com/?app=slick else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_SRV_RELAY | CAPF_OSCAR_FILE | CAPF_CONTACTS) && MatchShortCapability(caps, wLen, &capAimFileShare) && MatchShortCapability(caps, wLen, &capAimIcon)) szClient = "Digsby"; // http://www.digsby.com else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_SRV_RELAY | CAPF_CONTACTS) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchCapability(caps, wLen, &capFakeHtml)) szClient = "mundu IM"; // http://messenger.mundu.com else if (CheckContactCapabilities(hContact, CAPF_UTF | CAPF_OSCAR_FILE) && MatchCapability(caps, wLen, &capOscarChat)) { if (CheckContactCapabilities(hContact, CAPF_TYPING)) szClient = "eBuddy"; // http://www.ebuddy.com else szClient = "eBuddy (Mobile)"; } else if (CheckContactCapabilities(hContact, CAPF_CONTACTS | CAPF_OSCAR_FILE) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchShortCapability(caps, wLen, &capAimDirect) && MatchCapability(caps, wLen, &capOscarChat)) szClient = "IloveIM"; //http://www.iloveim.com/ } } } } else if (!nIsICQ) { // detect AIM clients if (caps) { if (capId = MatchCapability(caps, wLen, &capAimOscar, 8)) { // AimOscar Signature DWORD aver = (*capId)[0xC] << 0x18 | (*capId)[0xD] << 0x10 | (*capId)[0xE] << 8 | (*capId)[0xF]; DWORD mver = (*capId)[0x8] << 0x18 | (*capId)[0x9] << 0x10 | (*capId)[0xA] << 8 | (*capId)[0xB]; szClient = MirandaVersionToStringEx(szClientBuf, 0, "AimOscar", aver, mver); bMirandaIM = TRUE; } else if (capId = MatchCapability(caps, wLen, &capSim, 0xC)) { // Sim is universal unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; szClient = makeClientVersion(szClientBuf, "SIM ", ver1, ver2, ver3, 0); if ((*capId)[0xF] & 0x80) mir_strcat(szClientBuf, "/Win32"); else if ((*capId)[0xF] & 0x40) mir_strcat(szClientBuf, "/MacOS X"); } else if (capId = MatchCapability(caps, wLen, &capKopete, 0xC)) { unsigned ver1 = (*capId)[0xC]; unsigned ver2 = (*capId)[0xD]; unsigned ver3 = (*capId)[0xE]; unsigned ver4 = (*capId)[0xF]; szClient = makeClientVersion(szClientBuf, "Kopete ", ver1, ver2, ver3, ver4); } else if (MatchCapability(caps, wLen, &capIcqIos) || MatchCapability(caps, wLen, &capIcqAndroid)) szClient = "ICQ (Mobile)"; else if (MatchCapability(caps, wLen, &capIm2)) // IM2 extensions szClient = cliIM2; else if (MatchCapability(caps, wLen, &capNaim, 0x8)) szClient = "naim"; // http://www.dibsby.com else if (MatchCapability(caps, wLen, &capDigsby, 0x06) || MatchCapability(caps, wLen, &capDigsbyBeta)) szClient = "Digsby"; else if (MatchShortCapability(caps, wLen, &capAimIcon) && MatchCapability(caps, wLen, &capOscarChat) && CheckContactCapabilities(hContact, CAPF_UTF | CAPF_TYPING) && wLen == 0x40) szClient = "Meebo"; // libpurple (e.g. Pidgin 2.7.x) else if (wLen == 0x90 && CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_TYPING | CAPF_XTRAZ) && MatchCapability(caps, wLen, &capOscarChat) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchShortCapability(caps, wLen, &capAimDirect) && MatchCapability(caps, wLen, &capFakeHtml)) szClient = "libpurple"; // libpurple - Meebo (without DirectIM and OFT) else if (wLen == 0x70 && CheckContactCapabilities(hContact, CAPF_SRV_RELAY | CAPF_UTF | CAPF_TYPING | CAPF_XTRAZ) && MatchCapability(caps, wLen, &capOscarChat) && MatchShortCapability(caps, wLen, &capAimIcon) && MatchCapability(caps, wLen, &capFakeHtml)) szClient = "Meebo"; else szClient = "AIM"; } else if (wUserClass & CLASS_WIRELESS) szClient = "AIM (Mobile)"; else { DWORD dwUin; uid_str szUid; getContactUid(hContact, &dwUin, &szUid); if (szUid[0]) { if (strstr(szUid, "@bk.ru") || strstr(szUid, "@list.ru") || strstr(szUid, "@mail.ru") || strstr(szUid, "@inbox.ru")) szClient = "MRA client"; } if (szClient == nullptr) szClient = "AIM"; } } // custom miranda packs if (caps && bMirandaIM) { capId = MatchCapability(caps, wLen, &capMimPack, 4); if (capId) { char szPack[16]; mir_snprintf(szPack, " [%.12s]", (*capId) + 4); // make sure client string is not constant if (szClient != szClientBuf) { mir_strcpy(szClientBuf, szClient); szClient = szClientBuf; } mir_strcat(szClientBuf, szPack); } } BOOL bClientDetected = (szClient != nullptr); // client detection failed, provide default clients if (!szClient) { *bClientId = CLID_GENERIC; switch (wVersion) { 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"; } } if (szClient) { char *szExtra = nullptr; if (MatchCapability(caps, wLen, &capSimpLite)) szExtra = " + SimpLite"; else if (MatchCapability(caps, wLen, &capSimpPro)) szExtra = " + SimpPro"; else if (MatchCapability(caps, wLen, &capIMsecure) || MatchCapability(caps, wLen, &capIMSecKey1, 6) || MatchCapability(caps, wLen, &capIMSecKey2, 6)) szExtra = " + IMsecure"; if (szExtra) { if (szClient != szClientBuf) { mir_strcpy(szClientBuf, szClient); szClient = szClientBuf; } mir_strcat(szClientBuf, szExtra); } } // Log the detection result if it has changed or contact just logged on... if (!szCurrentClient || mir_strcmp(szCurrentClient, szClient)) { if (bClientDetected) debugLogA("Client identified as %s", szClient); else debugLogA("No client identification, put default ICQ client for protocol."); } return szClient; }