From d60c379a9dcc439316b37aa135ed7c009a0c19f8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Robert=20P=C3=B6sel?= Date: Mon, 8 Dec 2014 22:55:24 +0000 Subject: Steam: Option to use big avatars (disabled by default) Until now big avatars were used, but it's better to lower traffic etc. by default. git-svn-id: http://svn.miranda-ng.org/main/trunk@11289 1316c22d-e87f-b044-9b9b-93d7a3e3ba9c --- protocols/Steam/src/steam_contacts.cpp | 1369 ++++++++++++++++---------------- 1 file changed, 685 insertions(+), 684 deletions(-) (limited to 'protocols/Steam/src/steam_contacts.cpp') diff --git a/protocols/Steam/src/steam_contacts.cpp b/protocols/Steam/src/steam_contacts.cpp index 582e7f5b20..1fca7bc85c 100644 --- a/protocols/Steam/src/steam_contacts.cpp +++ b/protocols/Steam/src/steam_contacts.cpp @@ -1,685 +1,686 @@ -#include "common.h" - -void CSteamProto::SetContactStatus(MCONTACT hContact, WORD status) -{ - WORD oldStatus = getWord(hContact, "Status", ID_STATUS_OFFLINE); - if (oldStatus != status) - setWord(hContact, "Status", status); -} - -void CSteamProto::SetAllContactsStatus(WORD status) -{ - for (MCONTACT hContact = db_find_first(this->m_szModuleName); hContact; hContact = db_find_next(hContact, this->m_szModuleName)) - { - if (this->isChatRoom(hContact)) - continue; - //if (this->IsContactOnline(hContact)) - setWord(hContact, "Status", status); - } -} - -MCONTACT CSteamProto::GetContactFromAuthEvent(HANDLE hEvent) -{ - DWORD body[3]; - DBEVENTINFO dbei = { sizeof(DBEVENTINFO) }; - dbei.cbBlob = sizeof(DWORD)* 2; - dbei.pBlob = (PBYTE)&body; - - if (db_event_get(hEvent, &dbei)) - return INVALID_CONTACT_ID; - - if (dbei.eventType != EVENTTYPE_AUTHREQUEST) - return INVALID_CONTACT_ID; - - if (strcmp(dbei.szModule, m_szModuleName) != 0) - return INVALID_CONTACT_ID; - - return DbGetAuthEventContact(&dbei); -} - -MCONTACT CSteamProto::FindContact(const char *steamId) -{ - MCONTACT hContact = NULL; - - EnterCriticalSection(&this->contact_search_lock); - - for (hContact = db_find_first(this->m_szModuleName); hContact; hContact = db_find_next(hContact, this->m_szModuleName)) - { - ptrA cSteamId(db_get_sa(hContact, this->m_szModuleName, "SteamID")); - if (!lstrcmpA(cSteamId, steamId)) - break; - } - - LeaveCriticalSection(&this->contact_search_lock); - - return hContact; -} - -void CSteamProto::UpdateContact(MCONTACT hContact, JSONNODE *data) -{ - JSONNODE *node = NULL; - - // set common data - node = json_get(data, "personaname"); - setWString(hContact, "Nick", json_as_string(node)); - - node = json_get(data, "profileurl"); - setString(hContact, "Homepage", ptrA(mir_u2a(json_as_string(node)))); - - // set name - node = json_get(data, "realname"); - if (node != NULL) - { - std::wstring realname = json_as_string(node); - if (!realname.empty()) - { - size_t pos = realname.find(L' ', 1); - if (pos != std::wstring::npos) - { - setWString(hContact, "FirstName", realname.substr(0, pos).c_str()); - setWString(hContact, "LastName", realname.substr(pos + 1).c_str()); - } - else - { - setWString(hContact, "FirstName", realname.c_str()); - delSetting(hContact, "LastName"); - } - } - } - else - { - delSetting(hContact, "FirstName"); - delSetting(hContact, "LastName"); - } - - // avatar - node = json_get(data, "avatarfull"); - std::string avatarUrl = _T2A(json_as_string(node)); - CheckAvatarChange(hContact, avatarUrl); - - // set country - node = json_get(data, "loccountrycode"); - if (node != NULL) - { - const char *iso = ptrA(mir_u2a(json_as_string(node))); - char *country = (char *)CallService(MS_UTILS_GETCOUNTRYBYISOCODE, (WPARAM)iso, 0); - setString(hContact, "Country", country); - } - else - delSetting(hContact, "Country"); - - node = json_get(data, "timecreated"); - setDword(hContact, "MemberTS", json_as_int(node)); - - // only for contacts - if (hContact) - { - node = json_get(data, "lastlogoff"); - setDword(hContact, "LogoffTS", json_as_int(node)); - - node = json_get(data, "gameid"); - DWORD gameId = node ? atol(_T2A(json_as_string(node))) : 0; - if (gameId > 0) - { - node = json_get(data, "gameextrainfo"); - const wchar_t *gameInfo = json_as_string(node); - - db_set_ws(hContact, "CList", "StatusMsg", gameInfo); - setWord(hContact, "Status", ID_STATUS_OUTTOLUNCH); - - setWString(hContact, "GameInfo", gameInfo); - setDword(hContact, "GameID", gameId); - } - else - { - node = json_get(data, "personastate"); - WORD status = SteamToMirandaStatus(json_as_int(node)); - setWord(hContact, "Status", status); - - db_unset(hContact, "CList", "StatusMsg"); - delSetting(hContact, "GameID"); - } - } -} - -void CSteamProto::ContactIsRemoved(MCONTACT hContact) -{ - if (!getDword(hContact, "DeletedTS", 0) && getByte(hContact, "Auth", 0) == 0) - { - setByte(hContact, "Auth", 1); - setDword(hContact, "DeletedTS", ::time(NULL)); - setWord(hContact, "Status", ID_STATUS_OFFLINE); - - ptrT nick(getTStringA(hContact, "Nick")); - TCHAR message[MAX_PATH]; - mir_sntprintf(message, MAX_PATH, TranslateT("%s has been removed from your contact list"), nick); - - ShowNotification(_T("Steam"), message); - } -} - -void CSteamProto::ContactIsFriend(MCONTACT hContact) -{ - if (getDword(hContact, "DeletedTS", 0) || getByte(hContact, "Auth", 0) != 0) - { - delSetting(hContact, "Auth"); - delSetting(hContact, "DeletedTS"); - delSetting(hContact, "Grant"); - - ptrT nick(getTStringA(hContact, "Nick")); - TCHAR message[MAX_PATH]; - mir_sntprintf(message, MAX_PATH, TranslateT("%s is back in your contact list"), nick); - - ShowNotification(_T("Steam"), message); - } -} - -void CSteamProto::ContactIsIgnored(MCONTACT hContact) -{ - // todo - setByte(hContact, "Block", 1); -} - -MCONTACT CSteamProto::AddContact(const char *steamId, bool isTemporary) -{ - MCONTACT hContact = this->FindContact(steamId); - if (!hContact) - { - // create contact - hContact = (MCONTACT)CallService(MS_DB_CONTACT_ADD, 0, 0); - CallService(MS_PROTO_ADDTOCONTACT, hContact, (LPARAM)this->m_szModuleName); - - setString(hContact, "SteamID", steamId); - - // update info - //UpdateContact(hContact, contact); - - if (isTemporary) - { - setByte(hContact, "Auth", 1); - //setByte(hContact, "Grant", 1); - db_set_b(hContact, "CList", "NotOnList", 1); - } - - // move to default group - DBVARIANT dbv; - if (!getWString("DefaultGroup", &dbv)) - { - db_set_ts(hContact, "CList", "Group", dbv.ptszVal); - db_free(&dbv); - } - } - - return hContact; -} - -void CSteamProto::ProcessContact(std::map::iterator *it, MCONTACT hContact) -{ - std::string steamId = (*it)->first; - JSONNODE *child = (*it)->second; - - if (!hContact) - hContact = AddContact(steamId.c_str()); - - JSONNODE *node = json_get(child, "friend_since"); - db_set_dw(hContact, "UserInfo", "ContactAddTime", json_as_int(node)); - - node = json_get(child, "relationship"); - ptrA relationship(mir_u2a(json_as_string(node))); - if (!lstrcmpiA(relationship, "friend")) - { - ContactIsFriend(hContact); - } - else if (!lstrcmpiA(relationship, "ignoredfriend")) - { - ContactIsIgnored(hContact); - } - else if (!lstrcmpiA(relationship, "requestrecipient")) - { - // todo - //RaiseAuthRequestThread((void*)hContact); - } -} - -void CSteamProto::OnGotFriendList(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL) - return; - - JSONROOT root(response->pData); - if (root == NULL) - return; - - std::string steamIds = ptrA(getStringA("SteamID")); - - std::map friends; - - // Remember contacts on server - JSONNODE *node = json_get(root, "friends"); - JSONNODE *nroot = json_as_array(node); - if (nroot != NULL) - { - for (size_t i = 0; i < json_size(nroot); i++) - { - JSONNODE *child = json_at(nroot, i); - if (child == NULL) - break; - - node = json_get(child, "steamid"); - if (node == NULL) - continue; - - std::string steamId = _T2A(json_as_string(node)); - friends.insert(std::make_pair(steamId, child)); - } - } - - // Check and update contacts in database - for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) - { - if (isChatRoom(hContact)) - continue; - - ptrA id(getStringA(hContact, "SteamID")); - if (id == NULL) - continue; - - std::map::iterator it = friends.find(std::string(id)); - - if (it != friends.end()) - { - // Contact is on server-list, update (and eventually notify) it - ProcessContact(&it, hContact); - - steamIds.append(",").append(it->first); - friends.erase(it); - } - else - { - // Contact was removed from server-list, notify it - ContactIsRemoved(hContact); - } - } - - // Check remaining contacts in map and add them to contact list - for (std::map::iterator it = friends.begin(); it != friends.end();) - { - // Contact is on server-list, but not in database, add (but not notify) it - ProcessContact(&it, NULL); - - steamIds.append(",").append(it->first); - it = friends.erase(it); - } - friends.clear(); - - if (!steamIds.empty()) - { - //steamIds.pop_back(); - ptrA token(getStringA("TokenSecret")); - - PushRequest( - new SteamWebApi::GetUserSummariesRequest(token, steamIds.c_str()), - &CSteamProto::OnGotUserSummaries); - } -} - -void CSteamProto::OnGotBlockList(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL) - return; - - JSONROOT root(response->pData); - if (root == NULL) - return; - - //std::string steamIds; - - JSONNODE *node = json_get(root, "friends"); - JSONNODE *nroot = json_as_array(node); - if (nroot != NULL) - { - for (size_t i = 0; i < json_size(nroot); i++) - { - JSONNODE *child = json_at(nroot, i); - if (child == NULL) - break; - - node = json_get(child, "steamid"); - ptrA steamId(mir_u2a(json_as_string(node))); - - /*MCONTACT hContact = FindContact(steamId); - if (!hContact) - { - hContact = AddContact(steamId); - steamIds.append(steamId).append(","); - }*/ - - node = json_get(child, "relationship"); - ptrA relationship(mir_u2a(json_as_string(node))); - - if (!lstrcmpiA(relationship, "ignoredfriend")) - { - // todo: open block list - } - else continue; - } - } -} - -void CSteamProto::OnGotUserSummaries(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL) - return; - - JSONROOT root(response->pData); - if (root == NULL) - return; - - JSONNODE *node = json_get(root, "players"); - JSONNODE *nroot = json_as_array(node); - if (nroot != NULL) - { - for (size_t i = 0; i < json_size(nroot); i++) - { - JSONNODE *item = json_at(nroot, i); - if (item == NULL) - break; - - node = json_get(item, "steamid"); - ptrA steamId(mir_u2a(json_as_string(node))); - - MCONTACT hContact = NULL; - if (!IsMe(steamId)) { - hContact = FindContact(steamId); - if (!hContact) - hContact = AddContact(steamId); - } - - UpdateContact(hContact, item); - } - } -} - -void CSteamProto::OnGotAvatar(const NETLIBHTTPREQUEST *response, void *arg) -{ - PROTO_AVATAR_INFORMATIONW pai = { sizeof(pai) }; - pai.hContact = (MCONTACT)arg; - GetDbAvatarInfo(pai); - - if (response == NULL || response->resultCode != HTTP_STATUS_OK) - { - ptrA steamId(getStringA(pai.hContact, "SteamID")); - debugLogA("CSteamProto::OnGotAvatar: failed to get avatar %s", steamId); - - if (pai.hContact) - ProtoBroadcastAck(pai.hContact, ACKTYPE_AVATAR, ACKRESULT_FAILED, (HANDLE)&pai, 0); - return; - } - - FILE *fp = _tfopen(pai.filename, _T("wb")); - if (fp) - { - fwrite(response->pData, sizeof(char), response->dataLength, fp); - fclose(fp); - - if (pai.hContact) - ProtoBroadcastAck(pai.hContact, ACKTYPE_AVATAR, ACKRESULT_SUCCESS, (HANDLE)&pai, 0); - else - CallService(MS_AV_REPORTMYAVATARCHANGED, (WPARAM)m_szModuleName, 0); - } -} - -void CSteamProto::OnFriendAdded(const NETLIBHTTPREQUEST *response, void *arg) -{ - SendAuthParam *param = (SendAuthParam*)arg; - - if (response == NULL || response->resultCode != HTTP_STATUS_OK || lstrcmpiA(response->pData, "true")) - { - ptrA steamId(getStringA(param->hContact, "SteamID")); - debugLogA("CSteamProto::OnFriendAdded: failed to add friend %s", steamId); - - ProtoBroadcastAck(param->hContact, ACKTYPE_AUTHREQ, ACKRESULT_FAILED, param->hAuth, 0); - - return; - } - - delSetting(param->hContact, "Auth"); - delSetting(param->hContact, "Grant"); - db_unset(param->hContact, "CList", "NotOnList"); - - ProtoBroadcastAck(param->hContact, ACKTYPE_AUTHREQ, ACKRESULT_SUCCESS, param->hAuth, 0); -} - -void CSteamProto::OnFriendBlocked(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL || response->resultCode != HTTP_STATUS_OK || lstrcmpiA(response->pData, "true")) - { - debugLogA("CSteamProto::OnFriendIgnored: failed to ignore friend %s", ptrA((char*)arg)); - return; - } -} - -void CSteamProto::OnFriendRemoved(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL || response->resultCode != HTTP_STATUS_OK || lstrcmpiA(response->pData, "true")) - { - MCONTACT hContact = (MCONTACT)arg; - ptrA who(getStringA(hContact, "SteamID")); - - debugLogA("CSteamProto::OnFriendRemoved: failed to remove friend %s", who); - return; - } -} - -void CSteamProto::OnAuthRequested(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL || response->resultCode != HTTP_STATUS_OK) - { - debugLogA("CSteamProto::OnAuthRequested: failed to request info for %s", ptrA((char*)arg)); - return; - } - - JSONROOT root(response->pData); - if (root == NULL) - return; - - JSONNODE *node = json_get(root, "players"); - JSONNODE *nroot = json_at(json_as_array(node), 0); - if (nroot != NULL) - { - node = json_get(nroot, "steamid"); - ptrA steamId(mir_u2a(json_as_string(node))); - - MCONTACT hContact = FindContact(steamId); - if (!hContact) - hContact = AddContact(steamId); - - UpdateContact(hContact, nroot); - - char *nickName = getStringA(hContact, "Nick"); - char *firstName = getStringA(hContact, "FirstName"); - if (firstName == NULL) - firstName = mir_strdup(""); - char *lastName = getStringA(hContact, "LastName"); - if (lastName == NULL) - lastName = mir_strdup(""); - - char reason[MAX_PATH]; - mir_snprintf(reason, SIZEOF(reason), Translate("%s has added you to his or her Friend List"), nickName); - - // blob is: 0(DWORD), hContact(DWORD), nick(ASCIIZ), firstName(ASCIIZ), lastName(ASCIIZ), sid(ASCIIZ), reason(ASCIIZ) - DWORD cbBlob = (DWORD)(sizeof(DWORD)* 2 + lstrlenA(nickName) + lstrlenA(firstName) + lstrlenA(lastName) + lstrlenA(steamId) + lstrlenA(reason) + 5); - - PBYTE pBlob, pCurBlob; - pCurBlob = pBlob = (PBYTE)mir_alloc(cbBlob); - - *((PDWORD)pCurBlob) = 0; - pCurBlob += sizeof(DWORD); - *((PDWORD)pCurBlob) = (DWORD)hContact; - pCurBlob += sizeof(DWORD); - strcpy((char*)pCurBlob, nickName); - pCurBlob += lstrlenA(nickName) + 1; - strcpy((char*)pCurBlob, firstName); - pCurBlob += lstrlenA(firstName) + 1; - strcpy((char*)pCurBlob, lastName); - pCurBlob += lstrlenA(lastName) + 1; - strcpy((char*)pCurBlob, steamId); - pCurBlob += lstrlenA(steamId) + 1; - strcpy((char*)pCurBlob, mir_strdup(reason)); - - AddDBEvent(hContact, EVENTTYPE_AUTHREQUEST, time(NULL), DBEF_UTF, cbBlob, pBlob); - } -} - -void CSteamProto::OnPendingApproved(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL || response->resultCode != HTTP_STATUS_OK) - { - debugLogA("CSteamProto::OnPendingApproved: failed to approve pending from %s", ptrA((char*)arg)); - return; - } - - JSONROOT root(response->pData); - if (root == NULL) - return; - - JSONNODE *node = json_get(root, "success"); - if (json_as_int(node) == 0) - { - node = json_get(root, "error_text"); - debugLogA("CSteamProto::OnPendingApproved: failed to approve pending from %s (%s)", ptrA((char*)arg), ptrA(mir_utf8encodeW(json_as_string(node)))); - } -} - -void CSteamProto::OnPendingIgnoreded(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL || response->resultCode != HTTP_STATUS_OK) - { - debugLogA("CSteamProto::OnPendingApproved: failed to ignore pending from %s", ptrA((char*)arg)); - return; - } - - JSONROOT root(response->pData); - if (root == NULL) - return; - - JSONNODE *node = json_get(root, "success"); - if (json_as_int(node) == 0) - { - node = json_get(root, "error_text"); - debugLogA("CSteamProto::OnPendingApproved: failed to ignore pending from %s (%s)", ptrA((char*)arg), ptrA(mir_utf8encodeW(json_as_string(node)))); - } -} - -void CSteamProto::OnSearchByIdEnded(const NETLIBHTTPREQUEST *response, void *arg) -{ - if (response == NULL || response->resultCode != HTTP_STATUS_OK) - { - ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_FAILED, (HANDLE)STEAM_SEARCH_BYID, 0); - debugLogA("CSteamProto::OnSearchByIdEnded: failed to get summaries for %s", ptrA((char*)arg)); - return; - } - - JSONROOT root(response->pData); - if (root == NULL) - return; - - JSONNODE *node = json_get(root, "players"); - JSONNODE *nroot = json_at(json_as_array(node), 0); - if (nroot != NULL) - { - STEAM_SEARCH_RESULT ssr = { 0 }; - ssr.hdr.cbSize = sizeof(STEAM_SEARCH_RESULT); - ssr.hdr.flags = PSR_TCHAR; - - ssr.hdr.id = (wchar_t*)arg; - - node = json_get(nroot, "personaname"); - ssr.hdr.nick = mir_wstrdup(json_as_string(node)); - - node = json_get(nroot, "realname"); - if (node != NULL) - { - std::wstring realname = json_as_string(node); - if (!realname.empty()) - { - size_t pos = realname.find(' ', 1); - if (pos != std::string::npos) - { - ssr.hdr.firstName = mir_wstrdup(realname.substr(0, pos).c_str()); - ssr.hdr.lastName = mir_wstrdup(realname.substr(pos + 1).c_str()); - } - else - ssr.hdr.firstName = mir_wstrdup(realname.c_str()); - } - } - - //ssr.contact = contact; - ssr.data = json_copy(nroot); - - ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, (HANDLE)STEAM_SEARCH_BYID, (LPARAM)&ssr); - ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, (HANDLE)STEAM_SEARCH_BYID, 0); - } -} - -void CSteamProto::OnSearchByNameStarted(const NETLIBHTTPREQUEST *response, void *arg) -{ -} - -//void CSteamProto::SearchByNameThread(void* arg) -//{ -// ptrW keywordsW((wchar_t*)arg); -// ptrA keywords(mir_utf8encodeW(keywordsW)); -// -// ptrA token(getStringA("TokenSecret")); -// -// SteamWebApi::SearchApi::SearchResult searchResult; -// debugLogA("CSteamProto::SearchByNameThread: call SteamWebApi::SearchApi::Search"); -// SteamWebApi::SearchApi::Search(m_hNetlibUser, token, keywords, &searchResult); -// -// if (!searchResult.IsSuccess()) -// { -// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_FAILED, (HANDLE)STEAM_SEARCH_BYNAME, 0); -// return; -// } -// -// CMStringA steamIds; -// for (int i = 0; i < searchResult.GetItemCount(); i++) -// { -// const SteamWebApi::SearchApi::SearchItem *item = searchResult.GetAt(i); -// if (steamIds.IsEmpty()) -// steamIds.Append(item->GetSteamId()); -// else -// steamIds.AppendFormat(",%s", item->GetSteamId()); -// } -// -// SteamWebApi::FriendApi::Summaries summarues; -// debugLogA("CSteamProto::SearchByNameThread: call SteamWebApi::FriendApi::LoadSummaries"); -// SteamWebApi::FriendApi::LoadSummaries(m_hNetlibUser, token, steamIds, &summarues); -// -// if (!summarues.IsSuccess()) -// { -// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_FAILED, (HANDLE)STEAM_SEARCH_BYNAME, 0); -// return; -// } -// -// for (size_t i = 0; i < summarues.GetItemCount(); i++) -// { -// const SteamWebApi::FriendApi::Summary *contact = summarues.GetAt(i); -// -// STEAM_SEARCH_RESULT ssr = { 0 }; -// ssr.hdr.cbSize = sizeof(STEAM_SEARCH_RESULT); -// ssr.hdr.flags = PSR_TCHAR; -// -// ssr.hdr.id = mir_a2u(contact->GetSteamId()); -// ssr.hdr.nick = mir_wstrdup(contact->GetNickName()); -// ssr.hdr.firstName = mir_wstrdup(contact->GetFirstName()); -// ssr.hdr.lastName = mir_wstrdup(contact->GetLastName()); -// -// ssr.contact = contact; -// -// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, (HANDLE)STEAM_SEARCH_BYNAME, (LPARAM)&ssr); -// } -// -// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, (HANDLE)STEAM_SEARCH_BYNAME, 0); +#include "common.h" + +void CSteamProto::SetContactStatus(MCONTACT hContact, WORD status) +{ + WORD oldStatus = getWord(hContact, "Status", ID_STATUS_OFFLINE); + if (oldStatus != status) + setWord(hContact, "Status", status); +} + +void CSteamProto::SetAllContactsStatus(WORD status) +{ + for (MCONTACT hContact = db_find_first(this->m_szModuleName); hContact; hContact = db_find_next(hContact, this->m_szModuleName)) + { + if (this->isChatRoom(hContact)) + continue; + //if (this->IsContactOnline(hContact)) + setWord(hContact, "Status", status); + } +} + +MCONTACT CSteamProto::GetContactFromAuthEvent(HANDLE hEvent) +{ + DWORD body[3]; + DBEVENTINFO dbei = { sizeof(DBEVENTINFO) }; + dbei.cbBlob = sizeof(DWORD)* 2; + dbei.pBlob = (PBYTE)&body; + + if (db_event_get(hEvent, &dbei)) + return INVALID_CONTACT_ID; + + if (dbei.eventType != EVENTTYPE_AUTHREQUEST) + return INVALID_CONTACT_ID; + + if (strcmp(dbei.szModule, m_szModuleName) != 0) + return INVALID_CONTACT_ID; + + return DbGetAuthEventContact(&dbei); +} + +MCONTACT CSteamProto::FindContact(const char *steamId) +{ + MCONTACT hContact = NULL; + + EnterCriticalSection(&this->contact_search_lock); + + for (hContact = db_find_first(this->m_szModuleName); hContact; hContact = db_find_next(hContact, this->m_szModuleName)) + { + ptrA cSteamId(db_get_sa(hContact, this->m_szModuleName, "SteamID")); + if (!lstrcmpA(cSteamId, steamId)) + break; + } + + LeaveCriticalSection(&this->contact_search_lock); + + return hContact; +} + +void CSteamProto::UpdateContact(MCONTACT hContact, JSONNODE *data) +{ + JSONNODE *node = NULL; + + // set common data + node = json_get(data, "personaname"); + setWString(hContact, "Nick", json_as_string(node)); + + node = json_get(data, "profileurl"); + setString(hContact, "Homepage", ptrA(mir_u2a(json_as_string(node)))); + + // set name + node = json_get(data, "realname"); + if (node != NULL) + { + std::wstring realname = json_as_string(node); + if (!realname.empty()) + { + size_t pos = realname.find(L' ', 1); + if (pos != std::wstring::npos) + { + setWString(hContact, "FirstName", realname.substr(0, pos).c_str()); + setWString(hContact, "LastName", realname.substr(pos + 1).c_str()); + } + else + { + setWString(hContact, "FirstName", realname.c_str()); + delSetting(hContact, "LastName"); + } + } + } + else + { + delSetting(hContact, "FirstName"); + delSetting(hContact, "LastName"); + } + + // avatar + bool biggerAvatars = getBool("UseBigAvatars", false); + node = json_get(data, biggerAvatars ? "avatarfull" : "avatarmedium"); + std::string avatarUrl = _T2A(json_as_string(node)); + CheckAvatarChange(hContact, avatarUrl); + + // set country + node = json_get(data, "loccountrycode"); + if (node != NULL) + { + const char *iso = ptrA(mir_u2a(json_as_string(node))); + char *country = (char *)CallService(MS_UTILS_GETCOUNTRYBYISOCODE, (WPARAM)iso, 0); + setString(hContact, "Country", country); + } + else + delSetting(hContact, "Country"); + + node = json_get(data, "timecreated"); + setDword(hContact, "MemberTS", json_as_int(node)); + + // only for contacts + if (hContact) + { + node = json_get(data, "lastlogoff"); + setDword(hContact, "LogoffTS", json_as_int(node)); + + node = json_get(data, "gameid"); + DWORD gameId = node ? atol(_T2A(json_as_string(node))) : 0; + if (gameId > 0) + { + node = json_get(data, "gameextrainfo"); + const wchar_t *gameInfo = json_as_string(node); + + db_set_ws(hContact, "CList", "StatusMsg", gameInfo); + setWord(hContact, "Status", ID_STATUS_OUTTOLUNCH); + + setWString(hContact, "GameInfo", gameInfo); + setDword(hContact, "GameID", gameId); + } + else + { + node = json_get(data, "personastate"); + WORD status = SteamToMirandaStatus(json_as_int(node)); + setWord(hContact, "Status", status); + + db_unset(hContact, "CList", "StatusMsg"); + delSetting(hContact, "GameID"); + } + } +} + +void CSteamProto::ContactIsRemoved(MCONTACT hContact) +{ + if (!getDword(hContact, "DeletedTS", 0) && getByte(hContact, "Auth", 0) == 0) + { + setByte(hContact, "Auth", 1); + setDword(hContact, "DeletedTS", ::time(NULL)); + setWord(hContact, "Status", ID_STATUS_OFFLINE); + + ptrT nick(getTStringA(hContact, "Nick")); + TCHAR message[MAX_PATH]; + mir_sntprintf(message, MAX_PATH, TranslateT("%s has been removed from your contact list"), nick); + + ShowNotification(_T("Steam"), message); + } +} + +void CSteamProto::ContactIsFriend(MCONTACT hContact) +{ + if (getDword(hContact, "DeletedTS", 0) || getByte(hContact, "Auth", 0) != 0) + { + delSetting(hContact, "Auth"); + delSetting(hContact, "DeletedTS"); + delSetting(hContact, "Grant"); + + ptrT nick(getTStringA(hContact, "Nick")); + TCHAR message[MAX_PATH]; + mir_sntprintf(message, MAX_PATH, TranslateT("%s is back in your contact list"), nick); + + ShowNotification(_T("Steam"), message); + } +} + +void CSteamProto::ContactIsIgnored(MCONTACT hContact) +{ + // todo + setByte(hContact, "Block", 1); +} + +MCONTACT CSteamProto::AddContact(const char *steamId, bool isTemporary) +{ + MCONTACT hContact = this->FindContact(steamId); + if (!hContact) + { + // create contact + hContact = (MCONTACT)CallService(MS_DB_CONTACT_ADD, 0, 0); + CallService(MS_PROTO_ADDTOCONTACT, hContact, (LPARAM)this->m_szModuleName); + + setString(hContact, "SteamID", steamId); + + // update info + //UpdateContact(hContact, contact); + + if (isTemporary) + { + setByte(hContact, "Auth", 1); + //setByte(hContact, "Grant", 1); + db_set_b(hContact, "CList", "NotOnList", 1); + } + + // move to default group + DBVARIANT dbv; + if (!getWString("DefaultGroup", &dbv)) + { + db_set_ts(hContact, "CList", "Group", dbv.ptszVal); + db_free(&dbv); + } + } + + return hContact; +} + +void CSteamProto::ProcessContact(std::map::iterator *it, MCONTACT hContact) +{ + std::string steamId = (*it)->first; + JSONNODE *child = (*it)->second; + + if (!hContact) + hContact = AddContact(steamId.c_str()); + + JSONNODE *node = json_get(child, "friend_since"); + db_set_dw(hContact, "UserInfo", "ContactAddTime", json_as_int(node)); + + node = json_get(child, "relationship"); + ptrA relationship(mir_u2a(json_as_string(node))); + if (!lstrcmpiA(relationship, "friend")) + { + ContactIsFriend(hContact); + } + else if (!lstrcmpiA(relationship, "ignoredfriend")) + { + ContactIsIgnored(hContact); + } + else if (!lstrcmpiA(relationship, "requestrecipient")) + { + // todo + //RaiseAuthRequestThread((void*)hContact); + } +} + +void CSteamProto::OnGotFriendList(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL) + return; + + JSONROOT root(response->pData); + if (root == NULL) + return; + + std::string steamIds = ptrA(getStringA("SteamID")); + + std::map friends; + + // Remember contacts on server + JSONNODE *node = json_get(root, "friends"); + JSONNODE *nroot = json_as_array(node); + if (nroot != NULL) + { + for (size_t i = 0; i < json_size(nroot); i++) + { + JSONNODE *child = json_at(nroot, i); + if (child == NULL) + break; + + node = json_get(child, "steamid"); + if (node == NULL) + continue; + + std::string steamId = _T2A(json_as_string(node)); + friends.insert(std::make_pair(steamId, child)); + } + } + + // Check and update contacts in database + for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) + { + if (isChatRoom(hContact)) + continue; + + ptrA id(getStringA(hContact, "SteamID")); + if (id == NULL) + continue; + + std::map::iterator it = friends.find(std::string(id)); + + if (it != friends.end()) + { + // Contact is on server-list, update (and eventually notify) it + ProcessContact(&it, hContact); + + steamIds.append(",").append(it->first); + friends.erase(it); + } + else + { + // Contact was removed from server-list, notify it + ContactIsRemoved(hContact); + } + } + + // Check remaining contacts in map and add them to contact list + for (std::map::iterator it = friends.begin(); it != friends.end();) + { + // Contact is on server-list, but not in database, add (but not notify) it + ProcessContact(&it, NULL); + + steamIds.append(",").append(it->first); + it = friends.erase(it); + } + friends.clear(); + + if (!steamIds.empty()) + { + //steamIds.pop_back(); + ptrA token(getStringA("TokenSecret")); + + PushRequest( + new SteamWebApi::GetUserSummariesRequest(token, steamIds.c_str()), + &CSteamProto::OnGotUserSummaries); + } +} + +void CSteamProto::OnGotBlockList(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL) + return; + + JSONROOT root(response->pData); + if (root == NULL) + return; + + //std::string steamIds; + + JSONNODE *node = json_get(root, "friends"); + JSONNODE *nroot = json_as_array(node); + if (nroot != NULL) + { + for (size_t i = 0; i < json_size(nroot); i++) + { + JSONNODE *child = json_at(nroot, i); + if (child == NULL) + break; + + node = json_get(child, "steamid"); + ptrA steamId(mir_u2a(json_as_string(node))); + + /*MCONTACT hContact = FindContact(steamId); + if (!hContact) + { + hContact = AddContact(steamId); + steamIds.append(steamId).append(","); + }*/ + + node = json_get(child, "relationship"); + ptrA relationship(mir_u2a(json_as_string(node))); + + if (!lstrcmpiA(relationship, "ignoredfriend")) + { + // todo: open block list + } + else continue; + } + } +} + +void CSteamProto::OnGotUserSummaries(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL) + return; + + JSONROOT root(response->pData); + if (root == NULL) + return; + + JSONNODE *node = json_get(root, "players"); + JSONNODE *nroot = json_as_array(node); + if (nroot != NULL) + { + for (size_t i = 0; i < json_size(nroot); i++) + { + JSONNODE *item = json_at(nroot, i); + if (item == NULL) + break; + + node = json_get(item, "steamid"); + ptrA steamId(mir_u2a(json_as_string(node))); + + MCONTACT hContact = NULL; + if (!IsMe(steamId)) { + hContact = FindContact(steamId); + if (!hContact) + hContact = AddContact(steamId); + } + + UpdateContact(hContact, item); + } + } +} + +void CSteamProto::OnGotAvatar(const NETLIBHTTPREQUEST *response, void *arg) +{ + PROTO_AVATAR_INFORMATIONW pai = { sizeof(pai) }; + pai.hContact = (MCONTACT)arg; + GetDbAvatarInfo(pai); + + if (response == NULL || response->resultCode != HTTP_STATUS_OK) + { + ptrA steamId(getStringA(pai.hContact, "SteamID")); + debugLogA("CSteamProto::OnGotAvatar: failed to get avatar %s", steamId); + + if (pai.hContact) + ProtoBroadcastAck(pai.hContact, ACKTYPE_AVATAR, ACKRESULT_FAILED, (HANDLE)&pai, 0); + return; + } + + FILE *fp = _tfopen(pai.filename, _T("wb")); + if (fp) + { + fwrite(response->pData, sizeof(char), response->dataLength, fp); + fclose(fp); + + if (pai.hContact) + ProtoBroadcastAck(pai.hContact, ACKTYPE_AVATAR, ACKRESULT_SUCCESS, (HANDLE)&pai, 0); + else + CallService(MS_AV_REPORTMYAVATARCHANGED, (WPARAM)m_szModuleName, 0); + } +} + +void CSteamProto::OnFriendAdded(const NETLIBHTTPREQUEST *response, void *arg) +{ + SendAuthParam *param = (SendAuthParam*)arg; + + if (response == NULL || response->resultCode != HTTP_STATUS_OK || lstrcmpiA(response->pData, "true")) + { + ptrA steamId(getStringA(param->hContact, "SteamID")); + debugLogA("CSteamProto::OnFriendAdded: failed to add friend %s", steamId); + + ProtoBroadcastAck(param->hContact, ACKTYPE_AUTHREQ, ACKRESULT_FAILED, param->hAuth, 0); + + return; + } + + delSetting(param->hContact, "Auth"); + delSetting(param->hContact, "Grant"); + db_unset(param->hContact, "CList", "NotOnList"); + + ProtoBroadcastAck(param->hContact, ACKTYPE_AUTHREQ, ACKRESULT_SUCCESS, param->hAuth, 0); +} + +void CSteamProto::OnFriendBlocked(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL || response->resultCode != HTTP_STATUS_OK || lstrcmpiA(response->pData, "true")) + { + debugLogA("CSteamProto::OnFriendIgnored: failed to ignore friend %s", ptrA((char*)arg)); + return; + } +} + +void CSteamProto::OnFriendRemoved(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL || response->resultCode != HTTP_STATUS_OK || lstrcmpiA(response->pData, "true")) + { + MCONTACT hContact = (MCONTACT)arg; + ptrA who(getStringA(hContact, "SteamID")); + + debugLogA("CSteamProto::OnFriendRemoved: failed to remove friend %s", who); + return; + } +} + +void CSteamProto::OnAuthRequested(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL || response->resultCode != HTTP_STATUS_OK) + { + debugLogA("CSteamProto::OnAuthRequested: failed to request info for %s", ptrA((char*)arg)); + return; + } + + JSONROOT root(response->pData); + if (root == NULL) + return; + + JSONNODE *node = json_get(root, "players"); + JSONNODE *nroot = json_at(json_as_array(node), 0); + if (nroot != NULL) + { + node = json_get(nroot, "steamid"); + ptrA steamId(mir_u2a(json_as_string(node))); + + MCONTACT hContact = FindContact(steamId); + if (!hContact) + hContact = AddContact(steamId); + + UpdateContact(hContact, nroot); + + char *nickName = getStringA(hContact, "Nick"); + char *firstName = getStringA(hContact, "FirstName"); + if (firstName == NULL) + firstName = mir_strdup(""); + char *lastName = getStringA(hContact, "LastName"); + if (lastName == NULL) + lastName = mir_strdup(""); + + char reason[MAX_PATH]; + mir_snprintf(reason, SIZEOF(reason), Translate("%s has added you to his or her Friend List"), nickName); + + // blob is: 0(DWORD), hContact(DWORD), nick(ASCIIZ), firstName(ASCIIZ), lastName(ASCIIZ), sid(ASCIIZ), reason(ASCIIZ) + DWORD cbBlob = (DWORD)(sizeof(DWORD)* 2 + lstrlenA(nickName) + lstrlenA(firstName) + lstrlenA(lastName) + lstrlenA(steamId) + lstrlenA(reason) + 5); + + PBYTE pBlob, pCurBlob; + pCurBlob = pBlob = (PBYTE)mir_alloc(cbBlob); + + *((PDWORD)pCurBlob) = 0; + pCurBlob += sizeof(DWORD); + *((PDWORD)pCurBlob) = (DWORD)hContact; + pCurBlob += sizeof(DWORD); + strcpy((char*)pCurBlob, nickName); + pCurBlob += lstrlenA(nickName) + 1; + strcpy((char*)pCurBlob, firstName); + pCurBlob += lstrlenA(firstName) + 1; + strcpy((char*)pCurBlob, lastName); + pCurBlob += lstrlenA(lastName) + 1; + strcpy((char*)pCurBlob, steamId); + pCurBlob += lstrlenA(steamId) + 1; + strcpy((char*)pCurBlob, mir_strdup(reason)); + + AddDBEvent(hContact, EVENTTYPE_AUTHREQUEST, time(NULL), DBEF_UTF, cbBlob, pBlob); + } +} + +void CSteamProto::OnPendingApproved(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL || response->resultCode != HTTP_STATUS_OK) + { + debugLogA("CSteamProto::OnPendingApproved: failed to approve pending from %s", ptrA((char*)arg)); + return; + } + + JSONROOT root(response->pData); + if (root == NULL) + return; + + JSONNODE *node = json_get(root, "success"); + if (json_as_int(node) == 0) + { + node = json_get(root, "error_text"); + debugLogA("CSteamProto::OnPendingApproved: failed to approve pending from %s (%s)", ptrA((char*)arg), ptrA(mir_utf8encodeW(json_as_string(node)))); + } +} + +void CSteamProto::OnPendingIgnoreded(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL || response->resultCode != HTTP_STATUS_OK) + { + debugLogA("CSteamProto::OnPendingApproved: failed to ignore pending from %s", ptrA((char*)arg)); + return; + } + + JSONROOT root(response->pData); + if (root == NULL) + return; + + JSONNODE *node = json_get(root, "success"); + if (json_as_int(node) == 0) + { + node = json_get(root, "error_text"); + debugLogA("CSteamProto::OnPendingApproved: failed to ignore pending from %s (%s)", ptrA((char*)arg), ptrA(mir_utf8encodeW(json_as_string(node)))); + } +} + +void CSteamProto::OnSearchByIdEnded(const NETLIBHTTPREQUEST *response, void *arg) +{ + if (response == NULL || response->resultCode != HTTP_STATUS_OK) + { + ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_FAILED, (HANDLE)STEAM_SEARCH_BYID, 0); + debugLogA("CSteamProto::OnSearchByIdEnded: failed to get summaries for %s", ptrA((char*)arg)); + return; + } + + JSONROOT root(response->pData); + if (root == NULL) + return; + + JSONNODE *node = json_get(root, "players"); + JSONNODE *nroot = json_at(json_as_array(node), 0); + if (nroot != NULL) + { + STEAM_SEARCH_RESULT ssr = { 0 }; + ssr.hdr.cbSize = sizeof(STEAM_SEARCH_RESULT); + ssr.hdr.flags = PSR_TCHAR; + + ssr.hdr.id = (wchar_t*)arg; + + node = json_get(nroot, "personaname"); + ssr.hdr.nick = mir_wstrdup(json_as_string(node)); + + node = json_get(nroot, "realname"); + if (node != NULL) + { + std::wstring realname = json_as_string(node); + if (!realname.empty()) + { + size_t pos = realname.find(' ', 1); + if (pos != std::string::npos) + { + ssr.hdr.firstName = mir_wstrdup(realname.substr(0, pos).c_str()); + ssr.hdr.lastName = mir_wstrdup(realname.substr(pos + 1).c_str()); + } + else + ssr.hdr.firstName = mir_wstrdup(realname.c_str()); + } + } + + //ssr.contact = contact; + ssr.data = json_copy(nroot); + + ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, (HANDLE)STEAM_SEARCH_BYID, (LPARAM)&ssr); + ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, (HANDLE)STEAM_SEARCH_BYID, 0); + } +} + +void CSteamProto::OnSearchByNameStarted(const NETLIBHTTPREQUEST *response, void *arg) +{ +} + +//void CSteamProto::SearchByNameThread(void* arg) +//{ +// ptrW keywordsW((wchar_t*)arg); +// ptrA keywords(mir_utf8encodeW(keywordsW)); +// +// ptrA token(getStringA("TokenSecret")); +// +// SteamWebApi::SearchApi::SearchResult searchResult; +// debugLogA("CSteamProto::SearchByNameThread: call SteamWebApi::SearchApi::Search"); +// SteamWebApi::SearchApi::Search(m_hNetlibUser, token, keywords, &searchResult); +// +// if (!searchResult.IsSuccess()) +// { +// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_FAILED, (HANDLE)STEAM_SEARCH_BYNAME, 0); +// return; +// } +// +// CMStringA steamIds; +// for (int i = 0; i < searchResult.GetItemCount(); i++) +// { +// const SteamWebApi::SearchApi::SearchItem *item = searchResult.GetAt(i); +// if (steamIds.IsEmpty()) +// steamIds.Append(item->GetSteamId()); +// else +// steamIds.AppendFormat(",%s", item->GetSteamId()); +// } +// +// SteamWebApi::FriendApi::Summaries summarues; +// debugLogA("CSteamProto::SearchByNameThread: call SteamWebApi::FriendApi::LoadSummaries"); +// SteamWebApi::FriendApi::LoadSummaries(m_hNetlibUser, token, steamIds, &summarues); +// +// if (!summarues.IsSuccess()) +// { +// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_FAILED, (HANDLE)STEAM_SEARCH_BYNAME, 0); +// return; +// } +// +// for (size_t i = 0; i < summarues.GetItemCount(); i++) +// { +// const SteamWebApi::FriendApi::Summary *contact = summarues.GetAt(i); +// +// STEAM_SEARCH_RESULT ssr = { 0 }; +// ssr.hdr.cbSize = sizeof(STEAM_SEARCH_RESULT); +// ssr.hdr.flags = PSR_TCHAR; +// +// ssr.hdr.id = mir_a2u(contact->GetSteamId()); +// ssr.hdr.nick = mir_wstrdup(contact->GetNickName()); +// ssr.hdr.firstName = mir_wstrdup(contact->GetFirstName()); +// ssr.hdr.lastName = mir_wstrdup(contact->GetLastName()); +// +// ssr.contact = contact; +// +// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, (HANDLE)STEAM_SEARCH_BYNAME, (LPARAM)&ssr); +// } +// +// ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, (HANDLE)STEAM_SEARCH_BYNAME, 0); //} \ No newline at end of file -- cgit v1.2.3