/* Facebook plugin for Miranda Instant Messenger _____________________________________________ Copyright � 2009-11 Michal Zelinka, 2011-15 Robert P�sel 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, see <http://www.gnu.org/licenses/>. */ #include "stdafx.h" #define MAX_NEWSFEED_LEN 500 /** * Helper function for loading name from database (or use default one specified as parameter), used for title of few notifications. */ std::string getContactName(FacebookProto *proto, MCONTACT hContact, const char *defaultName) { std::string name = defaultName; DBVARIANT dbv; if (!proto->getStringUtf(hContact, FACEBOOK_KEY_NICK, &dbv)) { name = dbv.pszVal; db_free(&dbv); } return name; } void FacebookProto::ProcessBuddyList(void*) { ScopedLock s(facy.buddies_lock_); if (isOffline()) return; facy.handle_entry("ProcessBuddyList"); // Prepare update data std::string post_data = "user=" + facy.self_.user_id + "&fetch_mobile=true&fb_dtsg=" + facy.dtsg_ + "&__user=" + facy.self_.user_id + "&cached_user_info_ids="; int counter = 0; for (List::Item< facebook_user >* i = facy.buddies.begin(); i != NULL; i = i->next, counter++) { post_data += i->data->user_id + "%2C"; } post_data += "&phstamp=" + facy.phstamp(post_data); // Get buddy list http::response resp = facy.flap(REQUEST_BUDDY_LIST, &post_data); if (resp.code != HTTP_CODE_OK) { facy.handle_error("buddy_list"); return; } debugLogA("*** Starting processing buddy list"); CODE_BLOCK_TRY facebook_json_parser* p = new facebook_json_parser(this); p->parse_buddy_list(&resp.data, &facy.buddies); delete p; for (List::Item< facebook_user >* i = facy.buddies.begin(); i != NULL;) { facebook_user* fbu = i->data; if (!fbu->deleted) { if (!fbu->handle) // just been added fbu->handle = AddToContactList(fbu, CONTACT_FRIEND); ptrT client(getTStringA(fbu->handle, "MirVer")); if (!client || mir_tstrcmp(client, fbu->getMirVer())) setTString(fbu->handle, "MirVer", fbu->getMirVer()); if (getDword(fbu->handle, "IdleTS", 0) != fbu->last_active) { if ((fbu->idle || fbu->status_id == ID_STATUS_OFFLINE) && fbu->last_active > 0) setDword(fbu->handle, "IdleTS", fbu->last_active); else delSetting(fbu->handle, "IdleTS"); } } if (fbu->status_id == ID_STATUS_OFFLINE || fbu->deleted) { if (fbu->handle) setWord(fbu->handle, "Status", ID_STATUS_OFFLINE); std::string to_delete(i->key); i = i->next; facy.buddies.erase(to_delete); } else { i = i->next; if (!fbu->handle) // just been added fbu->handle = AddToContactList(fbu, CONTACT_FRIEND); if (getWord(fbu->handle, "Status", 0) != (int)fbu->status_id) setWord(fbu->handle, "Status", fbu->status_id); if (getDword(fbu->handle, "LastActiveTS", 0) != fbu->last_active) { if (fbu->last_active > 0) setDword(fbu->handle, "LastActiveTS", fbu->last_active); else delSetting(fbu->handle, "LastActiveTS"); } if (getByte(fbu->handle, FACEBOOK_KEY_CONTACT_TYPE, 0) != CONTACT_FRIEND) { setByte(fbu->handle, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_FRIEND); // TODO: remove that popup and use "Contact added you" event? } // Wasn't contact removed from "server-list" someday? if (getDword(fbu->handle, FACEBOOK_KEY_DELETED, 0)) { delSetting(fbu->handle, FACEBOOK_KEY_DELETED); std::string url = FACEBOOK_URL_PROFILE + fbu->user_id; std::string contactname = getContactName(this, fbu->handle, !fbu->real_name.empty() ? fbu->real_name.c_str() : fbu->user_id.c_str()); ptrT szTitle(mir_utf8decodeT(contactname.c_str())); NotifyEvent(szTitle, TranslateT("Contact is back on server-list."), fbu->handle, FACEBOOK_EVENT_FRIENDSHIP, &url); } // Check avatar change CheckAvatarChange(fbu->handle, fbu->image_url); } } debugLogA("*** Buddy list processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing buddy list: %s", e.what()); CODE_BLOCK_END } void FacebookProto::ProcessFriendList(void*) { ScopedLock s(facy.buddies_lock_); if (isOffline()) return; facy.handle_entry("load_friends"); // Get buddy list http::response resp = facy.flap(REQUEST_USER_INFO_ALL); if (resp.code != HTTP_CODE_OK) { facy.handle_error("load_friends"); return; } debugLogA("*** Starting processing friend list"); CODE_BLOCK_TRY std::map<std::string, facebook_user*> friends; facebook_json_parser* p = new facebook_json_parser(this); p->parse_friends(&resp.data, &friends); delete p; // Check and update old contacts for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) { if (isChatRoom(hContact)) continue; // TODO RM: change name of "Deleted" key to "DeletedTS", remove this code in some next version int deletedTS = getDword(hContact, "Deleted", 0); if (deletedTS != 0) { delSetting(hContact, "Deleted"); setDword(hContact, FACEBOOK_KEY_DELETED, deletedTS); } facebook_user *fbu; ptrA id(getStringA(hContact, FACEBOOK_KEY_ID)); if (id != NULL) { std::map< std::string, facebook_user* >::iterator iter; if ((iter = friends.find(std::string(id))) != friends.end()) { // Found contact, update it and remove from map fbu = iter->second; // TODO RM: remove, because contacts cant change it, so its only for "first run" // - but what with contacts, that was added after logon? // Update gender if (getByte(hContact, "Gender", 0) != (int)fbu->gender) setByte(hContact, "Gender", fbu->gender); // TODO: remove this in some future version? // Remove old useless "RealName" field ptrA realname(getStringA(hContact, "RealName")); if (realname != NULL) { delSetting(hContact, "RealName"); } // Update real name and nick if (!fbu->real_name.empty()) { SaveName(hContact, fbu); } // Update username ptrA username(getStringA(hContact, FACEBOOK_KEY_USERNAME)); if (!username || mir_strcmp(username, fbu->username.c_str())) { if (!fbu->username.empty()) setString(hContact, FACEBOOK_KEY_USERNAME, fbu->username.c_str()); else delSetting(hContact, FACEBOOK_KEY_USERNAME); } // Update contact type if (getByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, 0) != CONTACT_FRIEND) { setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_FRIEND); // TODO: remove that popup and use "Contact added you" event? } // Wasn't contact removed from "server-list" someday? if (getDword(hContact, FACEBOOK_KEY_DELETED, 0)) { delSetting(hContact, FACEBOOK_KEY_DELETED); std::string url = FACEBOOK_URL_PROFILE + fbu->user_id; std::string contactname = getContactName(this, hContact, !fbu->real_name.empty() ? fbu->real_name.c_str() : fbu->user_id.c_str()); ptrT szTitle(mir_utf8decodeT(contactname.c_str())); NotifyEvent(szTitle, TranslateT("Contact is back on server-list."), hContact, FACEBOOK_EVENT_FRIENDSHIP, &url); } // Check avatar change CheckAvatarChange(hContact, fbu->image_url); // Mark this contact as deleted ("processed") and delete them later (as there may be some duplicit contacts to use) fbu->deleted = true; } else { // Contact was removed from "server-list", notify it // Wasnt we already been notified about this contact? And was this real friend? if (!getDword(hContact, FACEBOOK_KEY_DELETED, 0) && getByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, 0) == CONTACT_FRIEND) { setDword(hContact, FACEBOOK_KEY_DELETED, ::time(NULL)); setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_NONE); std::string url = FACEBOOK_URL_PROFILE + std::string(id); std::string contactname = getContactName(this, hContact, id); ptrT szTitle(mir_utf8decodeT(contactname.c_str())); NotifyEvent(szTitle, TranslateT("Contact is no longer on server-list."), hContact, FACEBOOK_EVENT_FRIENDSHIP, &url); } } } } // Check remaining contacts in map and add them to contact list for (std::map< std::string, facebook_user* >::iterator it = friends.begin(); it != friends.end();) { if (!it->second->deleted) AddToContactList(it->second, CONTACT_FRIEND, true); // we know this contact doesn't exists, so we force add it delete it->second; it = friends.erase(it); } friends.clear(); debugLogA("*** Friend list processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing friend list: %s", e.what()); CODE_BLOCK_END } void FacebookProto::ProcessUnreadMessages(void*) { if (isOffline()) return; facy.handle_entry("ProcessUnreadMessages"); // receive messages from all folders by default, use hidden setting to receive only inbox messages bool inboxOnly = getBool(FACEBOOK_KEY_INBOX_ONLY, 0); std::string data = "folders[0]=inbox"; if (!inboxOnly) data += "&folders[1]=other"; data += "&client=mercury"; data += "__user=" + facy.self_.user_id; data += "&fb_dtsg=" + facy.dtsg_; data += "&__a=1&__dyn=&__req=&ttstamp=" + facy.ttstamp(); http::response resp = facy.flap(REQUEST_UNREAD_THREADS, &data); if (resp.code != HTTP_CODE_OK) { facy.handle_error("ProcessUnreadMessages"); return; } CODE_BLOCK_TRY std::vector<std::string> threads; facebook_json_parser* p = new facebook_json_parser(this); p->parse_unread_threads(&resp.data, &threads, inboxOnly); delete p; ForkThread(&FacebookProto::ProcessUnreadMessage, new std::vector<std::string>(threads)); debugLogA("*** Unread threads list processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing unread threads list: %s", e.what()); CODE_BLOCK_END facy.handle_success("ProcessUnreadMessages"); } void FacebookProto::ProcessUnreadMessage(void *data) { if (data == NULL) return; std::vector<std::string> *threads = (std::vector<std::string>*)data; if (isOffline()) { delete threads; return; } facy.handle_entry("ProcessUnreadMessage"); int offset = 0; int limit = 21; // receive messages from all folders by default, use hidden setting to receive only inbox messages bool inboxOnly = getBool(FACEBOOK_KEY_INBOX_ONLY, 0); http::response resp; while (!threads->empty()) { std::string data = "client=mercury"; data += "&__user=" + facy.self_.user_id; data += "&fb_dtsg=" + facy.dtsg_; data += "&__a=1&__dyn=&__req=&ttstamp=" + facy.ttstamp(); for (std::vector<std::string>::size_type i = 0; i < threads->size(); i++) { std::string thread_id = utils::url::encode(threads->at(i)); // request messages from thread data += "&messages[thread_ids][" + thread_id; data += "][offset]=" + utils::conversion::to_string(&offset, UTILS_CONV_SIGNED_NUMBER); data += "&messages[thread_ids][" + thread_id; data += "][limit]=" + utils::conversion::to_string(&limit, UTILS_CONV_SIGNED_NUMBER); // request info about thread data += "&threads[thread_ids][" + utils::conversion::to_string(&i, UTILS_CONV_UNSIGNED_NUMBER); data += "]=" + thread_id; } resp = facy.flap(REQUEST_THREAD_INFO, &data); if (resp.code == HTTP_CODE_OK) { CODE_BLOCK_TRY std::vector<facebook_message*> messages; std::map<std::string, facebook_chatroom*> chatrooms; facebook_json_parser* p = new facebook_json_parser(this); p->parse_thread_messages(&resp.data, &messages, &chatrooms, false, inboxOnly); delete p; for (std::map<std::string, facebook_chatroom*>::iterator it = chatrooms.begin(); it != chatrooms.end();) { // TODO: refactor this too! // TODO: have all chatrooms in facy, in memory, and then handle them as needed... somehow think about it... /* facebook_chatroom *room = it->second; MCONTACT hChatContact = NULL; ptrA users(GetChatUsers(room->thread_id.c_str())); if (users == NULL) { AddChat(room->thread_id.c_str(), room->chat_name.c_str()); hChatContact = ChatIDToHContact(room->thread_id); // Set thread id (TID) for later setTString(hChatContact, FACEBOOK_KEY_TID, room->thread_id.c_str()); for (std::map<std::string, std::string>::iterator jt = room->participants.begin(); jt != room->participants.end(); ) { AddChatContact(room->thread_id.c_str(), jt->first.c_str(), jt->second.c_str()); ++jt; } } if (!hChatContact) hChatContact = ChatIDToHContact(room->thread_id); ForkThread(&FacebookProto::ReadMessageWorker, (void*)hChatContact);*/ delete it->second; it = chatrooms.erase(it); } chatrooms.clear(); ReceiveMessages(messages, true); debugLogA("*** Unread messages processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing unread messages: %s", e.what()); CODE_BLOCK_END facy.handle_success("ProcessUnreadMessage"); } else { facy.handle_error("ProcessUnreadMessage"); } offset += limit; limit = 20; // TODO: use better limits? threads->clear(); // TODO: if we have limit messages from one user, there may be more unread messages... continue with it... otherwise remove that threadd from threads list -- or do it in json parser? hm = allow more than "limit" unread messages to be parsed } delete threads; } void FacebookProto::LoadLastMessages(void *p) { if (p == NULL) return; if (isOffline()) { delete (MCONTACT*)p; return; } facy.handle_entry("LoadLastMessages"); MCONTACT hContact = *(MCONTACT*)p; delete (MCONTACT*)p; if (!isOnline()) return; std::string data = "client=mercury"; data += "&__user=" + facy.self_.user_id; data += "&fb_dtsg=" + facy.dtsg_; data += "&__a=1&__dyn=&__req=&ttstamp=" + facy.ttstamp(); bool isChat = isChatRoom(hContact); if (isChat && (!m_enableChat || IsSpecialChatRoom(hContact))) // disabled chats or special chatroom (e.g. nofitications) return; ptrA item_id(getStringA(hContact, isChat ? FACEBOOK_KEY_TID : FACEBOOK_KEY_ID)); if (item_id == NULL) { debugLogA("!!! LoadLastMessages(): Contact has no TID/ID"); return; } std::string id = utils::url::encode(std::string(item_id)); std::string type = isChat ? "thread_ids" : "user_ids"; int count = getByte(FACEBOOK_KEY_MESSAGES_ON_OPEN_COUNT, DEFAULT_MESSAGES_ON_OPEN_COUNT); count = min(count, FACEBOOK_MESSAGES_ON_OPEN_LIMIT); // request messages from thread data += "&messages[" + type + "][" + id; data += "][offset]=0"; data += "&messages[" + type + "][" + id; data += "][limit]=" + utils::conversion::to_string(&count, UTILS_CONV_UNSIGNED_NUMBER); // request info about thread data += "&threads[" + type + "][0]=" + id; http::response resp = facy.flap(REQUEST_THREAD_INFO, &data); if (resp.code != HTTP_CODE_OK || resp.data.empty()) { facy.handle_error("LoadLastMessages"); return; } // Temporarily disable marking messages as read for this contact facy.ignore_read.insert(hContact); CODE_BLOCK_TRY std::vector<facebook_message*> messages; std::map<std::string, facebook_chatroom*> chatrooms; facebook_json_parser* p = new facebook_json_parser(this); p->parse_thread_messages(&resp.data, &messages, &chatrooms, false, false); delete p; // TODO: do something with this, chat is loading somewhere else... (in receiveMessages method right now) /*for (std::map<std::string, facebook_chatroom*>::iterator it = chatrooms.begin(); it != chatrooms.end();) { facebook_chatroom *room = it->second; MCONTACT hChatContact = NULL; ptrA users(GetChatUsers(room->thread_id.c_str())); if (users == NULL) { AddChat(room->thread_id.c_str(), room->chat_name.c_str()); hChatContact = ChatIDToHContact(room->thread_id); // Set thread id (TID) for later setTString(hChatContact, FACEBOOK_KEY_TID, room->thread_id.c_str()); for (std::map<std::string, std::string>::iterator jt = room->participants.begin(); jt != room->participants.end();) { AddChatContact(room->thread_id.c_str(), jt->first.c_str(), jt->second.c_str()); ++jt; } } if (!hChatContact) hChatContact = ChatIDToHContact(room->thread_id); ForkThread(&FacebookProto::ReadMessageWorker, (void*)hChatContact); delete it->second; it = chatrooms.erase(it); } chatrooms.clear();*/ ReceiveMessages(messages, true); debugLogA("*** Thread messages processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing thread messages: %s", e.what()); CODE_BLOCK_END facy.handle_success("LoadLastMessages"); // Enable marking messages as read for this contact facy.ignore_read.erase(hContact); // And force mark read OnDbEventRead(hContact, NULL); } void FacebookProto::SyncThreads(void*) { facy.handle_entry("SyncThreads"); if (isOffline()) return; // Always load unread messages because syncthreads request is not reliable (probably doesn't load multi user chat messages at all) ProcessUnreadMessages(NULL); // Get timestamp of last action (message or other event) time_t timestamp = getDword(FACEBOOK_KEY_LAST_ACTION_TS, 0); // If last event is older than 2 day, we force sync to be max. 2 day old time_t daysBefore = ::time(NULL) - 24 * 60 * 60 * 2; if (timestamp < daysBefore) { debugLogA(" Last action timestamp is too old: %d, use 24 hours old instead: %d", timestamp, daysBefore); timestamp = daysBefore; } // Receive messages from all folders by default, use hidden setting to receive only inbox messages bool inboxOnly = getBool(FACEBOOK_KEY_INBOX_ONLY, 0); // Get milli timestamp string for Facebook std::string time = utils::conversion::to_string((void*)×tamp, UTILS_CONV_TIME_T) + "000"; std::string data = "client=mercury"; data += "&last_action_timestamp=" + time; data += "&__user=" + facy.self_.user_id; data += "&fb_dtsg=" + facy.dtsg_; data += "&folders[0]=inbox"; if (!inboxOnly) data += "&folders[1]=other"; data += "&__req=7&__a=1&__dyn=&__req=&__rev=&ttstamp=" + facy.ttstamp(); debugLogA(" Facebook's milli timestamp for sync: %s", time.c_str()); http::response resp = facy.flap(REQUEST_THREAD_SYNC, &data); if (resp.code != HTTP_CODE_OK || resp.data.empty()) { facy.handle_error("LoadLastMessages"); return; } CODE_BLOCK_TRY std::vector<facebook_message*> messages; std::map<std::string, facebook_chatroom*> chatrooms; facebook_json_parser* p = new facebook_json_parser(this); p->parse_thread_messages(&resp.data, &messages, &chatrooms, false, false); delete p; ReceiveMessages(messages, true); debugLogA("*** Thread messages processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing thread messages: %s", e.what()); CODE_BLOCK_END facy.handle_success("SyncThreads"); } void parseFeeds(const std::string &text, std::vector<facebook_newsfeed *> &news, DWORD &last_post_time, bool filterAds = true) { std::string::size_type pos = 0; UINT limit = 0; DWORD new_time = last_post_time; while ((pos = text.find("<div class=\"userContentWrapper", pos)) != std::string::npos && limit <= 25) { /*std::string::size_type pos2 = text.find("<div class=\"userContentWrapper", pos+5); if (pos2 == std::string::npos) pos2 = text.length(); std::string post = text.substr(pos, pos2 - pos);*/ std::string post = text.substr(pos, text.find("</form>", pos) - pos); pos += 5; std::string post_header = utils::text::source_get_value(&post, 3, "<h5", ">", "</h5>"); std::string post_message = utils::text::source_get_value(&post, 3, " userContent\"", ">", "<form"); std::string post_link = utils::text::source_get_value(&post, 4, "</h5>", "<a", "href=\"", "\""); //std::string post_attach = utils::text::source_get_value(&post, 4, "<div class=", "uiStreamAttachments", ">", "<form"); std::string post_time = utils::text::source_get_value(&post, 3, "</h5>", "<abbr", "</a>"); std::string time = utils::text::source_get_value(&post_time, 2, "data-utime=\"", "\""); std::string time_text = utils::text::source_get_value(&post_time, 2, ">", "</abbr>"); if (time.empty()) { // alternative parsing (probably page like or advertisement) time = utils::text::source_get_value(&post, 2, "content_timestamp":"", """); } DWORD ttime; if (!utils::conversion::from_string<DWORD>(ttime, time, std::dec)) { //debugLogA("!!! - Newsfeed with wrong/empty time (probably wrong parsing)\n%s", post.c_str()); continue; } if (ttime > new_time) { new_time = ttime; // remember newest time from all these posts //debugLogA(" - Newsfeed time: %d (new)", ttime); } else if (ttime <= last_post_time) { //debugLogA(" - Newsfeed time: %d (ignored)", ttime); continue; // ignore posts older than newest post of previous check } else { //debugLogA(" - Newsfeed time: %d (normal)", ttime); } std::string post_place = utils::text::source_get_value(&post, 4, "</abbr>", "<a", ">", "</a>"); std::string premsg = "\n" + time_text; post_place = utils::text::trim( utils::text::remove_html(post_place)); if (!post_place.empty()) { premsg += " - " + post_place; } premsg += "\n"; // in title keep only name, end of events like "X shared link" put into message std::string::size_type pos2 = post_header.find("?"); if (pos2 != std::string::npos) { utils::text::replace_first(&post_header, "?", " � "); } else { pos2 = post_header.find("</a></"); if (pos2 != std::string::npos) { pos2 += 4; std::string a = utils::text::trim(utils::text::remove_html(post_header.substr(pos2, post_header.length() - pos2))); if (a.length() > 2) premsg += a; post_header = post_header.substr(0, pos2); } } // Strip "Translate" link pos2 = post_message.find("role=\"button\">"); if (pos2 != std::string::npos) { post_message = post_message.substr(0, pos2 + 14); } post_message = premsg + post_message; // append attachement to message (if any) //post_message += utils::text::trim(post_attach); facebook_newsfeed* nf = new facebook_newsfeed; nf->title = utils::text::trim( utils::text::html_entities_decode( utils::text::remove_html(post_header))); nf->user_id = utils::text::source_get_value(&post_header, 2, "user.php?id=", "&"); nf->link = utils::text::html_entities_decode(post_link); // Check if we don't want to show ads posts bool filtered = filterAds && (nf->link.find("/about/ads") != std::string::npos || post.find("class=\"uiStreamSponsoredLink\"") != std::string::npos || post.find("href=\"/about/ads\"") != std::string::npos); nf->text = utils::text::trim( utils::text::html_entities_decode( utils::text::remove_html( utils::text::edit_html(post_message)))); if (filtered || nf->title.empty() || nf->text.empty()) { //debugLogA(" \\ Newsfeed (time: %d) is filtered: %s", ttime, filtered ? "advertisement" : (nf->title.empty() ? "title empty" : "text empty")); delete nf; continue; } else { //debugLogA(" Got newsfeed (time: %d)", ttime); } news.push_back(nf); pos++; limit++; } last_post_time = new_time; } void FacebookProto::ProcessOnThisDay(void*) { if (isOffline() || !getBool(FACEBOOK_KEY_EVENT_ON_THIS_DAY_ENABLE, DEFAULT_EVENT_ON_THIS_DAY_ENABLE)) return; facy.handle_entry(__FUNCTION__); time_t timestamp = ::time(NULL); std::string get_data = "&start_index=0&num_stories=20&last_section_header=0"; get_data += "×tamp=" + utils::conversion::to_string((void*)×tamp, UTILS_CONV_TIME_T); get_data += "&__dyn=&__req=&__rev=&__user=" + facy.self_.user_id; http::response resp = facy.flap(REQUEST_ON_THIS_DAY, NULL, &get_data); if (resp.code != HTTP_CODE_OK) { facy.handle_error(__FUNCTION__); return; } std::string jsonData = resp.data.substr(9); JSONNode root = JSONNode::parse(jsonData.c_str()); if (root) { const JSONNode &html_ = root["domops"].at((json_index_t)0).at((json_index_t)3).at("__html"); if (html_) { std::string html = utils::text::html_entities_decode(utils::text::slashu_to_utf8(html_.as_string())); std::vector<facebook_newsfeed *> news; DWORD new_time = 0; parseFeeds(html, news, new_time, true); debugLogA(" Last feeds update (new): %d", new_time); for (std::vector<facebook_newsfeed*>::size_type i = 0; i < news.size(); i++) { // Truncate text of newsfeed when it's too long std::tstring text = ptrT(mir_utf8decodeT(news[i]->text.c_str())); if (text.length() > MAX_NEWSFEED_LEN) text = text.substr(0, MAX_NEWSFEED_LEN) + _T("..."); ptrT tszTitle(mir_utf8decodeT(news[i]->title.c_str())); ptrT tszText(mir_tstrdup(text.c_str())); NotifyEvent(TranslateT("On this day"), tszText, NULL, FACEBOOK_EVENT_ON_THIS_DAY, &news[i]->link); delete news[i]; } news.clear(); } } facy.handle_success(__FUNCTION__); } void FacebookProto::ReceiveMessages(std::vector<facebook_message*> messages, bool check_duplicates) { bool naseemsSpamMode = getBool(FACEBOOK_KEY_NASEEMS_SPAM_MODE, false); // TODO: make this checking more lightweight as now it is not effective at all... if (check_duplicates) { // 1. check if there are some message that we already have (compare FACEBOOK_KEY_MESSAGE_ID = last received message ID) for (std::vector<facebook_message*>::size_type i = 0; i < messages.size(); i++) { MCONTACT hContact = messages[i]->isChat ? ChatIDToHContact(messages[i]->thread_id) : ContactIDToHContact(messages[i]->user_id); if (hContact == NULL) continue; ptrA lastId(getStringA(hContact, FACEBOOK_KEY_MESSAGE_ID)); if (lastId == NULL) continue; if (!messages[i]->message_id.compare(lastId)) { // Equal, ignore all older messages (including this) from same contact for (std::vector<facebook_message*>::size_type j = 0; j < messages.size(); j++) { bool equalsId = messages[i]->isChat ? (messages[j]->thread_id == messages[i]->thread_id) : (messages[j]->user_id == messages[i]->user_id); if (equalsId && messages[j]->time <= messages[i]->time) messages[j]->flag_ = 1; } } } // 2. remove all marked messages from list for (std::vector<facebook_message*>::iterator it = messages.begin(); it != messages.end();) { if ((*it)->flag_ == 1) it = messages.erase(it); else ++it; } } std::set<MCONTACT> *hChatContacts = new std::set<MCONTACT>(); for (std::vector<facebook_message*>::size_type i = 0; i < messages.size(); i++) { if (messages[i]->isChat) { if (!m_enableChat) { delete messages[i]; continue; } // Multi-user message debugLogA(" < Got chat message ID: %s", messages[i]->message_id.c_str()); facebook_chatroom *fbc; std::string tthread_id = messages[i]->thread_id.c_str(); auto it = facy.chat_rooms.find(tthread_id); if (it != facy.chat_rooms.end()) { fbc = it->second; } else { // In Naseem's spam mode we ignore outgoing messages sent from other instances if (naseemsSpamMode && !messages[i]->isIncoming) { delete messages[i]; continue; } // We don't have this chat loaded in memory yet, lets load some info (name, list of users) fbc = new facebook_chatroom(tthread_id); LoadChatInfo(fbc); facy.chat_rooms.insert(std::make_pair(tthread_id, fbc)); } MCONTACT hChatContact = NULL; // RM TODO: better use check if chatroom exists/is in db/is online... no? // like: if (ChatIDToHContact(tthread_id) == NULL) { ptrA users(GetChatUsers(fbc->thread_id.c_str())); if (users == NULL) { AddChat(fbc->thread_id.c_str(), fbc->chat_name.c_str()); hChatContact = ChatIDToHContact(fbc->thread_id); // Set thread id (TID) for later setString(hChatContact, FACEBOOK_KEY_TID, fbc->thread_id.c_str()); for (std::map<std::string, std::string>::iterator jt = fbc->participants.begin(); jt != fbc->participants.end(); ++jt) { AddChatContact(fbc->thread_id.c_str(), jt->first.c_str(), jt->second.c_str()); } } if (!hChatContact) hChatContact = ChatIDToHContact(fbc->thread_id); if (!hChatContact) { // hopefully shouldn't happen, but who knows? debugLog(_T("!!! No hChatContact for %s"), fbc->thread_id.c_str()); delete messages[i]; continue; } // We don't want to save (this) message ID for chatrooms // setString(hChatContact, FACEBOOK_KEY_MESSAGE_ID, messages[i]->message_id.c_str()); setDword(FACEBOOK_KEY_LAST_ACTION_TS, messages[i]->time); // Save TID if not exists already ptrA tid(getStringA(hChatContact, FACEBOOK_KEY_TID)); if (!tid || mir_strcmp(tid, messages[i]->thread_id.c_str())) setString(hChatContact, FACEBOOK_KEY_TID, messages[i]->thread_id.c_str()); // Try to map name of this chat participant to his id std::map<std::string, std::string>::iterator jt = fbc->participants.find(messages[i]->user_id); if (jt != fbc->participants.end()) { messages[i]->sender_name = jt->second; } // TODO: support also system messages (rename chat, user quit, etc.)! (here? or it is somewhere else? // ... we must add some new "type" field into facebook_message structure and use it also for Pokes and similar) UpdateChat(fbc->thread_id.c_str(), messages[i]->user_id.c_str(), messages[i]->sender_name.c_str(), messages[i]->message_text.c_str(), messages[i]->time); // Automatically mark message as read because chatroom doesn't support onRead event (yet) hChatContacts->insert(hChatContact); // std::set checks duplicates at insert automatically } else { // Single-user message debugLogA(" < Got message ID: %s", messages[i]->message_id.c_str()); facebook_user fbu; fbu.user_id = messages[i]->user_id; fbu.real_name = messages[i]->sender_name; MCONTACT hContact = ContactIDToHContact(fbu.user_id); if (hContact == NULL) { // In Naseem's spam mode we ignore outgoing messages sent from other instances if (naseemsSpamMode && !messages[i]->isIncoming) { delete messages[i]; continue; } // We don't have this contact, lets load info about him LoadContactInfo(&fbu); hContact = AddToContactList(&fbu, CONTACT_NONE); } if (!hContact) { // hopefully shouldn't happen, but who knows? debugLogA("!!! No hContact for %s", messages[i]->user_id.c_str()); delete messages[i]; continue; } // Save last (this) message ID setString(hContact, FACEBOOK_KEY_MESSAGE_ID, messages[i]->message_id.c_str()); // Save TID if not exists already ptrA tid(getStringA(hContact, FACEBOOK_KEY_TID)); if (!tid || mir_strcmp(tid, messages[i]->thread_id.c_str())) setString(hContact, FACEBOOK_KEY_TID, messages[i]->thread_id.c_str()); if (messages[i]->isIncoming && messages[i]->isUnread && messages[i]->type == MESSAGE) { PROTORECVEVENT recv = { 0 }; recv.szMessage = const_cast<char*>(messages[i]->message_text.c_str()); recv.timestamp = messages[i]->time; ProtoChainRecvMsg(hContact, &recv); } else { DBEVENTINFO dbei = { 0 }; dbei.cbSize = sizeof(dbei); if (messages[i]->type == CALL) dbei.eventType = FACEBOOK_EVENTTYPE_CALL; else dbei.eventType = EVENTTYPE_MESSAGE; dbei.flags = DBEF_UTF; if (!messages[i]->isIncoming) dbei.flags |= DBEF_SENT; if (!messages[i]->isUnread) dbei.flags |= DBEF_READ; dbei.szModule = m_szModuleName; dbei.timestamp = messages[i]->time; dbei.cbBlob = (DWORD)messages[i]->message_text.length() + 1; dbei.pBlob = (PBYTE)messages[i]->message_text.c_str(); db_event_add(hContact, &dbei); } } delete messages[i]; } messages.clear(); if (!hChatContacts->empty()) { ForkThread(&FacebookProto::ReadMessageWorker, (void*)hChatContacts); } else { delete hChatContacts; } } void FacebookProto::ProcessMessages(void* data) { if (data == NULL) return; std::string* resp = (std::string*)data; if (isOffline()) { delete resp; return; } // receive messages from all folders by default, use hidden setting to receive only inbox messages bool inboxOnly = getBool(FACEBOOK_KEY_INBOX_ONLY, 0); debugLogA("*** Starting processing messages"); CODE_BLOCK_TRY std::vector< facebook_message* > messages; facebook_json_parser* p = new facebook_json_parser(this); p->parse_messages(resp, &messages, &facy.notifications, inboxOnly); delete p; ReceiveMessages(messages); ShowNotifications(); debugLogA("*** Messages processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing messages: %s", e.what()); CODE_BLOCK_END delete resp; } void FacebookProto::ShowNotifications() { ScopedLock s(facy.notifications_lock_); if (!getBool(FACEBOOK_KEY_EVENT_NOTIFICATIONS_ENABLE, DEFAULT_EVENT_NOTIFICATIONS_ENABLE)) return; // Show popups for unseen notifications and/or write them to chatroom for (std::map<std::string, facebook_notification*>::iterator it = facy.notifications.begin(); it != facy.notifications.end(); ++it) { facebook_notification *notification = it->second; if (notification != NULL && !notification->seen) { debugLogA(" Showing popup for notification ID: %s", notification->id.c_str()); ptrT szText(mir_utf8decodeT(notification->text.c_str())); MCONTACT hContact = (notification->user_id.empty() ? NULL : ContactIDToHContact(notification->user_id)); notification->hWndPopup = NotifyEvent(m_tszUserName, szText, hContact, FACEBOOK_EVENT_NOTIFICATION, ¬ification->link, ¬ification->id); notification->seen = true; } } } void FacebookProto::ProcessNotifications(void*) { if (isOffline()) return; facy.handle_entry("notifications"); int count = FACEBOOK_NOTIFICATIONS_LOAD_COUNT; std::string data = "__dyn=&__req=&__rev="; data += "&__user=" + facy.self_.user_id; data += "&fb_dtsg=" + facy.dtsg_; data += "&cursor="; // when loading more data += "&length=" + utils::conversion::to_string(&count, UTILS_CONV_UNSIGNED_NUMBER); // number of items to load data += "&businessID="; // probably for pages? data += "&ttstamp=" + facy.ttstamp(); // Get notifications http::response resp = facy.flap(REQUEST_NOTIFICATIONS, &data); if (resp.code != HTTP_CODE_OK) { facy.handle_error("notifications"); return; } // Process notifications debugLogA("*** Starting processing notifications"); CODE_BLOCK_TRY facebook_json_parser* p = new facebook_json_parser(this); p->parse_notifications(&(resp.data), &facy.notifications); delete p; ShowNotifications(); debugLogA("*** Notifications processed"); CODE_BLOCK_CATCH debugLogA("*** Error processing notifications: %s", e.what()); CODE_BLOCK_END } void FacebookProto::ProcessFriendRequests(void*) { if (isOffline()) return; facy.handle_entry("friendRequests"); // Get notifications http::response resp = facy.flap(REQUEST_LOAD_FRIENDSHIPS); if (resp.code != HTTP_CODE_OK) { facy.handle_error("friendRequests"); return; } // Parse it std::string reqs = utils::text::source_get_value(&resp.data, 3, "id=\"friend_requests_section\"", "</h4>", "<h4"); std::string::size_type pos = 0; std::string::size_type pos2 = 0; bool last = (reqs.find("seenrequesttime=") == std::string::npos); // false when there are some requests while (!last && !reqs.empty()) { std::string req; if ((pos2 = reqs.find("<img src=", pos)) != std::string::npos) { req = reqs.substr(pos, pos2 - pos); pos = pos2 + 9; } else { req = reqs.substr(pos); last = true; } std::string get = utils::text::source_get_value(&req, 3, "<form", "action=\"", "\">"); std::string time = utils::text::source_get_value2(&get, "seenrequesttime=", "&\""); std::string reason = utils::text::remove_html(utils::text::source_get_value(&req, 3, "<span", ">", "</span>")); facebook_user fbu; fbu.real_name = utils::text::remove_html(utils::text::source_get_value(&req, 3, "<strong", ">", "</strong>")); fbu.user_id = utils::text::source_get_value2(&get, "id=", "&\""); if (!fbu.user_id.empty() && !fbu.real_name.empty()) { MCONTACT hContact = AddToContactList(&fbu, CONTACT_APPROVE); setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_APPROVE); bool isNew = false; ptrA oldTime(getStringA(hContact, "RequestTime")); if (oldTime == NULL || mir_strcmp(oldTime, time.c_str())) { // This is new request isNew = true; setString(hContact, "RequestTime", time.c_str()); //blob is: uin(DWORD), hContact(HANDLE), nick(ASCIIZ), first(ASCIIZ), last(ASCIIZ), email(ASCIIZ), reason(ASCIIZ) //blob is: 0(DWORD), hContact(HANDLE), nick(ASCIIZ), ""(ASCIIZ), ""(ASCIIZ), email(ASCIIZ), reason(ASCIIZ) DBEVENTINFO dbei = { 0 }; dbei.cbSize = sizeof(DBEVENTINFO); dbei.szModule = m_szModuleName; dbei.timestamp = ::time(NULL); dbei.flags = DBEF_UTF; dbei.eventType = EVENTTYPE_AUTHREQUEST; dbei.cbBlob = (DWORD)(sizeof(DWORD) * 2 + fbu.real_name.length() + fbu.user_id.length() + reason.length() + 5); PBYTE pCurBlob = dbei.pBlob = (PBYTE)mir_alloc(dbei.cbBlob); *(PDWORD)pCurBlob = 0; pCurBlob += sizeof(DWORD); // UID *(PDWORD)pCurBlob = (DWORD)hContact; pCurBlob += sizeof(DWORD); // Contact Handle mir_strcpy((char*)pCurBlob, fbu.real_name.data()); pCurBlob += fbu.real_name.length() + 1; // Nickname *pCurBlob = '\0'; pCurBlob++; // First Name *pCurBlob = '\0'; pCurBlob++; // Last Name mir_strcpy((char*)pCurBlob, fbu.user_id.data()); pCurBlob += fbu.user_id.length() + 1; // E-mail (we use it for string ID) mir_strcpy((char*)pCurBlob, reason.data()); pCurBlob += reason.length() + 1; // Reason (we use it for info about common friends) db_event_add(0, &dbei); } debugLogA(" < (%s) Friendship request [%s]", (isNew ? "New" : "Old"), time.c_str()); } else { debugLogA("!!! Wrong friendship request:\n%s", req.c_str()); } } facy.handle_success("friendRequests"); } void FacebookProto::ProcessFeeds(void*) { if (!isOnline()) return; facy.handle_entry("feeds"); // Get feeds http::response resp = facy.flap(REQUEST_FEEDS); if (resp.code != HTTP_CODE_OK || resp.data.empty()) { facy.handle_error("feeds"); return; } std::vector<facebook_newsfeed *> news; DWORD new_time = facy.last_feeds_update_; bool filterAds = getBool(FACEBOOK_KEY_FILTER_ADS, DEFAULT_FILTER_ADS); parseFeeds(resp.data, news, new_time, filterAds); for (std::vector<facebook_newsfeed*>::size_type i = 0; i < news.size(); i++) { // Truncate text of newsfeed when it's too long std::tstring text = ptrT(mir_utf8decodeT(news[i]->text.c_str())); if (text.length() > MAX_NEWSFEED_LEN) text = text.substr(0, MAX_NEWSFEED_LEN) + _T("..."); ptrT tszTitle(mir_utf8decodeT(news[i]->title.c_str())); ptrT tszText(mir_tstrdup(text.c_str())); NotifyEvent(tszTitle, tszText, this->ContactIDToHContact(news[i]->user_id), FACEBOOK_EVENT_NEWSFEED, &news[i]->link); delete news[i]; } news.clear(); // Set time of last update to time of newest post this->facy.last_feeds_update_ = new_time; facy.handle_success("feeds"); } void FacebookProto::ProcessPages(void*) { if (isOffline() || !getByte(FACEBOOK_KEY_LOAD_PAGES, DEFAULT_LOAD_PAGES)) return; facy.handle_entry("load_pages"); // Get feeds http::response resp = facy.flap(REQUEST_PAGES); if (resp.code != HTTP_CODE_OK) { facy.handle_error("load_pages"); return; } std::string content = utils::text::source_get_value(&resp.data, 2, "id=\"bookmarksSeeAllSection\"", "</code>"); std::string::size_type start, end; start = content.find("<li", 0); while (start != std::string::npos) { end = content.find("<li", start + 1); if (end == std::string::npos) end = content.length(); std::string item = content.substr(start, end - start); //item = utils::text::source_get_value(&item, 2, "data-gt=", ">"); start = content.find("<li", start + 1); std::string id = utils::text::source_get_value(&item, 3, "data-gt=", "bmid":"", """); std::string title = utils::text::slashu_to_utf8(utils::text::source_get_value(&item, 3, "data-gt=", "title=\"", "\"")); std::string href = utils::text::source_get_value(&item, 3, "data-gt=", "href=\"", "\""); // Ignore pages channel if (href.find("/pages/feed") != std::string::npos) continue; if (id.empty() || title.empty()) continue; debugLogA(" Got page ID: %s", id.c_str()); facy.pages[id] = title; } facy.handle_success("load_pages"); } void FacebookProto::SearchAckThread(void *targ) { facy.handle_entry("searchAckThread"); int count = 0; std::string search = utils::url::encode(T2Utf((TCHAR *)targ).str()); std::string ssid; while (count < 50 && !isOffline()) { std::string get_data = search + "&s=" + utils::conversion::to_string(&count, UTILS_CONV_UNSIGNED_NUMBER); if (!ssid.empty()) get_data += "&ssid=" + ssid; http::response resp = facy.flap(REQUEST_SEARCH, NULL, &get_data); if (resp.code == HTTP_CODE_OK) { std::string items = utils::text::source_get_value(&resp.data, 4, "<body", "name=\"charset_test\"", "<table", "</body>"); std::string::size_type pos = 0; std::string::size_type pos2 = 0; while ((pos = items.find("<tr", pos)) != std::string::npos) { std::string item = items.substr(pos, items.find("</tr>", pos) - pos); pos++; count++; std::string id = utils::text::source_get_value2(&item, "?id=", "&\""); if (id.empty()) id = utils::text::source_get_value2(&item, "?ids=", "&\""); std::string name = utils::text::source_get_value(&item, 3, "<a", ">", "</"); std::string surname; std::string nick; std::string common = utils::text::source_get_value(&item, 4, "</a>", "<span", ">", "</span>"); if ((pos2 = name.find("<span class=\"alternate_name\">")) != std::string::npos) { nick = name.substr(pos2 + 30, name.length() - pos2 - 31); // also remove brackets around nickname name = name.substr(0, pos2 - 1); } if ((pos2 = name.find(" ")) != std::string::npos) { surname = name.substr(pos2 + 1, name.length() - pos2 - 1); name = name.substr(0, pos2); } // ignore self contact and empty ids if (id.empty() || id == facy.self_.user_id) continue; ptrT tid(mir_utf8decodeT(id.c_str())); ptrT tname(mir_utf8decodeT(utils::text::html_entities_decode(name).c_str())); ptrT tsurname(mir_utf8decodeT(utils::text::html_entities_decode(surname).c_str())); ptrT tnick(mir_utf8decodeT(utils::text::html_entities_decode(nick).c_str())); ptrT tcommon(mir_utf8decodeT(utils::text::html_entities_decode(common).c_str())); PROTOSEARCHRESULT psr = { 0 }; psr.cbSize = sizeof(psr); psr.flags = PSR_TCHAR; psr.id.t = tid; psr.nick.t = tnick; psr.firstName.t = tname; psr.lastName.t = tsurname; psr.email.t = tcommon; ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, targ, (LPARAM)&psr); } ssid = utils::text::source_get_value(&items, 3, "id=\"more_objects\"", "ssid=", "&"); if (ssid.empty()) break; // No more results } } ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, targ, 0); facy.handle_success("searchAckThread"); mir_free(targ); } void FacebookProto::SearchIdAckThread(void *targ) { facy.handle_entry("searchIdAckThread"); std::string search = utils::url::encode(T2Utf((TCHAR*)targ).str()) + "?"; if (!isOffline()) { http::response resp = facy.flap(REQUEST_USER_INFO_MOBILE, NULL, &search); if (resp.code == HTTP_CODE_FOUND && resp.headers.find("Location") != resp.headers.end()) { search = utils::text::source_get_value(&resp.headers["Location"], 2, FACEBOOK_SERVER_MOBILE"/", "_rdr", true); resp = facy.flap(REQUEST_USER_INFO_MOBILE, NULL, &search); } if (resp.code == HTTP_CODE_OK) { std::string about = utils::text::source_get_value(&resp.data, 2, "<div id=\"root\"", "</body>"); std::string id = utils::text::source_get_value2(&about, ";id=", "&\""); if (id.empty()) id = utils::text::source_get_value2(&about, "?bid=", "&\""); std::string name = utils::text::source_get_value(&about, 3, "<strong", ">", "</strong"); std::string surname; std::string::size_type pos; if ((pos = name.find(" ")) != std::string::npos) { surname = name.substr(pos + 1, name.length() - pos - 1); name = name.substr(0, pos); } // ignore self contact and empty ids if (!id.empty() && id != facy.self_.user_id){ ptrT tid(mir_utf8decodeT(id.c_str())); ptrT tname(mir_utf8decodeT(name.c_str())); ptrT tsurname(mir_utf8decodeT(surname.c_str())); PROTOSEARCHRESULT psr = { 0 }; psr.cbSize = sizeof(psr); psr.flags = PSR_TCHAR; psr.id.t = tid; psr.firstName.t = tname; psr.lastName.t = tsurname; ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, targ, (LPARAM)&psr); } } } ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, targ, 0); facy.handle_success("searchIdAckThread"); mir_free(targ); }