/* Facebook plugin for Miranda Instant Messenger _____________________________________________ Copyright � 2009-11 Michal Zelinka, 2011-13 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 . */ #include "common.h" void updateStringUtf(FacebookProto *proto, MCONTACT hContact, const char *key, std::string value) { bool update_required = true; DBVARIANT dbv; if (!proto->getStringUtf(hContact, key, &dbv)) { update_required = strcmp(dbv.pszVal, value.c_str()) != 0; db_free(&dbv); } if (update_required) { proto->setStringUtf(hContact, key, value.c_str()); } } void FacebookProto::SaveName(MCONTACT hContact, const facebook_user *fbu) { // Save nick std::string nick = fbu->real_name; if (!getBool(FACEBOOK_KEY_NAME_AS_NICK, 1) && !fbu->nick.empty()) nick = fbu->nick; updateStringUtf(this, hContact, FACEBOOK_KEY_NICK, nick); // Explode whole name into first, second and last name std::vector names; utils::text::explode(fbu->real_name, " ", &names); updateStringUtf(this, hContact, FACEBOOK_KEY_FIRST_NAME, names.size() > 0 ? names.front().c_str() : ""); updateStringUtf(this, hContact, FACEBOOK_KEY_LAST_NAME, names.size() > 1 ? names.back().c_str() : ""); std::string middle = ""; if (names.size() > 2) { for (std::string::size_type i = 1; i < names.size() - 1; i++) { if (!middle.empty()) middle += " "; middle += names.at(i); } } updateStringUtf(this, hContact, FACEBOOK_KEY_SECOND_NAME, middle); } bool FacebookProto::IsMyContact(MCONTACT hContact, bool include_chat) { const char *proto = GetContactProto(hContact); if (proto && !strcmp(m_szModuleName, proto)) { if (include_chat) return true; return !isChatRoom(hContact); } return false; } MCONTACT FacebookProto::ChatIDToHContact(std::tstring chat_id) { for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) { if (!IsMyContact(hContact, true)) continue; ptrT id(getTStringA(hContact, "ChatRoomID")); if (id && !_tcscmp(id, chat_id.c_str())) return hContact; } return 0; } MCONTACT FacebookProto::ContactIDToHContact(std::string user_id) { for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) { if (!IsMyContact(hContact)) continue; ptrA id(getStringA(hContact, FACEBOOK_KEY_ID)); if (id && !strcmp(id, user_id.c_str())) return hContact; } return 0; } std::string FacebookProto::ThreadIDToContactID(std::string thread_id) { std::string user_id; for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) { if (!IsMyContact(hContact)) continue; ptrA tid(getStringA(hContact, FACEBOOK_KEY_TID)); if (tid && !strcmp(tid, thread_id.c_str())) { user_id = ptrA( getStringA(hContact, FACEBOOK_KEY_ID)); return user_id; } } // We don't have any contact with thish thread_id cached, we must ask server std::string data = "client=mercury"; data += "&__user=" + facy.self_.user_id; data += "&fb_dtsg=" + (facy.dtsg_.length() ? facy.dtsg_ : "0"); data += "&__a=1&__dyn=&__req=&ttstamp=0"; data += "&threads[thread_ids][0]=" + utils::url::encode(thread_id); http::response resp = facy.flap(REQUEST_THREAD_INFO, &data); if (resp.code == HTTP_CODE_OK) { CODE_BLOCK_TRY facebook_json_parser* p = new facebook_json_parser(this); p->parse_thread_info(&resp.data, &user_id); delete p; debugLogA("***** Thread info processed"); CODE_BLOCK_CATCH debugLogA("***** Error processing thread info: %s", e.what()); CODE_BLOCK_END } return user_id; } void FacebookProto::LoadContactInfo(facebook_user* fbu) { // TODO: support for more friends at once std::string get_query = "&ids[0]=" + utils::url::encode(fbu->user_id); http::response resp = facy.flap(REQUEST_LOAD_FRIEND, NULL, &get_query); if (resp.code == HTTP_CODE_OK) { CODE_BLOCK_TRY facebook_json_parser* p = new facebook_json_parser(this); p->parse_user_info(&resp.data, fbu); delete p; debugLogA("***** Thread info processed"); CODE_BLOCK_CATCH debugLogA("***** Error processing thread info: %s", e.what()); CODE_BLOCK_END } } MCONTACT FacebookProto::AddToContactList(facebook_user* fbu, ContactType type, bool force_save) { MCONTACT hContact; // First, check if this contact exists hContact = ContactIDToHContact(fbu->user_id); // If we have contact and don't need to force save data, just return it if (hContact && !force_save) return hContact; force_save = !hContact; // If not, try to make a new contact if (!hContact) { hContact = (MCONTACT)CallService(MS_DB_CONTACT_ADD, 0, 0); if (hContact && CallService(MS_PROTO_ADDTOCONTACT, hContact, (LPARAM)m_szModuleName) != 0) { CallService(MS_DB_CONTACT_DELETE, hContact, 0); hContact = NULL; } } // If we have some contact, we'll save its data if (hContact) { if (force_save) { // Save these values only when adding new contact, not when updating existing setString(hContact, FACEBOOK_KEY_ID, fbu->user_id.c_str()); std::string homepage = FACEBOOK_URL_PROFILE + fbu->user_id; setString(hContact, "Homepage", homepage.c_str()); setTString(hContact, "MirVer", fbu->getMirVer()); db_unset(hContact, "CList", "MyHandle"); ptrT group(getTStringA(NULL, FACEBOOK_KEY_DEF_GROUP)); if (group) db_set_ts(hContact, "CList", "Group", group); setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, type); if (getByte(FACEBOOK_KEY_DISABLE_STATUS_NOTIFY, 0)) CallService(MS_IGNORE_IGNORE, hContact, (LPARAM)IGNOREEVENT_USERONLINE); } if (!fbu->real_name.empty()) SaveName(hContact, fbu); if (fbu->gender) setByte(hContact, "Gender", fbu->gender); if (!fbu->image_url.empty()) setString(hContact, FACEBOOK_KEY_AV_URL, fbu->image_url.c_str()); } return hContact; } void FacebookProto::SetAllContactStatuses(int status) { for (MCONTACT hContact = db_find_first(m_szModuleName); hContact; hContact = db_find_next(hContact, m_szModuleName)) { if (isChatRoom(hContact)) continue; if (getWord(hContact, "Status", 0) != status) setWord(hContact, "Status", status); } } void FacebookProto::DeleteContactFromServer(void *data) { facy.handle_entry("DeleteContactFromServer"); if (data == NULL) return; std::string id = (*(std::string*)data); delete data; std::string query = "norefresh=true&unref=button_dropdown&confirmed=1&phstamp=0&__a=1"; query += "&fb_dtsg=" + facy.dtsg_; query += "&uid=" + id; query += "&__user=" + facy.self_.user_id; std::string get_query = "norefresh=true&unref=button_dropdown&uid=" + id; // Get unread inbox threads http::response resp = facy.flap(REQUEST_DELETE_FRIEND, &query, &get_query); if (resp.data.find("\"payload\":null", 0) != std::string::npos) { facebook_user* fbu = facy.buddies.find(id); if (fbu != NULL) fbu->deleted = true; MCONTACT hContact = ContactIDToHContact(id); // If contact wasn't deleted from database if (hContact != NULL) { setWord(hContact, "Status", ID_STATUS_OFFLINE); setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_NONE); setDword(hContact, FACEBOOK_KEY_DELETED, ::time(NULL)); } NotifyEvent(m_tszUserName, TranslateT("Contact was removed from your server list."), NULL, FACEBOOK_EVENT_OTHER); } else { facy.client_notify(TranslateT("Error occurred when removing contact from server.")); } if (resp.code != HTTP_CODE_OK) facy.handle_error("DeleteContactFromServer"); } void FacebookProto::AddContactToServer(void *data) { facy.handle_entry("AddContactToServer"); if (data == NULL) return; std::string id = (*(std::string*)data); delete data; std::string query = "action=add_friend&how_found=profile_button&ref_param=ts&outgoing_id=&unwanted=&logging_location=&no_flyout_on_click=false&ego_log_data=&lsd="; query += "&fb_dtsg=" + facy.dtsg_; query += "&to_friend=" + id; query += "&__user=" + facy.self_.user_id; // Get unread inbox threads http::response resp = facy.flap(REQUEST_REQUEST_FRIEND, &query); if (resp.data.find("\"success\":true", 0) != std::string::npos) { MCONTACT hContact = ContactIDToHContact(id); // If contact wasn't deleted from database if (hContact != NULL) setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_REQUEST); NotifyEvent(m_tszUserName, TranslateT("Request for friendship was sent."), NULL, FACEBOOK_EVENT_OTHER); } else facy.client_notify(TranslateT("Error occurred when requesting friendship.")); if (resp.code != HTTP_CODE_OK) facy.handle_error("AddContactToServer"); } void FacebookProto::ApproveContactToServer(void *data) { facy.handle_entry("ApproveContactToServer"); if (data == NULL) return; MCONTACT hContact = *(MCONTACT*)data; delete data; std::string post_data = "fb_dtsg=" + facy.dtsg_; post_data += "&charset_test=%e2%82%ac%2c%c2%b4%2c%e2%82%ac%2c%c2%b4%2c%e6%b0%b4%2c%d0%94%2c%d0%84&confirm_button="; std::string get_data = "id="; ptrA id(getStringA(hContact, FACEBOOK_KEY_ID)); get_data += id; http::response resp = facy.flap(REQUEST_APPROVE_FRIEND, &post_data, &get_data); setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_FRIEND); } void FacebookProto::CancelFriendsRequest(void *data) { facy.handle_entry("CancelFriendsRequest"); if (data == NULL) return; MCONTACT hContact = *(MCONTACT*)data; delete data; std::string query = "phstamp=0&confirmed=1"; query += "&fb_dtsg=" + facy.dtsg_; query += "&__user=" + facy.self_.user_id; ptrA id(getStringA(hContact, FACEBOOK_KEY_ID)); query += "&friend=" + std::string(id); // Get unread inbox threads http::response resp = facy.flap(REQUEST_CANCEL_REQUEST, &query); if (resp.data.find("\"payload\":null", 0) != std::string::npos) { setByte(hContact, FACEBOOK_KEY_CONTACT_TYPE, CONTACT_NONE); NotifyEvent(m_tszUserName, TranslateT("Request for friendship was canceled."), NULL, FACEBOOK_EVENT_OTHER); } else facy.client_notify(TranslateT("Error occurred when canceling friendship request.")); if (resp.code != HTTP_CODE_OK) facy.handle_error("CancelFriendsRequest"); } void FacebookProto::SendPokeWorker(void *p) { facy.handle_entry("SendPokeWorker"); if (p == NULL) return; std::string id = (*(std::string*)p); delete p; std::string data = "poke_target=" + id; data += "&do_confirm=0&phstamp=0"; data += "&fb_dtsg=" + (facy.dtsg_.length() ? facy.dtsg_ : "0"); data += "&__user=" + facy.self_.user_id; // Send poke http::response resp = facy.flap(REQUEST_POKE, &data); if (resp.data.find("\"payload\":null", 0) != std::string::npos) { resp.data = utils::text::slashu_to_utf8( utils::text::source_get_value(&resp.data, 2, "__html\":\"", "\"}")); std::string message = utils::text::source_get_value(&resp.data, 4, "", "<\\/div>"); if (message.empty()) // message has different format, show whole message message = resp.data; message = utils::text::special_expressions_decode( utils::text::remove_html(message)); ptrT tmessage( mir_utf8decodeT(message.c_str())); NotifyEvent(m_tszUserName, tmessage, NULL, FACEBOOK_EVENT_OTHER); } facy.handle_success("SendPokeWorker"); } HANDLE FacebookProto::GetAwayMsg(MCONTACT hContact) { return 0; // Status messages are disabled } int FacebookProto::OnContactDeleted(WPARAM wParam,LPARAM) { CancelFriendship(wParam, 1); return 0; }