/* UserinfoEx plugin for Miranda IM Copyright: © 2006-2010 DeathAxe, Yasnovidyashii, Merlin, K. Romanov, Kreol 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. */ #include "../stdafx.h" /** * name: CExImContactBase * class: CExImContactBase * desc: default constructor * param: none * return: nothing **/ CExImContactBase::CExImContactBase() { _pszNick = nullptr; _pszDisp = nullptr; _pszGroup = nullptr; _pszProto = nullptr; _pszProtoOld = nullptr; _pszAMPro = nullptr; _pszUIDKey = nullptr; _dbvUIDHash = NULL; memset(&_dbvUID, 0, sizeof(DBVARIANT)); _hContact = INVALID_CONTACT_ID; _isNewContact = FALSE; } /** * name: ~CExImContactBase * class: CExImContactBase * desc: default denstructor * param: none * return: nothing **/ CExImContactBase::~CExImContactBase() { MIR_FREE(_pszNick); MIR_FREE(_pszDisp); MIR_FREE(_pszGroup); MIR_FREE(_pszProtoOld); MIR_FREE(_pszProto); MIR_FREE(_pszAMPro); MIR_FREE(_pszUIDKey); db_free(&_dbvUID); memset(&_dbvUID, 0, sizeof(DBVARIANT)); } /** * name: fromDB * class: CExImContactBase * desc: get contact information from database * param: hContact - handle to contact whose information to read * return: TRUE if successful or FALSE otherwise **/ uint8_t CExImContactBase::fromDB(MCONTACT hContact) { uint8_t ret = FALSE; LPSTR pszProto; DBVARIANT dbv; _hContact = hContact; _dbvUIDHash = 0; MIR_FREE(_pszProtoOld); MIR_FREE(_pszProto); MIR_FREE(_pszAMPro); MIR_FREE(_pszNick); MIR_FREE(_pszDisp); MIR_FREE(_pszGroup); MIR_FREE(_pszUIDKey); db_free(&_dbvUID); memset(&_dbvUID, 0, sizeof(DBVARIANT)); // OWNER if (!_hContact) return TRUE; // Proto if (!(pszProto = Proto_GetBaseAccountName(_hContact))) return FALSE; _pszProto = mir_strdup(pszProto); // AM_BaseProto if (!DB::Setting::GetUString(NULL, pszProto, "AM_BaseProto", &dbv)) { _pszAMPro = mir_strdup(dbv.pszVal); db_free(&dbv); } LPCSTR uidSetting = Proto_GetUniqueId(pszProto); if (uidSetting != nullptr) { // valid _pszUIDKey = mir_strdup(uidSetting); if (!DB::Setting::GetAsIs(_hContact, pszProto, uidSetting, &_dbvUID)) { ret = TRUE; } } // fails because the protocol is no longer installed else { ret = TRUE; } // nickname if (!DB::Setting::GetUString(_hContact, pszProto, SET_CONTACT_NICK, &dbv)) { _pszNick = mir_strdup(dbv.pszVal); db_free(&dbv); } if (_hContact && ret) { // Clist Group if (!DB::Setting::GetUString(_hContact, MOD_CLIST, "Group", &dbv)) { _pszGroup = mir_strdup(dbv.pszVal); db_free(&dbv); } // Clist DisplayName if (!DB::Setting::GetUString(_hContact, MOD_CLIST, SET_CONTACT_MYHANDLE, &dbv)) { _pszDisp = mir_strdup(dbv.pszVal); db_free(&dbv); } } return ret; } /** * name: fromIni * class: CExImContactBase * desc: get contact information from a row of a ini file * param: row - the rows data * return: TRUE if successful or FALSE otherwise **/ uint8_t CExImContactBase::fromIni(LPSTR &row) { LPSTR p1, p2 = nullptr; LPSTR pszUIDValue, pszUIDSetting, pszProto = nullptr; LPSTR pszBuf = &row[0]; size_t cchBuf = mir_strlen(row); MIR_FREE(_pszProtoOld); MIR_FREE(_pszProto); MIR_FREE(_pszAMPro); MIR_FREE(_pszNick); MIR_FREE(_pszDisp); MIR_FREE(_pszGroup); MIR_FREE(_pszUIDKey); db_free(&_dbvUID); memset(&_dbvUID, 0, sizeof(DBVARIANT)); _dbvUIDHash = 0; // read uid value if (cchBuf > 10 && (p1 = mir_strrchr(pszBuf, '*{')) && (p2 = mir_strchr(p1, '}*')) && p1 + 2 < p2) { pszUIDValue = p1 + 1; *p1 = *(p2 - 1) = 0; // insulate the uid setting from buffer pointer if (cchBuf > 0 && (p1 = mir_strrchr(pszBuf, '*<')) && (p2 = mir_strchr(p1, '>*')) && p1 + 2 < p2) { pszUIDSetting = p1 + 1; *p1 = *(p2 - 1) = 0; // insulate the protocol name from buffer pointer if (cchBuf > 0 && (p1 = mir_strrchr(pszBuf, '*(')) && (p2 = mir_strchr(p1, ')*')) && p1 + 2 < p2) { pszProto = p1 + 1; *(--p1) = *(p2 - 1) = 0; // DBVT_DWORD if (strspn(pszUIDValue, "0123456789") == mir_strlen(pszUIDValue)) { _dbvUID.dVal = _atoi64(pszUIDValue); _dbvUID.type = DBVT_DWORD; } else { // DBVT_UTF8 _dbvUID.pszVal = mir_strdup(pszUIDValue); _dbvUID.type = DBVT_UTF8; } _pszUIDKey = mir_strdup(pszUIDSetting); _pszProto = mir_strdup(pszProto); } //end insulate the protocol name from buffer pointer } //end insulate the uid setting from buffer pointer } //end read uid value // create valid nickname _pszNick = mir_strdup(pszBuf); size_t i = mir_strlen(_pszNick) - 1; while (i > 0 && (_pszNick[i] == ' ' || _pszNick[i] == '\t')) { _pszNick[i] = 0; i--; } // finally try to find contact in contact list findHandle(); return FALSE; } /** * name: toDB * class: CExImContactBase * desc: searches the database for a contact representing the one * identified by this class or creates a new one if it was not found * param: hMetaContact - a meta contact to add this contact to * return: handle of the contact if successful **/ MCONTACT CExImContactBase::toDB() { // create new contact if none exists if (_hContact == INVALID_CONTACT_ID && _pszProto && _pszUIDKey && _dbvUID.type != DBVT_DELETED) { PROTOACCOUNT *pszAccount = Proto_GetAccount(_pszProto); if (pszAccount == nullptr) { //account does not exist return _hContact = INVALID_CONTACT_ID; } // create new contact _hContact = db_add_contact(); if (!_hContact) { return _hContact = INVALID_CONTACT_ID; } // Add the protocol to the new contact if (Proto_AddToContact(_hContact, _pszProto)) { db_delete_contact(_hContact); return _hContact = INVALID_CONTACT_ID; } // write uid to protocol module if (db_set(_hContact, _pszProto, _pszUIDKey, &_dbvUID)) { db_delete_contact(_hContact); return _hContact = INVALID_CONTACT_ID; } // write nick and display name if (_pszNick) db_set_utf(_hContact, _pszProto, SET_CONTACT_NICK, _pszNick); if (_pszDisp) db_set_utf(_hContact, MOD_CLIST, SET_CONTACT_MYHANDLE, _pszDisp); // add group if (_pszGroup) { ptrW ptszGroup(mir_utf8decodeW(_pszGroup)); Clist_SetGroup(_hContact, ptszGroup); Clist_GroupCreate(NULL, ptszGroup); } } return _hContact; } /** * name: toIni * class: CExImContactBase * desc: writes the line to an opened ini file which identifies the contact * whose information are stored in this class * param: file - pointer to the opened file * return: nothing **/ void CExImContactBase::toIni(FILE *file, int modCount) { // getting dbeditor++ NickFromHContact(hContact) static char name[512] = ""; if (_hContact) { int loaded = _pszUIDKey ? 1 : 0; if (_pszProto == nullptr || !loaded) { if (_pszProto) { if (_pszNick) mir_snprintf(name, "%s (%s)", _pszNick, _pszProto); else mir_snprintf(name, "(UNKNOWN) (%s)", _pszProto); } else strncpy_s(name, "(UNKNOWN)", _TRUNCATE); } else { // Proto loaded - GetContactName(hContact,pszProto,0) ptrW pszCI(Contact::GetInfo(CNF_DISPLAY, _hContact, _pszProto)); ptrA pszUID(uid2String(FALSE)); if (_pszUIDKey && pszUID) mir_snprintf(name, "%S *(%s)*<%s>*{%s}*", pszCI.get(), _pszProto, _pszUIDKey, pszUID.get()); else mir_snprintf(name, "%S (%s)", pszCI.get(), _pszProto); } // it is not the best solution (but still works if only basic modules export) - need rework if (modCount > 3) fprintf(file, "CONTACT: %s\n", name); else fprintf(file, "FROM CONTACT: %s\n", name); } // end *if (_hContact) else { fprintf(file, "SETTINGS:\n"); } } uint8_t CExImContactBase::compareUID(DBVARIANT *dbv) { uint32_t hash = 0; switch (dbv->type) { case DBVT_BYTE: if (dbv->bVal == _dbvUID.bVal) { _dbvUID.type = dbv->type; return TRUE; } break; case DBVT_WORD: if (dbv->wVal == _dbvUID.wVal) { _dbvUID.type = dbv->type; return TRUE; } break; case DBVT_DWORD: if (dbv->dVal == _dbvUID.dVal) { _dbvUID.type = dbv->type; return TRUE; } break; case DBVT_ASCIIZ: hash = mir_hashstr(dbv->pszVal); case DBVT_WCHAR: if (!hash) hash = mir_hashstrW(dbv->pwszVal); case DBVT_UTF8: if (!hash) { LPWSTR tmp = mir_utf8decodeW(dbv->pszVal); hash = mir_hashstrW(tmp); mir_free(tmp); } if (hash == _dbvUIDHash) return TRUE; break; case DBVT_BLOB: //'n' cpbVal and pbVal are valid if (_dbvUID.type == dbv->type && _dbvUID.cpbVal == dbv->cpbVal && memcmp(_dbvUID.pbVal, dbv->pbVal, dbv->cpbVal) == 0) { return TRUE; } break; } return FALSE; } LPSTR CExImContactBase::uid2String(uint8_t bPrependType) { CHAR szUID[4096]; LPSTR ptr = szUID; LPSTR r; SIZE_T baselen; switch (_dbvUID.type) { case DBVT_BYTE: //'b' bVal and cVal are valid if (bPrependType) *ptr++ = 'b'; _itoa(_dbvUID.bVal, ptr, 10); break; case DBVT_WORD: //'w' wVal and sVal are valid if (bPrependType) *ptr++ = 'w'; _itoa(_dbvUID.wVal, ptr, 10); break; case DBVT_DWORD: //'d' dVal and lVal are valid if (bPrependType) *ptr++ = 'd'; _itoa(_dbvUID.dVal, ptr, 10); break; case DBVT_WCHAR: //'u' pwszVal is valid r = mir_utf8encodeW(_dbvUID.pwszVal); if (r == nullptr) return nullptr; if (bPrependType == FALSE) return r; *ptr++ = 'u'; mir_strncpy(ptr, r, sizeof(szUID) - 1); mir_free(r); break; case DBVT_UTF8: //'u' pszVal is valid if (bPrependType) *ptr++ = 'u'; mir_strncpy(ptr, _dbvUID.pszVal, sizeof(szUID) - 1); break; case DBVT_ASCIIZ: r = mir_utf8encode(_dbvUID.pszVal); if (r == nullptr) return nullptr; if (bPrependType == FALSE) return r; *ptr++ = 's'; mir_strncpy(ptr, r, sizeof(szUID) - 1); mir_free(r); break; case DBVT_BLOB: //'n' cpbVal and pbVal are valid if (bPrependType) { //True = XML baselen = mir_base64_encode_bufsize(_dbvUID.cpbVal); r = (LPSTR)mir_alloc((baselen + 8)); if (r == nullptr) return nullptr; memset((r + baselen), 0, 8); ptr = r; if (bPrependType) { // Allways true. ptr[0] = 'n'; ptr++; } if (!mir_base64_encodebuf(_dbvUID.pbVal, _dbvUID.cpbVal, ptr, baselen)) { mir_free(r); return nullptr; } return r; } else { //FALSE = INI baselen = ((_dbvUID.cpbVal * 3) + 8); r = (LPSTR)mir_alloc(baselen); if (r == nullptr) return nullptr; memset(r, 0, baselen); ptr = r; for (SIZE_T j = 0; j < _dbvUID.cpbVal; j++, ptr += 3) { mir_snprintf(ptr, ((r + baselen) - ptr), "%02X ", (uint8_t)_dbvUID.pbVal[j]); } return r; } break; default: return nullptr; } return mir_strdup(szUID); } uint8_t CExImContactBase::isMeta() const { return DB::Module::IsMeta(_pszProto); } uint8_t CExImContactBase::isHandle(MCONTACT hContact) { LPCSTR pszProto; DBVARIANT dbv; uint8_t isEqual = FALSE; // owner contact ? if (!_pszProto) return hContact == NULL; // compare protocols pszProto = Proto_GetBaseAccountName(hContact); if (pszProto == nullptr || (INT_PTR)pszProto == CALLSERVICE_NOTFOUND || mir_strcmp(pszProto, _pszProto)) return FALSE; // compare uids if (_pszUIDKey) { // get uid if (DB::Setting::GetAsIs(hContact, pszProto, _pszUIDKey, &dbv)) return FALSE; isEqual = compareUID(&dbv); db_free(&dbv); } // compare nicknames if no UID else if (!DB::Setting::GetUString(hContact, _pszProto, SET_CONTACT_NICK, &dbv)) { if (dbv.type == DBVT_UTF8 && dbv.pszVal && !mir_strcmpi(dbv.pszVal, _pszNick)) { LPTSTR ptszNick = mir_utf8decodeW(_pszNick); LPTSTR ptszProto = mir_a2u(_pszProto); int ans = MsgBox(nullptr, MB_ICONQUESTION | MB_YESNO, LPGENW("Question"), LPGENW("contact identification"), LPGENW("The contact %s(%s) has no unique ID in the vCard,\nbut there is a contact in your contact list with the same nick and protocol.\nDo you wish to use this contact?"), ptszNick, ptszProto); MIR_FREE(ptszNick); MIR_FREE(ptszProto); isEqual = ans == IDYES; } db_free(&dbv); } return isEqual; } /** * name: handle * class: CExImContactBase * desc: return the handle of the contact * whose information are stored in this class * param: none * return: handle if successful, INVALID_HANDLE_VALUE otherwise **/ MCONTACT CExImContactBase::findHandle() { for (auto &hContact : Contacts()) { if (isHandle(hContact)) { _hContact = hContact; _isNewContact = FALSE; return hContact; } } _isNewContact = TRUE; return _hContact = INVALID_CONTACT_ID; }